recvfrom函式(經socket接收數據):
函式原型:ssize_t recvfrom(int sockfd,void *buf,size_t len,unsigned int flags, struct sockaddr *from,socket_t *fromlen); ssize_t 相當於 long int,socket_t 相當於int ,這裡用這個名字為的是提高代碼的自說明性。
基本介紹
- 中文名:接收函式
- 外文名:recvfrom
- 函式原型::ssize_t recvfrom
- 參數:sockfd
- 返回值:recv,recvmsg
參數,返回值,範例,
參數
sockfd:標識一個已連線套接口的描述字。
buf:接收數據緩衝區。
len:緩衝區長度。
flags:調用操作方式。是以下一個或者多個標誌的組合體,可通過“ | ”操作符連在一起:
MSG_DONTWAIT:操作不會被阻塞。
MSG_ERRQUEUE: 指示應該從套接字的錯誤佇列上接收錯誤值,依據不同的協定,錯誤值以某種輔佐性訊息的方式傳遞進來,使用者應該提供足夠大的緩衝區。導致錯誤的原封包通過msg_iovec作為一般的數據來傳遞。導致錯誤的數據報原目標地址作為msg_name被提供。錯誤以sock_extended_err結構形態被使用,定義如下
#define SO_EE_ORIGIN_NONE 0
#define SO_EE_ORIGIN_LOCAL 1
#define SO_EE_ORIGIN_ICMP 2
#define SO_EE_ORIGIN_ICMP6 3
struct sock_extended_err
{
u_int32_t ee_errno;
u_int8_t ee_origin;
u_int8_t ee_type;
u_int8_t ee_code;
u_int8_t ee_pad;
u_int32_t ee_info;
u_int32_t ee_data;
};
MSG_PEEK:指示數據接收後,在接收佇列中保留原數據,不將其刪除,隨後的讀操作還可以接收相同的數據。
MSG_TRUNC:返回封包的實際長度,即使它比所提供的緩衝區更長, 只對packet套接字有效。
MSG_WAITALL:要求阻塞操作,直到請求得到完整的滿足。然而,如果捕捉到信號,錯誤或者連線斷開發生,或者下次被接收的數據類型不同,仍會返回少於請求量的數據。
MSG_EOR:指示記錄的結束,返回的數據完成一個記錄。
MSG_TRUNC:指明數據報尾部數據已被丟棄,因為它比所提供的緩衝區需要更多的空間。
MSG_CTRUNC:指明由於緩衝區空間不足,一些控制數據已被丟棄。
MSG_OOB:指示接收到out-of-band數據(即需要優先處理的數據)。
MSG_ERRQUEUE:指示除了來自套接字錯誤佇列的錯誤外,沒有接收到其它數據。
from:(可選)指針,指向裝有源地址的緩衝區。
fromlen:(可選)指針,指向from緩衝區長度值。
返回值
如果正確接收返回接收到的位元組數,失敗返回-1.
相關函式 recv,recvmsg,send,sendto,socket
函式說明:recvfrom()用來接收遠程主機經指定的socket傳來的數據,並把數據傳到由參數buf指向的記憶體空間,參數len為可接收數據的最大長度.參數flags一般設0,其他數值定義參考recv().參數from用來指定欲傳送的網路地址,結構sockaddr請參考bind()函式.參數fromlen為sockaddr的結構長度.
返回值:成功則返回接收到的字元數,失敗返回-1.
EBADF 參數s非合法的socket處理代碼
EFAULT 參數中有一指針指向無法存取的記憶體空間。
ENOTSOCK 參數s為一檔案描述詞,非socket。
EINTR 被信號所中斷。
EAGAIN 此動作會令進程阻斷,但參數s的socket為不可阻斷。
ENOBUFS 系統的緩衝記憶體不足
ENOMEM 核心記憶體不足
EINVAL 傳給系統調用的參數不正確。
範例
/*利用socket的UDPclient此程式會連線UDPserver,並將鍵盤輸入的字元串傳給server。UDPserver範例請參考sendto()。*/#include<sys/stat.h>#include<fcntl.h>#include<unistd.h>#include<sys/types.h>#include<sys/socket.h>#include<netinet/in.h>#include<arpa/inet.h>#define PORT 2345#define SERVER_IP “127.0.0.1”main(){int s,len;struct sockaddr_in addr;int addr_len=sizeof(struct sockaddr_in);char buffer[256];/*建立socket*/if((s=socket(AF_INET,SOCK_DGRAM,0))<0){perror(“socket”);exit(1);}/*填寫sockaddr_in*/bzero(&addr,sizeof(addr));addr.sin_family=AF_INET;addr.sin_port=htons(PORT);addr.sin_addr.s_addr=inet_addr(SERVER_IP);while(1){bzero(buffer,sizeof(buffer));/*從標準輸入設備取得字元串*/len=read(STDIN_FILENO,buffer,sizeof(buffer));/*將字元串傳送給server端*/sendto(s,buffer,len,0,(struct sockaddr*)&addr,addr_len);/*接收server端返回的字元串*/len=recvfrom(s,buffer,sizeof(buffer),0,(struct sockaddr*)&addr,(socklen_t*)&addr_len);printf(“receive:%s”,buffer);}}執行(先執行udpserver再執行udpclient)hello/*從鍵盤輸入字元串*/receive:hello/*server端返回來的字元串*/