=================================================
本文為HeYuanHui原作
轉載必須確保本文完整並完整保留原作者資訊及本文原始連結!
NN: khler
E-mail: khler@163.com
QQ: 23381103
MSN: pragmac@hotmail.com
=================================================
最近碰到一個問題,對於阻塞模式的socket通訊,如果要實現裝置的命令控制,那麼進入命令流前,緩衝區不能存有上次通訊沒有取回的資訊,否則一旦命令發出,然後讀取緩衝區,很顯然會讀到上一次的剩餘資料。做法當然很簡單,就是先清除接收區的緩衝資料,可是如何清除?
網上有很多這樣的問題,但都沒什麼規範的解決辦法,有的甚至為了達到清空的目的,建議先close一下socket,這個太大手筆了,為瞭解決一個小問題而大動幹戈,根本不是個合理的解決辦法。
還有就是用recv讀取,但是由於不知道緩衝裡有多少資料,如果是阻塞模式,到最後必然等到逾時才知道資料已經讀取完畢,這是個問題。
另一個是用fgetc,通過返回判斷是否是feof:
whlie (1) {
a=fgetc(f);
if (feof(f)) break;
//…
b=fgetc(f);
if (feof(f)) break;
//…
}
當然,我不知道讀取完畢後最後一次調用fgetc會不會堵塞,需要測試。
在非阻塞模式下,我們用recv就可以輕鬆搞定了,但是阻塞模式下,由於我們不知道緩衝區有多少資料,不能直接調用recv嘗試清除。
使用一個小小的技巧,利用select函數,我們可以輕鬆搞定這個問題:
select函數用於監視一個檔案描述符集合,如果集合中的描述符沒有變化,則一直阻塞在這裡,直到逾時時間到達;在逾時時間內,一旦某個描述符觸發了你所關心的事件,select立即返回,通過檢索檔案描述符集合處理相應事件;select函數出錯則返回小於零的值,如果有事件觸發,則返回觸發事件的描述符個數;如果逾時,返回0,即沒有資料可讀。
重點在於:我們可以用select的逾時特性,將逾時時間設定為0,通過檢測select的傳回值,就可以判斷緩衝是否被清空。通過這個技巧,使一個阻塞的socket成了‘非阻塞’socket。
現在就可以得出解決方案了:使用select函數來監視要清空的socket描述符,並把逾時時間設定為0,每次讀取一個位元組然後丟棄(或者按照業務需要進行處理,隨你便了),一旦select返回0,說明緩衝區沒資料了(“逾時”了)。
struct timeval tmOut;
tmOut.tv_sec = 0;
tmOut.tv_usec = 0;
fd_set fds;
FD_ZEROS(&fds);
FD_SET(skt, &fds);
int nRet;
char tmp[2];
memset(tmp, 0, sizeof(tmp));
while(1)
{ nRet= select(FD_SETSIZE, &fds, NULL, NULL, &tmOut);
if(nRet== 0) break;
recv(skt, tmp, 1,0);
}
這種方式的好處是,不再需要用recv、recvfrom等阻塞函數直接去讀取,而是使用select,利用其逾時特性檢測緩衝區是否為空白來判斷是否有資料,有資料時才調用recv進行清除。
有人說同樣可以用recv和socket的逾時設定去清空啊,這個沒錯,但是你需要直接對socket描述符設定逾時時間,而為了清空資料而直接修改socket描述符的屬性,可能會影響到其他地方的使用,造成系統奇奇怪怪的問題,所以,不推薦使用。