>UDP是一种不面向连接的不可靠的网络传输协议
UDP协议前面大部分和TCP协议差不多,只是他的客户程序不需要连接,而是直接发送数据。
>sendto() 和recvfrom() 函数
使用这两个函数,则数据会在没有建立过任何连接的网络上传输。正好可以在进行无连接的UDP 通讯时使用的。因为数据报套接字无法对远程主机进行连接,想想我们在发送数据前需要知道些什么呢?是远程主机的IP 地址和端口!
下面是sendto()函数和recvfrom()函数的声明:
#include <sys/types.h>
#include <sys/socket.h>
int sendto(int sockfd, const void *msg, int len, unsigned int flags,const struct sockaddr *to, int tolen);
和你所看到的一样,这个函数和send()函数基本一致。
sockfd 是代表你与远程程序连接的套接字描述符。
msg 是一个指针,指向你想发送的信息的地址。
len 是你想发送信息的长度。
flags 发送标记。一般都设为0。(你可以查看send 的man pages 来获得其他的参数值并且明白各个参数所代表的含义)
to 是一个指向struct sockaddr 结构的指针,里面包含了远程主机的IP 地址和端口数据。
tolen 只是指出了struct sockaddr 在内存中的大小sizeof(struct sockaddr)。和send()一样,sendto()返回它所真正发送的字节数(当然也和send()一样,它所真正发送的字节数可能小于你所给它的数据的字节数)。当它发生错误的时候,也是返回 –1 ,同时全局变量errno 存储了错误代码。同样的,recv()函数和recvfrom()函数也基本一致。
recvfrom()的声明为:
#include <sys/types.h>
#include <sys/socket.h>
int recvfrom(int sockfd, void *buf, int len, unsigned int flags
struct sockaddr *from, int *fromlen);
其参数含义如下:
sockfd 是你要读取数据的套接字描述符。
buf 是一个指针,指向你能存储数据的内存缓存区域。
len 是缓存区的最大尺寸。
flags 是recv() 函数的一个标志,一般都为0 (具体的其他数值和含义请参考recv()的man pages)。
from 是一个本地指针,指向一个struct sockaddr 的结构(里面存有源IP 地址和端口数).
fromlen 是一个指向一个int 型数据的指针,它的大小应该是sizeof ( structsockaddr).当函数返回的时候formlen 指向的数据是form 指向的struct sockaddr 的实际大小.
recvfrom() 返回它接收到的字节数,如果发生了错误,它就返回-1
>具体程序
//server.c
1 #include<stdio.h>
2 #include<sys/types.h>
3 #include<sys/socket.h>
4 #include<stdlib.h>
5 #include<unistd.h>
6 #include<netinet/in.h>
7 #include<arpa/inet.h>
8 #include<string.h>
9 static void usage(const char* proc)
10 {
11 printf("usage:%s [ip] [port]\n",proc);
12 }
13 int main(int argc,char* argv[])
14 {
15 if(argc!=3)
16 {
17 usage(argv[0]);
18 exit(1);
19 }
20 int sock=socket(AF_INET,SOCK_DGRAM,0);
21 if(sock<0)
22 {
23 perror("sock");
24 exit(2);
25 }
26 struct sockaddr_in local;
27 local.sin_family=AF_INET;
28 local.sin_addr.s_addr=inet_addr(argv[1]);
29 local.sin_port=htons(atoi(argv[2]));
30
31 if(bind(sock,(struct sockaddr*)&local,sizeof(local))<0)
32 {
33 perror("bind");
34 return 3;
35 }
36 int done=0;
37 struct sockaddr_in peer;
38 socklen_t len=sizeof(peer);
39 char buf[1024];
40
41 while(!done)
42 {
43 memset(buf,‘\0‘,sizeof(buf));
44 recvfrom(sock,buf,sizeof(buf),0,(struct sockaddr*)&peer,&len);
45 printf("#####################################\n");
46 printf("get a client,socket->%s:%d\n",inet_ntoa(peer.sin_addr),ntohs (peer.sin_port));
47 printf("client#:%s echo client!\n",buf);
48 printf("#####################################\n");
49 sendto(sock,buf,sizeof(buf),0,(struct sockaddr*)&peer,len);
50 }
51 return 0;
52 }
//client.c
1 #include<stdio.h>
2 #include<stdlib.h>
3 #include<sys/types.h>
4 #include<sys/socket.h>
5 #include<netinet/in.h>
6 #include<arpa/inet.h>
7 #include<string.h>
8 static void usage(const char* proc)
9 {
10 printf("usage:%s[remote_ip] [remote_port]\n",proc);
11 }
12 int main(int argc,char* argv[])
13 {
14 if(argc!=3)
15 {
16 usage(argv[0]);
17 return 1;
18 }
19
20 int sock=socket(AF_INET,SOCK_DGRAM,0);
21 if(sock<0)
22 {
23 perror("socket");
24 return 2;
25 }
26
27 struct sockaddr_in remote;
28 remote.sin_family=AF_INET;
29 remote.sin_addr.s_addr=inet_addr(argv[1]);
30 remote.sin_port=htons(atoi(argv[2]));
31
32 int done=0;
33 char buf[1024];
34 struct sockaddr_in peer;
35 socklen_t len=sizeof(peer);
36
37 while(!done)
38 {
39 printf("Please Enter:");
40 fflush(stdout);
41 ssize_t _s=read(0,buf,sizeof(buf)-1);
42 if(_s>0)
43 {
44 buf[_s]=‘\0‘;
45 sendto(sock,buf,sizeof(buf),0,(struct sockaddr*)&remote,sizeof(r emote));
46 memset(buf,‘\0‘,sizeof(buf));
47 recvfrom(sock,buf,sizeof(buf),0,(struct sockaddr*)&peer,&len);
48 printf("server echo %s,socket>%s:%d\n",buf,inet_ntoa(peer.sin_ad dr),ntohs(peer.sin_port));
49 }
50 }
51 return 0;
52 }
运行结果:
本文出自 “11275984” 博客,请务必保留此出处http://11285984.blog.51cto.com/11275984/1835425
原文地址:http://11285984.blog.51cto.com/11275984/1835425