ImageVerifierCode 换一换
格式:DOCX , 页数:13 ,大小:20.84KB ,
资源ID:23965740      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/23965740.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(04VC实现最简单的UDP通信包含客户端与服务端代码.docx)为本站会员(b****7)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

04VC实现最简单的UDP通信包含客户端与服务端代码.docx

1、04VC实现最简单的UDP通信包含客户端与服务端代码VC实现最简单的UDP通信1客户端/Client端代码#include #include #include void initClient();/函数声明/*主函数*/int main() initClient();/初始化客户端 return 0;/*初始化客户端函数*/void initClient() WSADATA wsaData; int error=WSAStartup(MAKEWORD(2,2),&wsaData); if(error!=0) cout初始化DLL失败endl; return; if(LOBYTE(wsaData

2、.wVersion)!=2 | HIBYTE(wsaData.wVersion)!=2) WSACleanup(); cout版本出错endl; return; SOCKET s=socket(AF_INET,SOCK_DGRAM,0); SOCKADDR_IN sockSend; sockSend.sin_addr.S_un.S_addr=inet_addr(127.0.0.1); sockSend.sin_port=htons(4000); sockSend.sin_family=AF_INET; char buff1024; strcpy(buff,hello,its the first

3、!); int i=500; while(-i) int lenword; lenword=sendto(s,buff,strlen(buff)+1,0,(sockaddr *)&sockSend,sizeof(sockaddr); coutlenword,sockSend.sin_port:sockSend.sin_addr.S_un.S_addrendl; closesocket(s); WSACleanup();2服务端/Server端代码#include #include #include using namespace std;void initNet(); /函数声明/*主函数*/

4、int main() initNet();/网络初始化 return 0;/返回0值/*网络初始化函数*/void initNet() WSADATA wsaData; int error=WSAStartup(MAKEWORD(1,1),&wsaData); if(error!=0) cout初始化DLL失败endl; return; if(LOBYTE(wsaData.wVersion)!=1 | HIBYTE(wsaData.wVersion)!=1) WSACleanup(); cout版本出错endl; return; SOCKET s=socket(AF_INET,SOCK_DGR

5、AM,0); SOCKADDR_IN sockSrc; sockSrc.sin_addr.S_un.S_addr=htonl(INADDR_ANY); sockSrc.sin_port=htons(4000); sockSrc.sin_family=AF_INET; bind(s,(SOCKADDR *)&sockSrc,sizeof(SOCKADDR); char recBuff1024; memset(recBuff,0,1024); SOCKADDR_IN sockRec; int len=sizeof(SOCKADDR); int x=-1; coutsockSrc.sin_port:

6、sockSrc.sin_addr.S_un.S_addrendl; while(x=-1) x=recvfrom(s,recBuff,sizeof(recBuff),0,(sockaddr *)&sockRec,&len); printf(the receive is:%s,%d n,recBuff,x); closesocket(s); WSACleanup();程序能够运行,客户端发送的数据服务端收不到,运行时客户端和服务都再一台机器Windows 95环境下,基于TCP/IP协议,用Winsock完成了话音的一端传输摘要:在Windows 95环境下,基于TCP/IP协议,用Winsoc

7、k完成了话音的端到端传输。采用双套接字技术,阐述了主要函数的使用要点,以及基于异步选择机制的应用方法。同时,给出了相应的实例程序。3引言 Windows 95作为微机的操作系统,已经完全融入了网络与通信功能,不仅可以建立纯Windows 95环境下的“对等网络”,而且支持多种协议,如TCP/IP、IPX/SPX、NETBUI等。在TCP/IP协议组中,TPC是一种面向连接的协议,为用户提供可靠的、全双工的字节流服务,具有确认、流控制、多路复用和同步等功能,适于数据传输。UDP协议则是无连接的,每个分组都携带完整的目的地址,各分组在系统中独立传送。它不能保证分组的先后顺序,不进行分组出错的恢复与

8、重传,因此不保证传输的可靠性,但是,它提供高传输效率的数据报服务,适于实时的语音、图像传输、广播消息等网络传输。Winsock接口为进程间通信提供了一种新的手段,它不但能用于同一机器中的进程之间通信,而且支持网络通信功能。随着Windows 95的推出。Winsock已经被正式集成到了Windows系统中,同时包括了16位和32位的编程接口。而Winsock的开发工具也可以在Borland C+4.0、Visual C+2.0这些C编译器中找到,主要由一个名为winsock.h的头文件和动态连接库winsock.dll或wsodk32.dll组成,这两种动态连接库分别用于Win16和Win32

9、的应用程序。本文针对话音的全双工传输要求,采用UDP协议实现了实时网络通信。使用Visual C+2.0编译环境,其动态连接库名为wsock32.dll。4主要函数的使用要点 通过建立双套接字,可以很方便地实现全双工网络通信。 4.1套接字建立函数: SOCKET socket(int family,int type,int protocol) 对于UDP协议,写为: SOCKRET s; s=socket(AF_INET,SOCK_DGRAM,0); 或s=socket(AF_INET,SOCK_DGRAM,IPPROTO_UDP) 为了建立两个套接字,必须实现地址的重复绑定,即,当一个套接

10、字已经绑定到某本地地址后,为了让另一个套接字重复使用该地址,必须为调用bind()函数绑定第二个套接字之前,通过函数setsockopt()为该套接字设置SO_REUSEADDR套接字选项。通过函数getsockopt()可获得套接字选项设置状态。需要注意的是,两个套接字所对应的端口号不能相同。 此外,还涉及到套接字缓冲区的设置问题,按规定,每个区的设置范围是:不小于512个字节,大大于8k字节,根据需要,文中选用了4k字节。 4.2套接字绑定函数int bind(SOCKET s,struct sockaddr_in*name,int namelen) s是刚才创建好的套接字,name指向描

11、述通讯对象的结构体的指针,namelen是该结构体的长度。该结构体中的分量包括:IP地址(对应name.sin_addr.s_addr)、端口号(name.sin_port)、地址类型(name.sin_family,一般都赋成AF_INET,表示是internet地址)。 (1)IP地址的填写方法:在全双工通信中,要把用户名对应的点分表示法地址转换成32位长整数格式的IP地址,使用inet_addr()函数。 (2)端口号是用于表示同一台计算机不同的进程(应用程序),其分配方法有两种:1)进程可以让系统为套接字自动分配一端口号,只要在调用bind前将端口号指定为0即可。由系统自动分配的端口号

12、位于10245000之间,而11023之间的任一TCP或UDP端口都是保留的,系统不允许任一进程使用保留端口,除非其有效用户ID是零(超级用户)。 2)进程可为套接字指定一特定端口。这对于需要给套接字分配一众所端口的服务器是很有用的。指定范围为1024和65536之间。可任意指定。 在本程序中,对两个套接字的端口号规定为2000和2001,前者对应发送套接字,后者对应接收套接字。 端口号要从一个16位无符号数(u_short类型数)从主机字节顺序转换成网络字节顺序,使用htons()函数。 根据以上两个函数,可以给出双套接字建立与绑定的程序片断。 /设置有关的全局变量 SOCKET sr,ss

13、; HPSTR sockBufferS,sockBufferR; HANDLE hSendData,hReceiveData; DWROD dwDataSize=1024*4; struct sockaddr_in therel.there2; #DEFINE LOCAL_HOST_ADDR 200.200.200.201 #DEFINE REMOTE_HOST-ADDR 200.200.200.202 #DEFINE LOCAL_HOST_PORT 2000 #DEFINE LOCAL_HOST_PORT 2001 /套接字建立函数 BOOL make_skt(HWND hwnd) stru

14、ct sockaddr_in here,here1; ss=socket(AF_INET,SOCK_DGRAM,0); sr=socket(AF_INET,SOCK_DGRAM,0); if(ss=INVALID_SOCKET)|(sr=INVALID_SOCKET) MessageBox(hwnd,“套接字建立失败!”,“”,MB_OK); return(FALSE); here.sin_family=AF_INET; here.sin_addr.s_addr=inet_addr(LOCAL_HOST_ADDR); here.sin_port=htons(LICAL_HOST_PORT);

15、/another socket herel.sin_family=AF_INET; herel.sin_addr.s_addr(LOCAL_HOST_ADDR); herel.sin_port=htons(LOCAL_HOST_PORT1); SocketBuffer();/套接字缓冲区的锁定设置 setsockopt(ss,SOL_SOCKET,SO_SNDBUF,(char FAR*)sockBufferS,dwDataSize); if(bind(ss,(LPSOCKADDR)&here,sizeof(here) MessageBox(hwnd,“发送套接字绑定失败!”,“”,MB_OK

16、); return(FALSE); setsockopt(sr SQL_SOCKET,SO_RCVBUF|SO_REUSEADDR,(char FAR*) sockBufferR,dwDataSize); if(bind(sr,(LPSOCKADDR)&here1,sizeof(here1) MessageBox(hwnd,“接收套接字绑定失败!”,“”,MB_OK); return(FALSE); return(TRUE); /套接字缓冲区设置 void sockBuffer(void) hSendData=GlobalAlloc(GMEM_MOVEABLE|GMEM_SHARE,dwDat

17、aSize); if(!hSendData) MessageBox(hwnd,“发送套接字缓冲区定位失败!”,NULL, MB_OK|MB_ICONEXCLAMATION); return; if(sockBufferS=GlobalLock(hSendData)=NULL) MessageBox(hwnd,“发送套接字缓冲区锁定失败!”,NULL, MB_OK|MB_ICONEXCLAMATION); GlobalFree(hRecordData0; return; hReceiveData=globalAlloc(GMEM_MOVEABLE|GMEM_SHARE,dwDataSize);

18、if(!hReceiveData) MessageBox(hwnd,“接收套接字缓冲区定位败!”,NULL MB_OK|MB_ICONEXCLAMATION); return; if(sockBufferT=Globallock(hReceiveData)=NULL) MessageBox(hwnd,发送套接字缓冲区锁定失败!”,NULL, MB_OK|MB_ICONEXCLAMATION); GlobalFree(hRecordData0); return; 4.3数据发送与接收函数; int sendto(SOCKET s.char*buf,int len,int flags,struct

19、 sockaddr_in to,int tolen); int recvfrom(SOCKET s.char*buf,int len,int flags,struct sockaddr_in fron,int*fromlen) 其中,参数flags一般取0。 recvfrom()函数实际上是读取sendto()函数发过来的一个数据包,当读到的数据字节少于规定接收的数目时,就把数据全部接收,并返回实际接收到的字节数;当读到的数据多于规定值时,在数据报文方式下,多余的数据将被丢弃。而在流方式下,剩余的数据由下recvfrom()读出。为了发送和接收数据,必须建立数据发送缓冲区和数据接收缓冲区。规定

20、:IP层的一个数据报最大不超过64K(含数据报头)。当缓冲区设置得过多、过大时,常因内存不够而导致套接字建立失败。在减小缓冲区后,该错误消失。经过实验,文中选用了4K字节。 此外,还应注意这两个函数中最后参数的写法,给sendto()的最后参数是一个整数值,而recvfrom()的则是指向一整数值的指针。4.4套接字关闭函数:closesocket(SOCKET s) 通讯结束时,应关闭指定的套接字,以释与之相关的资源。 在关闭套接字时,应先对锁定的各种缓冲区加以释放。其程序片断为: void CloseSocket(void) GlobalUnlock(hSendData); GlobalF

21、ree(hSenddata); GlobalUnlock(hReceiveData); GlobalFree(hReceiveDava); if(WSAAysncSelect(ss,hwnd,0,0)=SOCKET_ERROR) MessageBos(hwnd,“发送套接字关闭失败!”,“”,MB_OK); return; if(WSAAysncSelect(sr,hwnd,0,0)=SOCKET_ERROR) MessageBox(hwnd,“接收套接字关闭失败!”,“”,MB_OK); return; WSACleanup(); closesockent(ss); closesockent

22、(sr); return; 5Winsock的编程特点与异步选择机制 5.1阻塞及其处理方式 在网络通讯中,由于网络拥挤或一次发送的数据量过大等原因,经常会发生交换的数据在短时间内不能传送完,收发数据的函数因此不能返回,这种现象叫做阻塞。Winsock对有可能阻塞的函数提供了两种处理方式:阻塞和非阻塞方式。在阻塞方式下,收发数据的函数在被调用后一直要到传送完毕或者出错才能返回。在阻塞期间,被阻的函数不会断调用系统函数GetMessage()来保持消息循环的正常进行。对于非阻塞方式,函数被调用后立即返回,当传送完成后由Winsock给程序发一个事先约定好的消息。 在编程时,应尽量使用非阻塞方式。

23、因为在阻塞方式下,用户可能会长时间的等待过程中试图关闭程序,因为消息循环还在起作用,所以程序的窗口可能被关闭,这样当函数从Winsock的动态连接库中返回时,主程序已经从内存中删除,这显然是极其危险的。 5.2异步选择函数WSAAsyncSelect()的使用 Winsock通过WSAAsyncSelect()自动地设置套接字处于非阻塞方式。使用WindowsSockets实现Windows网络程序设计的关键就是它提供了对网络事件基于消息的异步存取,用于注册应用程序感兴趣的网络事件。它请求Windows Sockets DLL在检测到套接字上发生的网络事件时,向窗口发送一个消息。对UDP协议,

24、这些网络事件主要为: FD_READ 期望在套接字收到数据(即读准备好)时接收通知; FD_WRITE 期望在套接字可发送数(即写准备好)时接收通知; FD_CLOSE 期望在套接字关闭时接电通知 消息变量wParam指示发生网络事件的套接字,变量1Param的低字节描述发生的网络事件,高字包含错误码。如在窗口函数的消息循环中均加一个分支: int ok=sizeof(SOCKADDR); case wMsg; switch(1Param) case FD_READ: /套接字上读数据 if(recvfrom(sr.lpPlayDataj,dwDataSize,0,(struct sockad

25、dr FAR*)&there1, (int FAR*)&ok)=SOCKET_ERROR0 MessageBox)hwnd,“数据接收失败!”,“”,MB_OK); return(FALSE); case FD_WRITE: /套接字上写数据 break; 在程序的编制中,应根据需要灵活地将WSAAsyncSelect()函灵敏放在相应的消息循环之中,其它说明可参见文献1。此外,应该指出的是,以上程序片断中的消息框主要是为程序调试方便而设置的,而在正式产品中不再出现。同时,按照程序容错误设计,应建立一个专门的容错处理函数。程序中可能出现的各种错误都将由该函数进行处理,依据错误的危害程度不同,建立几种不同的处理措施。这样,才能保证双方通话的顺利和可靠。

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1