按键盘上方向键 ← 或 → 可快速上下翻页,按键盘上的 Enter 键可回到本书目录页,按键盘上方向键 ↑ 可回到本页顶部!
————未阅读完?加入书签已便下次继续阅读!
态,直到有客户机向该服务器程序提出连接请求。这时,服务器程序被“唤醒”并开
始响应客户机提出的连接请求,双方协商数据由谁来接收和由谁来发送。在数据传输
完毕时,双方再分别关闭连接并释放因创建套接字而占用的资源。
o 无连接的编程模型:在传输数据前,不需要事先进行连接,有数据就进行发送,但却
不对数据的顺序和正确性负责。相对于面向连接的模型,它的传输效率较高,但准确
率稍低。
11。2。2 建立套接字对象
函数 socket()可以创建一个 socket 对象,socket()函数的原型如下:
SOCKET socket(int af; int type; int protocol );
各参数意义如下。
o af :协议的地址家族。在 Windows 操作系统中,它的取值只能是 AF_INET ,表示该套
接字在 Internet 域中进行通信。
o type :套接字类型。当第一个参数 af 取值为 AF_INET 时,它只有 3 种取值,如表 11…1
所示。
o protocol :指定网络协议。具体取值如表 11…2 所示。
如果创建成功,则返回一个创建好的套接字,如果创建失败,则返回 INVALID_SOCKET,
详细信息可以通过调用函数 WSAGetLastError()查看错误信息。
·282 ·
…………………………………………………………Page 294……………………………………………………………
第 11 章 网络编程
表 11…1 套接字类型取值及说明
取值 类型 说明
面向连接、可靠的数据传输服务,数据无差错发送,且按
SOCK_STREAM TCP
发送顺序接收;数据被看作是字节流,无长度限制
无连接服务。数据包以独立包形式发送,不提供无错保证,
SOCK_DGRAM UDP
数据可能丢失,且接收顺序混乱
SOCK_RAW Raw sockets 允许对较低层协议,如 IP、ICMP 直接访问
表 11…2 网络协议取值与套接字类型关系
套接字类型 网络协议
TCP IPPROTO_IP
UDP IPPROTO_UDP
Raw sockets IPPROTO_RAW
11。2。3 绑定地址
创建好 Socket 后,通常要将本地地址附加到所创建的套接字上以便能够有效地标识此套
接字。这个过程由 bind() 函数来实现,它的原型如下:
int bind(SOCKET s; const struct sockaddr FAR *name; int namelen);
各参数意义如下:
o s:要绑定的套接字。
o name :用来赋予套接字地址。
o namelen :name 参数的长度。
其中,sockaddr 结构如下:
struct sockaddr_in
{
short sin_family;
u_short sin_port;
struct in_addr sin_addr;
char sin_zero'8';
};
各参数意义如下。
o sin_family:在 Windows 操作系统中,此值为 AF_INET 。
o sin_port:服务的端口号。
o sin_addr:保存 IP 地址。
o sin_zero'8':其值为 0,只起填充作用。
如果绑定成功,则返回 0 ;出错则返回 SOCKET_ERROR。
11。2。4 建立连接
函数 connect()可以实现客户机和服务器的连接,其原型如下:
int connect(SOCKET s; const struct sockaddr FAR *name; int namelen);
各参数意义如下。
·283 ·
…………………………………………………………Page 295……………………………………………………………
Visual C++ 6。0 程序设计从入门到精通
o s:进行连接的套接字。
o name :要连接服务器的套接字地址结构。
o namelen :name 参数的长度。
如果连接成功,此函数返回 0 ;失败则返回 SOCKET_ERROR。
对于服务器来说,当客户机发来连接请求后,服务器要调用 accept()函数来响应对方的连
接请求,该函数原型如下:
SOCKET accept(SOCKET s; struct sockaddr FAR *addr; int FAR *addrlen);
各参数意义如下。
o s:处在监听(下小节介绍)模式下的套接字。
o addr :在函数调用过程中被填充发出连接请求的客户机的 IP 地址信息。
o addrlen :addr 参数的长度。
如果成功,则返回一个新的套接字,它对应于已经接受的那个客户机连接。对于该客户
机所有的后续操作,都使用这个新的套接字。原来的套接字则仍处于监听模式,继续接受其
他客户机的连接。如果失败,则返回 INVALID_SOCKET 。
11。2。5 监听 socket
对于服务器端来说,在它接受客户机的连接之前,首先要监听。只有进入了监听模式,
才能接受来自客户机的连接。这一点可以通过 listen()函数实现,它的原型如下:
int listen(SOCKET s; int backlog);
各参数意义如下。
o s:进行监听的套接字。
o backlog :正在等待连接的最大队列的长度。如果 backlog 的值为 3,有 4 个客户机同
时发出连接请求,则前 3 个会放在等待连接队列中,最后一个将被忽略。
如果函数成功,则返回 0 ;否则返回 SOCKET_ERROR。
11。2。6 数据传输
当客户机和服务器的连接建立起来以后,便可以进行数据的传输。数据的传输是网络通
信的最终目的,前面所作的工作就是为了客户机可以和服务器传输数据。数据传输又包括数
据发送和数据接收。
1.数据发送
数据发送是通过 send()函数来实现的,它的原型如下:
int send(SOCKET s; const char FAR *buf; int len; int flags );
各参数意义如下。
o s:已经建立连接的套接字。
o buf :字符缓冲区,区内包含即将发送的数据。
o len :缓冲区内的字符数目。
o flags:指定数据传输方式,取值可为 0、MSG_DONTROUTE 和 MSG_OOB ,或者是
·284 ·
…………………………………………………………Page 296……………………………………………………………
第 11 章 网络编程
这些取值进行按位或运算的结果。其中,0 表示无特殊行为;MSG_DONTROUTE 表
示传输层不要将它发出的包路由出去;MSG_OOB 表示数据应该被带外发送。
如果发送成功,则返回发送的字节数,如果失败则返回 SOCKET_ERROR。
2 .数据接收
数据接收通过函数 recv()实现,其原型如下:
int recv(SOCKET s; const char FAR *buf; int len; int flags );
各参数意义如下。
o s:准备接收数据的套接字。
o buf :指向即将接收数据的字符缓冲区的指针。
o len :缓冲区的大小。
o flags:指定传输控制方式,取值可为 0、MSG_PKKE 和 MSG_OOB ,或者是这些取值
进行按位或运算的结果 。其中,0 表示无特殊行为;MSG_PKKE 表示把数据从接收端
口复制到接收缓冲区中,并且没有从系统缓冲区中将数据删除;MSG_OOB 表示数据
是带外发送的。
如果接收成功,则返回接收的字节数,如果失败则返回 SOCKET_ERROR。
11。3 WinSock 类
在上一节中简单介绍了直接利用 WinSock API 进行网络传输的基本步骤以及主要函数的
使用方法,而实际利用 Visual C++ 6。0 开发网络应用程序的时候,很少直接利用这些 API 进
行编程,因为 MFC 已经把这些 API 都封装到 MFC 提供的类中了。本节将详细介绍在网络编
程中经常用到的 MFC 提供的两个类,即 CAsyncSocket 类和 CSocket 类。它们的继承关系如
图 11…1 所示。
图 11…1 CAsyncSocket 类和 CSocket 类的继承关系
11。3。1 CAsyncSocket 类
CAsyncSocket 类在很低的级别上封装了 Windows Socket API ,该类可以使程序员用面向
对象的方法进行 Socket 编程,而且可以非常方便地调用其他 MFC 对象。这个类要求程序员
对 Socket 编程有较为深入的了解,要面对和在直接使用 Windows Socket API 进行程序设计时
一样的问题,如阻塞处理、网络字节顺序等。因为 CAsyncSocket 类几乎是一一对应地封装了
Windows Socket API ,因此具有直接调用 WinSock API 的灵活性。
要使用一个 CAsyncSocket 对象,则先调用它的构造函数,然后调用 Create()函数,以创
建一个套接字句柄(SOCKET 类型)。CAsyncSocket 对象既可以在栈中,也可以在堆中。对
·285 ·
…………………………………………………………Page 297……………………………………………………………
Visual C++ 6。0 程序设计从入门到精通
于一个服务器套接字调用 Listen()成员函数进行监听,对于一个客户套接字调用 Connect()成
员函数来请求连接。在接收一个连接请求时,服务器套接字应该调用一个 Accept() 函数来接
收连接请求。完成之后,如果 CAsyncSocket 对象在栈中构造,则当对象超出范围时,会自动
调用析构函数;如果是在堆上被创建的,必须调用 delete