在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

ROS中的序列化實現

麥辣雞腿堡 ? 來源:古月居 ? 作者:古月居 ? 2023-09-14 17:26 ? 次閱讀

理解了序列化,再回到ROS。我們發現,ROS沒有采用第三方的序列化工具,而是選擇自己實現,代碼在roscpp_core項目下的roscpp_serialization中,見下圖。這個功能涉及的代碼量不是很多。

為什么ROS不使用現成的序列化工具或者庫呢?可能ROS誕生的時候(2007年),有些序列化庫可能還不存在(protobuf誕生于2008年),更有可能是ROS的創造者認為當時沒有合適的工具。

1.2.1 serialization.h

核心的函數都在serialization.h里,簡而言之,里面使用了C語言標準庫的memcpy函數把消息拷貝到流中。

下面來看一下具體的實現。

序列化功能的特點是要處理很多種數據類型,針對每種具體的類型都要實現相應的序列化函數。

為了盡量減少代碼量,ROS使用了模板的概念,所以代碼里有一堆的template。

從后往前梳理,先看Stream這個結構體吧。在C++里結構體和類基本沒什么區別,結構體里也可以定義函數。

Stream翻譯為流,流是一個計算機中的抽象概念,前面我們提到過字節流,它是什么意思呢?

在需要傳輸數據的時候,我們可以把數據想象成傳送帶上連續排列的一個個被傳送的物體,它們就是一個流。

更形象的,可以想象磁帶或者圖靈機里連續的紙帶。在文件讀寫、使用串口、網絡Socket通信等領域,流經常被使用。例如我們常用的輸入輸出流:

cout<<"helllo"; 由于使用很多,流的概念也在演變。想了解更多可以看這里。

struct Stream
{
  // Returns a pointer to the current position of the stream
  inline uint8_t* getData() { return data_; }
  // Advances the stream, checking bounds, and returns a pointer to the position before it was advanced.
  // throws StreamOverrunException if len would take this stream past the end of its buffer
  ROS_FORCE_INLINE uint8_t* advance(uint32_t len)
{
    uint8_t* old_data = data_;
    data_ += len;
    if (data_ > end_)
    {
      // Throwing directly here causes a significant speed hit due to the extra code generated for the throw statement
      throwStreamOverrun();
    }
    return old_data;
  }
  // Returns the amount of space left in the stream
  inline uint32_t getLength() { return static_cast< uint32_t >(end_ - data_); }
  
protected:
  Stream(uint8_t* _data, uint32_t _count) : data_(_data), end_(_data + _count) {}


private:
  uint8_t* data_;
  uint8_t* end_;
};

注釋表明Stream是個基類,輸入輸出流IStream和OStream都繼承自它。

Stream的成員變量data_是個指針,指向序列化的字節流開始的位置,它的類型是uint8_t。

在Ubuntu系統中,uint8_t的定義是typedef unsigned char uint8_t;

所以uint8_t就是一個字節,可以用size_of()函數檢驗。data_指向的空間就是保存字節流的。

輸出流類OStream用來序列化一個對象,它引用了serialize函數,如下。

struct OStream : public Stream
{
  static const StreamType stream_type = stream_types::Output;
  OStream(uint8_t* data, uint32_t count) : Stream(data, count) {}
  /* Serialize an item to this output stream*/
  template< typename T >
  ROS_FORCE_INLINE void next(const T& t)
{
    serialize(*this, t);
  }
  template< typename T >
  ROS_FORCE_INLINE OStream& operator< const T& t)
  {
    serialize(*this, t);
    return *this;
  }
};

輸入流類IStream用來反序列化一個字節流,它引用了deserialize函數,如下。

struct ROSCPP_SERIALIZATION_DECL IStream : public Stream
{
  static const StreamType stream_type = stream_types::Input;
  IStream(uint8_t* data, uint32_t count) : Stream(data, count) {}
  /* Deserialize an item from this input stream */
  template< typename T >
  ROS_FORCE_INLINE void next(T& t)
{
    deserialize(*this, t);
  }
  template< typename T >
  ROS_FORCE_INLINE IStream& operator >>(T& t)
  {
    deserialize(*this, t);
    return *this;
  }
};

自然,serialize函數和deserialize函數就是改變數據形式的地方,它們的定義在比較靠前的地方。它們都接收兩個模板,都是內聯函數,然后里面沒什么東西,只是又調用了Serializer類的成員函數write和read。所以,serialize和deserialize函數就是個二道販子。

// Serialize an object.  Stream here should normally be a ros::serialization::OStream
template< typename T, typename Stream >
inline void serialize(Stream& stream, const T& t)
{
  Serializer< T >::write(stream, t);
}
// Deserialize an object.  Stream here should normally be a ros::serialization::IStream
template< typename T, typename Stream >
inline void deserialize(Stream& stream, T& t)
{
  Serializer< T >::read(stream, t);
}

所以,我們來分析Serializer類,如下。我們發現,write和read函數又調用了類型里的serialize函數和deserialize函數。

頭別暈,這里的serialize和deserialize函數跟上面的同名函數不是一回事。

注釋中說:“Specializing the Serializer class is the only thing you need to do to get the ROS serialization system to work with a type”(要想讓ROS的序列化功能適用于其它的某個類型,你唯一需要做的就是特化這個Serializer類)。

這就涉及到的另一個知識點——模板特化(template specialization)。

template< typename T > struct Serializer
{
  // Write an object to the stream.  Normally the stream passed in here will be a ros::serialization::OStream
  template< typename Stream >
  inline static void write(Stream& stream, typename boost::call_traits< T >::param_type t)
{
    t.serialize(stream.getData(), 0);
  }
   // Read an object from the stream.  Normally the stream passed in here will be a ros::serialization::IStream
  template< typename Stream >
  inline static void read(Stream& stream, typename boost::call_traits< T >::reference t)
{
    t.deserialize(stream.getData());
  }
  // Determine the serialized length of an object.
  inline static uint32_t serializedLength(typename boost::call_traits< T >::param_type t)
{
    return t.serializationLength();
  }
};

接著又定義了一個帶參數的宏函數ROS_CREATE_SIMPLE_SERIALIZER(Type),然后把這個宏作用到了ROS中的10種基本數據類型,分別是:uint8_t, int8_t, uint16_t, int16_t, uint32_t, int32_t, uint64_t, int64_t, float, double。

說明這10種數據類型的處理方式都是類似的。看到這里大家應該明白了,write和read函數都使用了memcpy函數進行數據的移動。

注意宏定義中的template<>語句,這正是模板特化的標志,關鍵詞template后面跟一對尖括號。

關于模板特化可以看這里。

#define ROS_CREATE_SIMPLE_SERIALIZER(Type) 
  template<  > struct Serializer Type > 
  { 
    template< typename Stream > inline static void write(Stream& stream, const Type v) 
{ 
      memcpy(stream.advance(sizeof(v)), &v, sizeof(v) ); 
    } 
    template< typename Stream > inline static void read(Stream& stream, Type& v) 
{ 
      memcpy(&v, stream.advance(sizeof(v)), sizeof(v) ); 
    } 
    inline static uint32_t serializedLength(const Type&) 
{ 
      return sizeof(Type); 
    } 
};
ROS_CREATE_SIMPLE_SERIALIZER(uint8_t)
ROS_CREATE_SIMPLE_SERIALIZER(int8_t)
ROS_CREATE_SIMPLE_SERIALIZER(uint16_t)
ROS_CREATE_SIMPLE_SERIALIZER(int16_t)
ROS_CREATE_SIMPLE_SERIALIZER(uint32_t)
ROS_CREATE_SIMPLE_SERIALIZER(int32_t)
ROS_CREATE_SIMPLE_SERIALIZER(uint64_t)
ROS_CREATE_SIMPLE_SERIALIZER(int64_t)
ROS_CREATE_SIMPLE_SERIALIZER(float)
ROS_CREATE_SIMPLE_SERIALIZER(double)

對于其它類型的數據,例如bool、std::string、std::vector、ros::Time、ros::Duration、boost::array等等,它們各自的處理方式有細微的不同,所以不再用上面的宏函數,而是用模板特化的方式每種單獨定義,這也是為什么serialization.h這個文件這么冗長。

對于int、double這種單個元素的數據,直接用上面特化的Serializer類中的memcpy函數實現序列化。

對于vector、array這種多個元素的數據類型怎么辦呢?方法是分成幾種情況,對于固定長度簡單類型的(fixed-size simple types),還是用各自特化的Serializer類中的memcpy函數實現,沒啥太大區別。

對于固定但是類型不簡單的(fixed-size non-simple types)或者既不固定也不簡單的(non-fixed-size, non-simple types)或者固定但是不簡單的(fixed-size, non-simple types),用for循環遍歷,一個元素一個元素的單獨處理。

那怎么判斷一個數據是不是固定是不是簡單呢?這是在roscpp_traits文件夾中的message_traits.h完成的。

其中采用了萃取Type Traits,這是相對高級一點的編程技巧了,筆者也不太懂。

對序列化的介紹暫時就到這里了,有一些細節還沒講,等筆者看懂了再補。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 機器人
    +關注

    關注

    211

    文章

    28467

    瀏覽量

    207353
  • 操作系統
    +關注

    關注

    37

    文章

    6838

    瀏覽量

    123389
  • 函數
    +關注

    關注

    3

    文章

    4333

    瀏覽量

    62705
  • ROS
    ROS
    +關注

    關注

    1

    文章

    278

    瀏覽量

    17024
收藏 人收藏

    評論

    相關推薦

    如何使用Serde進行序列化和反序列化

    。它是 Rust 生態中最受歡迎的序列化庫之一。 基礎用法 安裝 在 Rust 項目中使用 Serde,需要在 Cargo.toml 文件添加如下依賴: [dependencies] serde
    的頭像 發表于 09-30 17:09 ?1303次閱讀

    Java序列化的機制和原理

    ,我們用一個實例來示范序列化以后的字節是如何描述一個對象的信息的。序列化的必要性Java,一切都是對象,在分布式環境中經常需要將Object從這一端網絡或設備傳遞到另一端。這就需要有一種可以在兩端傳輸
    發表于 07-10 07:27

    c語言序列化和反序列化有何區別

    這里寫自定義目錄標題c語言序列化和反序列化tplut.htplut.c測試代碼參考c語言序列化和反序列化網絡調用,數據傳輸都需要把數據序列化
    發表于 07-14 07:32

    關于c語言序列化和反序列化的知識點看完你就懂了

    關于c語言序列和反序列化的知識點你就懂了
    發表于 10-15 08:47

    SpringMVC JSON框架的自定義序列化與反序列化

    限于createTime和updateTime,更貼近于需求缺點就是需要轉換的字段都需要使用注解,工作量有點大當然有其他的統一處理方案,這里不贅述。自定義反序列化在jackson框架上實現自定義序列化
    發表于 10-10 16:02

    理解PHP反序列化漏洞

    理解PHP反序列化漏洞
    發表于 09-07 11:03 ?7次下載
    理解PHP反<b class='flag-5'>序列化</b>漏洞

    java序列化和反序列化范例和JDK類庫序列化API

    存放在一個文件; 2) 在網絡上傳送對象的字節序列。 在很多應用,需要對某些對象進行序列化,讓它們離開內存空間,入住物理硬盤,以便長期保存。比如最常見的是Web服務器
    發表于 09-27 10:13 ?6次下載

    static屬性為什么不會被序列化

    實現序列化和反序列化為什么要實現Serializable接口?
    的頭像 發表于 07-15 11:03 ?1773次閱讀

    C#實現對象序列化的三種方式是什么

    很多小伙伴一提到序列化,都會想到二進制序列化,但其實序列化并不僅僅只是二進制序列化,我們常說的對象序列化有三種方式,分別是二進制
    的頭像 發表于 02-22 16:11 ?1214次閱讀
    C#<b class='flag-5'>實現</b>對象<b class='flag-5'>序列化</b>的三種方式是什么

    python序列化對象

    序列化對象:將對象轉換為可以存儲或傳輸的形式。 (1) 用于存儲:將對象的字節序列存儲到文件,程序退出后不會消失,便于后續使用。
    的頭像 發表于 03-10 09:57 ?2348次閱讀

    ROS機器人操作系統的實現原理(上)

    本文介紹ROS機器人操作系統(Robot Operating System)的實現原理,從最底層分析ROS代碼是如何實現的。 **1、序列化
    的頭像 發表于 05-19 17:41 ?999次閱讀
    <b class='flag-5'>ROS</b>機器人操作系統的<b class='flag-5'>實現</b>原理(上)

    ROS機器人操作系統的實現原理(下)

    本文介紹ROS機器人操作系統(Robot Operating System)的實現原理,從最底層分析ROS代碼是如何實現的。 **1、序列化
    的頭像 發表于 05-19 17:42 ?1184次閱讀

    什么是序列化 為什么要序列化

    什么是序列化? “序列化”(Serialization )的意思是將一個對象轉化為字節流。 這里說的對象可以理解為“面向對象”里的那個對象,具體的就是存儲在內存的對象數據。 與之相反的過程是“反
    的頭像 發表于 09-14 17:22 ?2564次閱讀
    什么是<b class='flag-5'>序列化</b> 為什么要<b class='flag-5'>序列化</b>

    Java序列化怎么使用

    轉換方式就叫做序列化。將文件或者網絡傳輸得到的 byte[] 數組轉換為 java 對象就叫做反序列化。 怎么使用 如果一個 Java 對象要能被序列化,必須
    的頭像 發表于 10-10 14:19 ?457次閱讀

    什么時候需要Boost序列化

    程序開發序列化是經常需要用到的。像一些相對高級語言,比如JAVA, C#都已經很好的支持了序列化,那么C++呢?當然一個比較好的選擇就是用Boost,這個號稱C++準標準庫的東西。 什么時候需要
    的頭像 發表于 11-10 10:14 ?428次閱讀
    主站蜘蛛池模板: 天天天干干干| 狠狠干欧美| 丁香婷婷激情综合| a天堂在线观看| 夜夜夜网| 日韩 ed2k| 四虎在线观看免费视频| 视频网站黄色| 91黄视频在线观看| 老师我好爽再深一点好大| 国产欧美另类第一页| 亚洲综合色一区| 国产美女精品久久久久中文| 免费播放欧美毛片欧美aaaaa| 国产亚洲一区二区三区在线| 亚洲欧美一区二区三区麻豆| 202z欧美成人| 97人人揉人人捏人人添| 五月天激激婷婷大综合丁香| 亚洲天堂网站在线| 色噜噜在线视频| 美女被免费网站视频九色| 日本三级午夜| 日产毛片| 深爱综合网| 四虎地址8848最新章节| 人人乳乳香蕉大免费| 国产一区二区精品| 久久精品人人做人人看| 欧美伊人网| 国产农村女人一级毛片了| 天天综合网久久| 婷婷色香五月激情综合2020| 久久久久国产一级毛片高清片| 亚洲mv在线观看| 久久亚洲成人| 午夜免费观看福利片一区二区三区| 国产在线精品香蕉综合网一区| 性做久久久久久| 思思久99久女女精品| 激情六月网|