《Socket编程基础.ppt》由会员分享,可在线阅读,更多相关《Socket编程基础.ppt(58页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、Socket编程基础套接字的概念与编程原理Winsock API基本函数套接口与连接的建立Winsock API基本函数数据传输Winsock API基本函数套接口与连接的关闭面向连接的客户/服务器程序工作流程以及举例无连接的客户/服务器程序工作流程以及举例Socket编程接口来源起源于UNIX操作系统Socket编程接口是对TCP/IP协议的一种封装Socket本身并不是协议,而是一个调用接口(API),通过Socket,我们才能使用TCP/IP协议。套接口的概念套接口的概念套接口的本质通信过程中所要使用一些缓冲区和一些相关的数据结构在TCP/IP协议中,“IP地址+TCP或UDP端口号”唯
2、一标识网络通讯中的一个进程,“IP地址+端口号”就称为socket。在TCP协议中,建立连接的两个进程各自有一个socket来标识,那么这两个socket组成的socketpair就唯一标识一个连接。Socket的服务方式和类型流式套接口(SOCK_STREAM)数据报套接口(SOCK_DGRAM)原始套接口(SOCK_RAW)比较项目流式套接字数据报式套接字建立和释放连接建立和释放连接保证数据到达保证数据到达按发送顺序接收数据按发送顺序接收数据通通讯讯数数据据包包含含完完整整的的目目的地址信息的地址信息原始套接字原始套接字是公开的套接字编程接口,使用它可以在IP层上对套接字进行编程,发送和接
3、收IP层上的原始数据包。ICMP、TCP和UDP等协议的数据包。网络字节顺序不同的主机对字节值的存储顺序不同。在存储由多个字节组成的一个字时,计算机在起始地址处存放整数的低序号字节,这种存储格式叫“小序在前”(Little-endian);在存储由多个字节组成的一个字时,计算机在起始地址处存放整数的高序号字节,这种存储格式叫“大序在前”(Big-endian)。网络字节顺序在计算机中,TCP/IP协议使用的16位整数(如端口号)和32位整数(如IP地址)是按计算机各自的“主机字节”(Host-byte)来表示的。在网络中,为了保证数据的正确性,网络通信协议中必须指定网络字节顺序。如果在网络中使
4、用IP地址和端口号,按“互联网联网标准”的要求,指定的多字节值必须用“大序在前”的形式来表示,一般称之为“网络字节”(Network-byte)顺序。4个API函数htonl():参数是主机字节顺序的一个4字节数,函数返回网络字节顺序的数;WSAHtonl():参数是主机字节顺序的一个4字节数,函数返回网络字节顺序的数;htons():参数是主机字节顺序的一个2字节数,函数返回网络字节顺序的数;WSAHtons():参数是主机字节顺序的一个2字节数,函数返回网络字节顺序的数。sockaddr结构体通用结构,用来保存socket信息structsockaddru_shortsa_family;c
5、harsa_data14;sockaddr_in结构体指定IPv4地址结构structsockaddr_inshortsin_family;u_shortsin_port;structin_addrsin_addr;charsin_zero8;in_addr结构体IP地址常用点分法来表示:192.168.0.1计算机中使用无符号长整数(unsigned long)来存储和表示IP地址structin_addrunionstructu_chars_b1,s_b2,s_b3,s_b4;S_un_b;structu_shorts_w1,s_w2;S_un_w;u_longS_addr;S_un;#d
6、efines_addrS_un.S_addr#defines_hostS_un.S_un_b.s_b2#defines_netS_un.S_un_b.s_b1#defines_impS_un.S_un_w.s_w2#defines_impnoS_un.S_un_b.s_b4#defines_lhS_un.S_un_b.s_b3;inet_addr()将点分法IP地址字符串转换为in_addr结构体中的IP地址格式:unsigned long inet_addr(const char*cp);参数cp表示点分法IP地址字符串。如果调用inet_addr()函数时没有出现错误,则函数返回unsig
7、nedlong类型的网络字节顺序格式IP地址;如果参数cp不是有效的IP地址字符串,则inet_addr()函数返回INADDR_NONE。inet_ntoa()将in_addr结构体中的IP地址转换为点分法IP地址字符串charFAR*inet_ntoa(structin_addrin);参数in是in_addr结构体类型,表示要进行转换的IP地址,返回结果为char*类型的IP地址。sa_familysa_datasockaddr结构体s_b1s_b2s_b3s_b4s_w1s_w2S_addrin_addr结构体sin_familysin_portsin_addrsin_zerosock
8、addr_in结构体Winsock版本WinSock1和WinSock2。平台WinSock版本Windows951.1(2.2)Windows982.2WindowsNT4.02.2Windows20002.2WindowsXP2.2套接字的概念与编程原理Winsock API基本函数套接口与连接的建立Winsock API基本函数数据传输Winsock API基本函数套接口与连接的关闭面向连接的客户/服务器程序工作流程以及举例无连接的客户/服务器程序工作流程以及举例Winsock2的使用通常使用WinSock2.2实现网络通信的功能,则需要引用头文件winsock2.h和库文件ws2_32
9、.lib,代码如下:#include#pragmacomment(lib,ws2_32.lib)告诉连接器连接的时候要找ws2_32.libWSAStartup函数作用:启动win_socket的dll库,初始化winsock所对应的ws2_32.dll,完成套接字初始化加载WinSockDLL的相应版本intWSAStartup(WORDwVersionRequested,LPWSADATAlpWSAData);返回一个WSADATA结构WinSock库的版本高位字节指定副版本号低位字节指定主版本号可以用宏MAKEWORD(X,Y)方便地设置WSACleanup()中止WindowsSock
10、etsDLL的使用,卸载所加载的库intWSACleanup(void);返回值成功:返回0失败:返回SOCKET_ERROR.WSAGetLastError()调用任何一个WinSock函数之后可用WSAGetLastError函数来获得详细的错误代码intWSAGetLastError(void);返回值:详细的错误代码不成功的Winsock函数调用返回的最常见的值是宏定义SOCKET_ERRORWSAStartup()函数调用失败能不能用WSAGetLastError()获取详细错误代码?通常依靠应用程序调用WSAGetLastError()机制获得的错误代码是不能使用的,因为Windo
11、wsSocketsDLL可能没有建立“上一错误”信息储存的客户数据区域。Socket()或WSASocket()Winsock1中提供的创建套接口函数的格式:SOCKETsocket(intaf,inttype,intprotocol);在Winsock2中提供的该函数的扩展格式SOCKETWSASocket(intaf,inttype,intprotocol,LPWSAPROTOCOL_INFOlpProtocolInfo,Groupg,intiFlags);创建一个流套接字三种格式SOCKETsockid=socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);SO
12、CKETsockid=WSASocket(AF_INET,SOCK_STREAM,0);SOCKETsockid=WSASocket(AF_INET,SOCK_STREAM,0,NULL,0,WSA_FLAG_OVERLAPPED);创建一个数据报套接口SOCKETsockid=socket(AF_INET,SOCK_GDRAM,IPPROTO_UDP);创建一个原始套接口SOCKETsockid=socket(AF_INET,SOCK_RAW,IPPROTO_ICMP);指定本地地址bind()函数intbind(SOCKET s,const struct sockaddr FAR*name
13、,intnamelen);把一个套接口与一个主机地址和端口号联系起来socket()sockaddr_in结构体bind()、connect()监听连接listen()listen(函数格式:intlisten(SOCKETs,intbacklog);s:标识一个已绑定地址,未建立连接套接字。backlog:指定正在等待连接的最大队列长度。请求连接connect()或WSAConnect()Winsock1中提供的connect()函数:intconnect(SOCKETs,const struct sockaddr FAR*name,intnamelen);Winsock2中提供的扩展格式是
14、:int WSAConnect(SOCKETs,const struct sockaddr FAR*name,intnamelen,LPWSABUFlpCallerData,LPWSABUFlpCalleeData,LPQOSlpSQOS,LPQOSlpGQOS);S:将要建立连接的套接口描述字。name:是一个指向远端套接口地址结构(sockaddr_in)的指针,表示s套接口欲与其建立一条连接。namelen:name名字的长度。接受连接accept()或WSAAccept()Winsock1提供的accept()函数的格式:SOCKETaccept(SOCKETs,structsocka
15、ddrFAR*addr,intFAR*addrlen);Winsock2提供的accept()函数的扩展格式:SOCKETWSAAccept(SOCKETs,structsockaddrFAR*addr,intFAR*addrlen,LPCONDITIONPROC lpfnCondition,DWORDdwCallbackData);s:标识一个套接口描述字,该套接口处于监听连接的状态。addr:是一个地址结构的指针,用来存放发出连接请求的那个客户机的IP地址信息。addr参数的实际格式由套接口创建时所使用的地址族确定。addrlen:该参数指出客户套接口地址结构的长度。该函数就返回一个新的套
16、接口字,它对应于已经接受的那个客户机的连接。对该客户机后续的所有操作,都使用这个新的套接字,成为已连接套接字(ConnectedSocket)。原来那个监听套接字仍然用于接收其他客户机发送的连接请求,仍处于监听模式,因此把它叫监听套接字(ListeningSocket)。套接字的概念与编程原理Winsock API基本函数套接口与连接的建立Winsock API基本函数数据传输Winsock API基本函数套接口与连接的关闭面向连接的客户/服务器程序工作流程以及举例无连接的客户/服务器程序工作流程以及举例在已建立连接的套接口上发送数据send()或WSASend()在Winsock1中提供的格
17、式:Intsend(SOCKETs,constcharFAR*buf,intlen,intflags);在Winsock2中提供的WSASend()函数的扩展格式:intWSASend(SOCKETs,LPWSABUFlpBuffers,DWORDdwBufferCount,LPDWORDlpNumberOfBytesSent,DWORDdwFlags,LPWSAOVERLAPPEDlpOverlapped,LPWSAOVERLAPPED_COMPLETION_ROUTINElpCompletionRoutine);s:用于标识已建立连接的套接口描述字。buf:它是一个字符缓冲区,内有将要发送
18、的数据。len:即将发送的缓冲区中的字符数。flags:用于控制数据传输方式,它可以是0、宏定义MSG_DONTROUTE或MSG_OOB。0表示按正常方式发送数据;MSG_DONTROUTE标志说明系统目标主机就在直接连接的本地网络中,无需路由选择,但如果传输协议的实现不支持该选项,则该标志被忽略;MSG_OOB标志指出数据是按带外数据发送的。在已建立连接的套接口上接收数据recv()或WSARecv()在Winsock1中提供的recv()函数的格式是:intrecv(SOCKETs,charFAR*buf,intlen,intflags);在Winsock2中提供的WSARecv()函数
19、扩展格式是:intWSARecv(SOCKETs,LPWSABUFlpBuffers,DWORDdwBufferCount,LPDWORDlpNumberOfBytesRecvd,LPINTlpFlags,LPWSAOVERLAPPEDlpOverlapped,LPWSAOVERLAPPED_COMPLETION_ROUTINElpCompletionRoutine);s:已建立连接的套接口描述字。buf:用于接收数据的缓冲区。len:缓冲区长度。flags:指定调用方式。flags参数可以是下面的值:0、MSG_PEEK或MSG_OOB。0表 示 接 收 的 是 正 常 数 据,无 特 殊
20、行 为。MSG_PEEK表示会使有用的数据复制到所提供的接收端缓冲区内,但是没有从系统缓冲区中将数据删除。MSG_OOB标志表示处理带外数据。在无连接的套接口上接收数据recvfrom()或WSARecvFrom()Winsock1中提供的recvfrom()函数的格式:intrecvfrom(SOCKETs,charFAR*buf,intlen,intflags,structsockaddrFAR*from,intFAR*fromlen);Winsock2中提供的WSARecvFrom()函数的扩展格式:intWSARecvFrom(SOCKETs,LPWSABUFlpBuffers,DWO
21、RDdwBufferCount,LPDWORDlpNumberOfBytesRecvd,LPINTlpFlags,LPVOIDlpFrom,LPINTlpFromlen,LPWSAOVERLAPPEDlpOverlapped,LPWSAOVERLAPPED_COMPLETION_ROUTINElpCompletionRoutine);s:标识一个套接口的描述字;buf:接收数据缓冲区;len:缓冲区长度;flags:调用操作方式,同recv()中的flags;from:(可选)指针,指向装有源地址的缓冲区,对监听套 接 字 的 具 体 协 议 来 说,from参 数 是 一 个 指 向SOCK
22、ADDR结构的指针,它的长度由指针fromlen指向的数确定,这个API调用返回数据时,SOCKADDR结构内便填入发送数据的那个工作站的地址;fromlen:(可选)指针,指向from缓冲区长度值;在无连接的套接口上发送数据sendto()或WSASendTo()Winsock1中提供的sendto()函数的格式:intsendto(SOCKETs,constcharFAR*buf,intlen,intflags,conststructsockaddrFAR*to,inttolen);Winsock2中提供的WSASendTo()函数的扩展格式:intWSASendTo(SOCKETs,LP
23、WSABUFlpBuffers,DWORDdwBufferCount,LPDWORDlpNumberOfBytesSent,intiFlags,LPVOIDlpTo,intiToLen,LPWSAOVERLAPPEDlpOverlapped,LPWSAOVERLAPPED_COMPLETION_ROUTINElpCompletionRoutine);s:一个标识套接口的描述字;buf:待发送数据的缓冲区;len:指明buf缓冲区中要发送的数据长度;flags:调用方式标志位,同send()中该参数的含义相同;to:指针,指向接收数据的目的套接口的地址;tolen:to所指地址的长度套接字的概念
24、与编程原理Winsock API基本函数套接口与连接的建立Winsock API基本函数数据传输Winsock API基本函数套接口与连接的关闭面向连接的客户/服务器程序工作流程以及举例无连接的客户/服务器程序工作流程以及举例关闭读写通道shutdown()shutdown()函数的格式:intshutdown(SOCKETs,inthow);s:标识一个套接口的描述字;how:用于描述禁止哪些操作标志关闭套接口closesocket()一个套接口不再使用时一定要关闭这个套接口,以释放与该套接口关联的所有资源,包括等候处理的数据。closesocket()函数的格式如下:intclosesoc
25、ket(SOCKETs);套接字的概念与编程原理Winsock API基本函数套接口与连接的建立Winsock API基本函数数据传输Winsock API基本函数套接口与连接的关闭面向连接的客户/服务器程序工作流程以及举例无连接的客户/服务器程序工作流程以及举例面向连接的客户面向连接的客户/服务器程序工作模型服务器程序工作模型面向连接的面向连接的服务器端程序调试环境:VisualC+6.0服务器IP地址:由系统指定服务器端口号:5050程序名称:server.cpp程序功能:服务器端的程序当有客户提出连接请求时,在端口5050与客户端进行TCP连接,连接成功后,显示客户IP地址和端口号,并给
26、客户端发送Hello!Iamaserver.字符串。如果发送正确则显示所发送的字节数命令格式:server面向连接的客户面向连接的客户端程序调试环境:VC+6.0程序名称:client.cpp客户IP地址和端口:由系统指定程序功能:客户端程序向服务器提出TCP连接的请求,当连接建立后,从服务器的端口5050接收数据并进行显示,然后断开与服务器的连接命令格式:client服务器IP地址命令举例:client192.168.0.1说 明:上 面 设 将 要 连 接 的 服 务 器 IP地 址 为192.168.0.1,端口号为5050面向连接的通信过程中五元组的建立套接字的概念与编程原理Winso
27、ck API基本函数套接口与连接的建立Winsock API基本函数数据传输Winsock API基本函数套接口与连接的关闭面向连接的客户/服务器程序工作流程以及举例无连接的客户/服务器程序工作流程以及举例无连接的客户/服务器程序工作模型基于UDP的服务器端程序调试环境:VC+6.0程序名称:server.cpp服务器IP地址:由系统指定服务器端口:5050功能:从端口5050接收客户端发送来的数据,接收成功后显示从客户端收到的数据、客户端的IP地址和端口号;给客户端发送Hello!Iamaserver.字符串命令格式:server基于UDP的客户端程序调试环境:VC+6.0程序名称:clie
28、nt.cpp客户端IP地址和端口:由系统指定程序功能:客户端程序向服务器发送数据Hello!Iamaclient.;客户端程序从服务器接收数据并进行显示命令格式:client服务器IP地址命令举例:client192.168.0.1说明:上面设要给IP地址为192.168.0.1、端口号为5050的服务器发送数据说明:(1)通信的一方可以不用bind()绑定IP地址和端口,而由系统自动分配。(2)不绑定IP地址和端口的一方必须首先向绑定地址的一方发送数据。(3)无连接应用程序也可以调用connect()函数,但是它并不向对方发出建立连接的请求,内核只是将connect()中指定的目标方的IP地址和端口号记录下来,在以后的通信中就可以使用面向连接的数据发送函数send()和数据接收函数revc()。(4)无连接的数据报传输过程中,作为服务器的一方也必须先启动,否则客户请求传不到服务进程。(5)由于无连接的客户端一般不调用connect(),因此在数据发送之前,客户与服务器之间尚未建立一个全相关(五元组),但各自通过socket()和bind()调用建立了半相关。发送数据时,发送方除指定本地套接口的地址外,还需指定接收方套接口的地址,从而在数据收发过程中动态地建立全相关。说明: