0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

ROS機(jī)器人操作系統(tǒng)的實現(xiàn)原理(上)

jf_78858299 ? 來源:古月居 ? 作者:robinvista ? 2023-05-19 17:41 ? 次閱讀

本文介紹ROS機(jī)器人操作系統(tǒng)(Robot Operating System)的實現(xiàn)原理,從最底層分析ROS代碼是如何實現(xiàn)的。

1、序列化

通信的內(nèi)容(也就是消息message)序列化是通信的基礎(chǔ),所以我們先研究序列化。

盡管筆者從事機(jī)器人學(xué)習(xí)和研發(fā)很長時間了,但是在研究ROS的過程中,“序列化”這個詞還是這輩子第一次聽到。

所以可想而知很多人在看到“把一個消息序列化”這樣的描述時是如何一臉懵逼。

但其實序列化是一個比較常見的概念,你雖然不知道它但一定接觸過它。

下面我們先介紹“序列化”的一些常識,然后解釋ROS里的序列化是怎么做的?

1.1什么是序列化?

“序列化”(Serialization )的意思是將一個對象轉(zhuǎn)化為字節(jié)流。

這里說的對象可以理解為“面向?qū)ο蟆崩锏哪莻€對象,具體的就是存儲在內(nèi)存中的對象數(shù)據(jù)。

與之相反的過程是“反序列化”(Deserialization )。

雖然掛著機(jī)器人的羊頭,但是后面的介紹全部是計算機(jī)知識,跟機(jī)器人一丁點關(guān)系都沒有,序列化就是一個純粹的計算機(jī)概念。

序列化的英文Serialize就有把一個東西變成一串連續(xù)的東西之意。

形象的描述,數(shù)據(jù)對象是一團(tuán)面,序列化就是將面團(tuán)拉成一根面條,反序列化就將面條捏回面團(tuán)。

另一個形象的類比是我們在對話或者打電話時,一個人的思想轉(zhuǎn)換成一維的語音,然后在另一個人的頭腦里重新變成結(jié)構(gòu)化的思想,這也是一種序列化。

圖片

面對序列化,很多人心中可能會有很多疑問。

首先,為什么要序列化?或者更具體的說,既然對象的信息本來就是以字節(jié)的形式儲存在內(nèi)存中,那為什么要多此一舉把一些字節(jié)數(shù)據(jù)轉(zhuǎn)換成另一種形式的、一維的、連續(xù)的字節(jié)數(shù)據(jù)呢?

如果我們的程序在內(nèi)存中存儲了一個數(shù)字,比如25。那要怎么傳遞25這個數(shù)字給別的程序節(jié)點或者把這個數(shù)字永久存儲起來呢?

很簡單,直接傳遞25這個數(shù)字(的字節(jié)表示,即0X19,當(dāng)然最終會變成二進(jìn)制表示11001以高低電平傳輸存儲)或者直接把這個數(shù)字(的字節(jié)表示)寫進(jìn)硬盤里即可。

所以,對于本來就是連續(xù)的、一維的、一連串的數(shù)據(jù)(例如字符串),序列化并不需要做太多東西,其本質(zhì)是就是由內(nèi)存向其它地方拷貝數(shù)據(jù)而已。

所以,如果你在一個序列化庫里看到memcpy函數(shù)不用覺得奇怪,因為你知道序列化最底層不過就是在操作內(nèi)存數(shù)據(jù)而已(還有些庫使用了流的ostream.rdbuf()->sputn函數(shù))。

可是實際程序操作的對象很少是這么簡單的形式,大多數(shù)時候我們面對的是包含不同數(shù)據(jù)類型(int、double、string)的復(fù)雜數(shù)據(jù)結(jié)構(gòu)(比如vector、list),它們很可能在內(nèi)存中是不連續(xù)存儲的而是分散在各處。比如ROS的很多消息都包含向量。

數(shù)據(jù)中還有各種指針和引用。而且,如果數(shù)據(jù)要在運行于不同架構(gòu)的計算機(jī)之上的、由不同編程語言所編寫的節(jié)點程序之間傳遞,那問題就更復(fù)雜了,它們的字節(jié)順序endianness規(guī)定有可能不一樣,基本數(shù)據(jù)類型(比如int)的長度也不一樣(有的int是4個字節(jié)、有的是8個字節(jié))。

這些都不是通過簡單地、原封不動地復(fù)制粘貼原始數(shù)據(jù)就能解決的。這時候就需要序列化和反序列化了。

所以在程序之間需要通信時(ROS恰好就是這種情況),或者希望保存程序的中間運算結(jié)果時,序列化就登場了。

另外,在某種程度上,序列化還起到統(tǒng)一標(biāo)準(zhǔn)的作用。

圖片

我們把被序列化的東西叫object(對象),它可以是任意的數(shù)據(jù)結(jié)構(gòu)或者對象:結(jié)構(gòu)體、數(shù)組、類的實例等等。

把序列化后得到的東西叫archive,它既可以是人類可讀的文本形式,也可以是二進(jìn)制形式。

前者比如JSON和XML,這兩個是網(wǎng)絡(luò)應(yīng)用里最常用的序列化格式,通過記事本就能打開閱讀;

后者就是原始的二進(jìn)制文件,比如后綴名是bin的文件,人類是沒辦法直接閱讀一堆的0101或者0XC9D23E72的。

序列化算是一個比較常用的功能,所以大多數(shù)編程語言(比如C++Python、Java等)都會附帶用于序列化的庫,不需要你再去造輪子。

以C++為例,雖然標(biāo)準(zhǔn)STL庫沒有提供序列化功能,但是第三方庫Boost提供了[ 2 ]谷歌的protobuf也是一個序列化庫,還有Fast-CDR,以及不太知名的Cereal,Java自帶序列化函數(shù),python可以使用第三方的pickle模塊實現(xiàn)。

總之,序列化沒有什么神秘的,用戶可以看看這些開源的序列化庫代碼,或者自己寫個小程序試試簡單數(shù)據(jù)的序列化,例如這個例子,或者這個,有助于更好地理解ROS中的實現(xiàn)。

1.2ROS中的序列化實現(xiàn)

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

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

1.2.1serialization.h

核心的函數(shù)都在serialization.h里,簡而言之,里面使用了C語言標(biāo)準(zhǔn)庫的memcpy函數(shù)把消息拷貝到流中。

下面來看一下具體的實現(xiàn)。

序列化功能的特點是要處理很多種數(shù)據(jù)類型,針對每種具體的類型都要實現(xiàn)相應(yīng)的序列化函數(shù)。

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

從后往前梳理,先看Stream這個結(jié)構(gòu)體吧。在C++里結(jié)構(gòu)體和類基本沒什么區(qū)別,結(jié)構(gòu)體里也可以定義函數(shù)。

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

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

更形象的,可以想象磁帶或者圖靈機(jī)里連續(xù)的紙帶。在文件讀寫、使用串口、網(wǎng)絡(luò)Socket通信等領(lǐng)域,流經(jīng)常被使用。例如我們常用的輸入輸出流:

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_;
};
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 機(jī)器人
    +關(guān)注

    關(guān)注

    211

    文章

    28445

    瀏覽量

    207230
  • 操作系統(tǒng)
    +關(guān)注

    關(guān)注

    37

    文章

    6834

    瀏覽量

    123350
  • ROS
    ROS
    +關(guān)注

    關(guān)注

    1

    文章

    278

    瀏覽量

    17019
收藏 人收藏

    評論

    相關(guān)推薦

    ROS機(jī)器人開發(fā)更便捷,基于RK3568J+Debian系統(tǒng)發(fā)布!

    本帖最后由 Tronlong創(chuàng)龍科技 于 2024-7-19 17:18 編輯 ROS系統(tǒng)是什么 ROS(Robot Operating System)是一個適用于機(jī)器人的開源的
    發(fā)表于 07-09 11:38

    【IntoRobot Atom試用體驗】解決Atom搭建機(jī)器人操作系統(tǒng)問題(二)

    根據(jù)一篇文章解決Atom搭建機(jī)器人操作系統(tǒng)問題(一),在做庫移植的時候會遇到以下問題。。。。/main.cpp:1:17: fatal error: ros.h: No such f
    發(fā)表于 04-07 22:00

    請問能分享ROS機(jī)器人操作系統(tǒng)的一些資料嗎?

    菜鳥想要學(xué)習(xí)ROS機(jī)器人操作系統(tǒng),但是沒能找到系統(tǒng)的資料,請問有哪位大神研究過,可以分享一些資料嗎?跪謝了?。。。。。。。?!
    發(fā)表于 02-19 23:14

    創(chuàng)龍TL5728-EasyEVM-A4開發(fā)板如何移植ros機(jī)器人操作系統(tǒng)?

    創(chuàng)龍TL5728-EasyEVM-A4開發(fā)板如何移植ros機(jī)器人操作系統(tǒng)?
    發(fā)表于 01-02 09:32

    ROS 2 Crystal Clemmys版機(jī)器人操作系統(tǒng)補(bǔ)充說明

    ROS 2 Crystal Clemmys版機(jī)器人操作系統(tǒng)補(bǔ)充說明
    發(fā)表于 06-10 11:29

    ROS RIKIBOT基礎(chǔ)--使用系列 第一章節(jié)】ROS機(jī)器人硬件系統(tǒng) 精選資料分享

    主控制器ROS控制器采用樹莓派4B或者Jetson Nano,運行運行Ubuntu Mate18.04或Ubuntu18.04系統(tǒng),具體型號有差異。系統(tǒng)安裝有ROS
    發(fā)表于 07-30 06:59

    ROS機(jī)器人開發(fā)更便捷,基于RK3568J+Debian系統(tǒng)發(fā)布!

    ROS系統(tǒng)是什么 ROS(Robot Operating System)是一個適用于機(jī)器人的開源的元操作系統(tǒng)。它提供了
    發(fā)表于 11-30 16:01

    ROS是什么?機(jī)器人操作系統(tǒng)ROS的介紹

    ROS 是 Robot Operating System 的縮寫,原本是斯坦福大學(xué)的一個機(jī)器人項目,后 來由 WillowGarage 公司發(fā)展,目前由 OSRF(Open Source
    發(fā)表于 09-13 17:23 ?14次下載
    <b class='flag-5'>ROS</b>是什么?<b class='flag-5'>機(jī)器人</b><b class='flag-5'>操作系統(tǒng)</b><b class='flag-5'>ROS</b>的介紹

    ROS的含義與機(jī)器人操作系統(tǒng)ROS的介紹

    Operating System 的縮寫,原本是斯坦福大學(xué)的一個機(jī)器人項目,后來由 WillowGarage 公司發(fā)展,目前由 OSRF(Open Source Robotics Foundation,Inc)公司維護(hù)的開源項目。 1、首先是一個操作系統(tǒng) 根據(jù) wiki
    發(fā)表于 09-26 14:08 ?12次下載

    微軟宣布Window10將正式引入機(jī)器人操作系統(tǒng)

    日前,在西班牙召開的第七屆機(jī)器人操作系統(tǒng)開發(fā)者大會(ROSCon 2018),微軟宣布Window10正式引入機(jī)器人操作系統(tǒng)(
    的頭像 發(fā)表于 10-10 15:42 ?3344次閱讀

    機(jī)器人操作系統(tǒng)ROS)是一種用于機(jī)器人開發(fā)的元操作系統(tǒng)

    拋開它的名字,ROS并不是一個真正的操作系統(tǒng)。相反,它是一個框架和服務(wù)的集合,這些框架和服務(wù)提供了類似操作系統(tǒng)的功能,運行在由機(jī)器人硬件支撐的異構(gòu)計算機(jī)集群之上。
    的頭像 發(fā)表于 10-12 16:53 ?7125次閱讀

    機(jī)器人操作系統(tǒng)ROS詳細(xì)介紹

    機(jī)器人操作系統(tǒng)ROS詳細(xì)介紹 ROS機(jī)器人 操作系統(tǒng),RobotOperatingSyste
    發(fā)表于 11-18 18:09 ?2391次閱讀

    ROS機(jī)器人操作系統(tǒng)實現(xiàn)原理(下)

    本文介紹ROS機(jī)器人操作系統(tǒng)(Robot Operating System)的實現(xiàn)原理,從最底層分析ROS代碼是如何
    的頭像 發(fā)表于 05-19 17:42 ?1177次閱讀

    ROS機(jī)器人操作系統(tǒng)

    可能很多初學(xué)者聽到機(jī)器人操作系統(tǒng),就被“操作系統(tǒng)”幾個字嚇住了。其實簡單點說,**ROS就是一個分布式的通信框架,幫助程序進(jìn)程之間更方便地通信。
    的頭像 發(fā)表于 05-19 17:46 ?2203次閱讀
    <b class='flag-5'>ROS</b><b class='flag-5'>機(jī)器人</b><b class='flag-5'>操作系統(tǒng)</b>

    一個機(jī)器人操作系統(tǒng)(ROS)節(jié)點系統(tǒng)

    電子發(fā)燒友網(wǎng)站提供《一個機(jī)器人操作系統(tǒng)(ROS)節(jié)點系統(tǒng).zip》資料免費下載
    發(fā)表于 07-06 10:49 ?1次下載
    一個<b class='flag-5'>機(jī)器人</b><b class='flag-5'>操作系統(tǒng)</b>(<b class='flag-5'>ROS</b>)節(jié)點<b class='flag-5'>系統(tǒng)</b>