tcp封包是 長度(2位元組) + 資料體,其中長度包含了這2個位元組,所以再解包的時候需要把這2位元組長度資訊去掉
unsigned char* tcp_protocol::package(const unsigned char* raw_data, int len, int* pkg_len)
所以再解包的時候需要把這2位元組長度資訊去掉
bool tcp_protocol::read_header(unsigned char* data, int data_len,int* pkg_size, int* out_head_size)
*pkg_size = ((data[0]) | (data[1] << 8));
*out_head_size = 2;
return true;
}
在c語言中,修飾符extern用在變數或者函式的宣告前,用來說明「此變數/函式是在別處定義的,要在此處引用」。
1. extern修飾變數的宣告。
舉例來說,如果檔案a.c需要引用b.c中變數int v,就可以在a.c中宣告extern int v,然後就可以引用變數v。能夠被其他模組以extern修飾符引用到的變數通常是全域性變數。還有很重要的一點是,extern int v可以放在a.c中的任何地方,比如你可以在a.c中的函式fun定義的開頭處宣告extern int v,然後就可以引用到變數v了,只不過這樣只能在函式fun作用域中引用v罷了,這還是變數作用域的問題。對於這一點來說,很多人使用的時候都心存顧慮。好像extern宣告只能用於檔案作用域似的。
2. extern修飾函式宣告。從本質上來講,變數和函式沒有區別。函式名是指向函式二進位制塊開頭處的指標。如果檔案a.c需要引用b.c中的函式,比如在b.c中原型是int fun(int mu),那麼就可以在a.c中宣告extern int fun(int mu),然後就能使用fun來做任何事情。就像變數的宣告一樣,extern int fun(int mu)可以放在a.c中任何地方,而不一定非要放在a.c的檔案作用域的範圍中。對其他模組中函式的引用,最常用的方法是包含這些函式宣告的標頭檔案。
使用extern和包含標頭檔案來引用函式有什麼區別呢?extern的引用方式比包含標頭檔案要簡潔得多!extern的使用方法是直接了當的,想引用哪個函式就用extern宣告哪個函式。這大概是kiss原則的一種體現吧!這樣做的乙個明顯的好處是,會加速程式的編譯(確切的說是預處理)的過程,節省時間。在大型c程式編譯過程中,這種差異是非常明顯的。
3. 此外,extern修飾符可用於指示c或者c++函式的呼叫規範。
比如在c++中呼叫c庫函式,就需要在c++程式中用extern 「c」宣告要引用的函式。這是給鏈結器用的,告訴鏈結器在鏈結的時候用c函式規範來鏈結。主要原因是c++和c程式編譯完成後在目標**中命名規則不同。
4. 舉個簡單的例子:
用c語言編寫程式的時候,我們經常會遇到這樣一種情況:希望在標頭檔案中定義乙個全域性變數,然後包含到兩個不同的c檔案中,希望這個全域性變數能在兩個檔案中共用。
舉例說明:專案資料夾project下有main.c、common.c和common.h三個檔案,其中
common.h檔案分別#include在main.c和common.c檔案中。現在希望宣告乙個字元型變數key,在main.c和common.c中公用。如下圖所示:
有人想,既然是想兩個檔案都用,那就在common.h中宣告乙個unsigned char key,然後由於包含關係,在main.c和common.c中都是可見的,所以就能共用了。
這種想法其實是很多初學者都會想到的,想起來確實有道理,但是實際寫出來,我們發現編譯的時候編譯器提示出錯,一般提示大概都類似於:error: l6200e: symbol key multiply defined (by common.o and main.o).也就是說編譯器認為我們重複定義了key這個變數。這是因為#include命令就是原封不同的把頭檔案中的內容搬到#include的位置,所以相當於main.c和common.c中都執行了一次unsigned char key,而c語言中全域性變數是專案內(或者叫工程內)可見的,這樣就造成了乙個專案中兩個變數key,編譯器就認為是重複定義。
正確的解決辦法:使用extern關鍵字來宣告變數為外部變數。具體說就是在其中乙個c檔案中定義乙個全域性變數key,然後在另乙個要使用key這個變數的c檔案中使用extern關鍵字宣告一次,說明這個變數為外部變數,是在其他的c檔案中定義的全域性變數。請注意我這裡的用詞:定義和宣告。例如在main.c檔案中定義變數key,在common.c檔案中宣告key變數為外部變數,這樣這兩個檔案中就能共享這個變數key了。
node 第七章TCP通訊的拆包與封包
1 在通訊的過程中,我們可能有傳送多個資料報,資料報a,資料報b,資料報c,此時我們最好的期望是每次收到資料報a,資料報b,資料報c。但是tcp底層為了傳送效能,可能會一次把abc所有資料一起傳過來,這個時候收到的是a b c,這個時候上層傻眼了,無法區分a,b,c這個叫做 粘包 1 要解決abc資...
Socket 封包與拆包入門程式示例
socket開發應解決的兩個基本問題 1 client 與 server的連線通訊,可參考 mfc socket網路程式設計 1 伺服器 mfc socket網路程式設計 1 客戶端 2 資料的封包與拆包,下面將部分關鍵 摘出,自己仔細把邏輯理通,相信很快能寫自己的封包與拆包程式。封包 bool t...
python學習筆記三包與模組 反射
1.庫的安裝與解除安裝 python有2種庫 第三方的庫 不知道庫版本時可以這樣寫 pip3 install selenium 2.xx,就會返回所有2.0以上的版本號 離線安裝的方式 2.模組與模組之間的呼叫 乙個python檔案就是乙個模組,包含 類 函式 變數 呼叫模組裡的類 函式 變數 fr...