长春能开发网站的公司,网站黑链怎么做的,华为建站wordpress,微信公众号菜单跳转网页怎么制作话题回到“黑社会办公室”的例子#xff0c;讲概念已经扯得比较远了#xff0c;不过#xff0c;这一节我们还得讲概念#xff0c;不过好在有些程序的例子。如果大家不想翻回去看TcpServer类的原型#xff0c;我这里直接给出这个头文件的完整源代码#xff1a;//Filename:…话题回到“黑社会办公室”的例子讲概念已经扯得比较远了不过这一节我们还得讲概念不过好在有些程序的例子。如果大家不想翻回去看TcpServer类的原型我这里直接给出这个头文件的完整源代码//Filename: TcpServerClass.hpp #ifndef TCPSERVERCLASS_HPP_INCLUDED #define TCPSERVERCLASS_HPP_INCLUDED #include unistd.h #include iostream #include sys/socket.h #include arpa/inet.h class TcpServer { private: int listenSock; int communicationSock; sockaddr_in servAddr; sockaddr_in clntAddr; public: TcpServer(int listen_port); bool isAccept(); void handleEcho(); }; #endif // TCPSERVERCLASS_HPP_INCLUDED我们已经解释了为什么listenSock和communicationSock的类型是int以及sockaddr_in是什么结构现在来写这个类的构造函数TcpServer::TcpServer(int listen_port) { if ( (listenSock socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) 0 ) { throw socket() failed; } memset(servAddr, 0, sizeof(servAddr)); servAddr.sin_family AF_INET; servAddr.sin_addr.s_addr htonl(INADDR_ANY); servAddr.sin_port htons(listen_port); if ( bind(listenSock, (sockaddr*)servAddr, sizeof(servAddr)) 0 ) { throw bind() failed; } if ( listen(listenSock, 10) 0 ) { throw listen() failed; } }好先看看程序培养一下感觉我们还得说概念。数据封装Data Encapsutation我们前面说到了网络分层链路——网络——传输——应用。数据从应用程序里诞生传送到互联网上每一层都会进行一次封装DataApplicationTCP/UDPIPOS(Driver, Kernel Physical Address)我们用socket重点描述的是协议包括网络协议IP和传输协议TCP/UDP。sockaddr重点描述的是地址包括IP地址和TCP/UDP端口。socket()函数我们从TcpServer::TcpServer()函数可以看到socket和sockaddr的产生是可以相互独立的。socket()的函数原型是int socket(int protocolFamily, int type, int protocol);在Linux中的实现为#include sys/socket.h /* Create a new socket of type TYPE in domain DOMAIN, using protocol PROTOCOL. If PROTOCOL is zero, one is chosen automatically. Returns a file descriptor for the new socket, or -1 for errors. */ extern int socket (int __domain, int __type, int __protocol) __THROW;第一个参数是协议簇Linux里面叫作域意思一样的还是那句话我们这篇教程用到的就仅仅是一个PF_INETprotocol family : internet很多时候你会发现人们也经常在这里赋值为AF_INET事实上当前AF_INET就是PF_INET的一个#define但是写成PF_INET从语义上会更加严谨。这也就是TCP/IP协议簇中的IP协议Internet Protocol网络层的协议。后面两个参数定义传输层的协议。第二个参数是传输层协议类型我们教程里用到的宏只有两个SOCK_STREAM数据流格式和SOCK_DGRAM数据报格式具体是什么我们以后讨论第三个参数是具体的传输层协议。当赋值为0的时候系统会根据传输层协议类型自动匹配和选择。事实上当前匹配SOCK_STREAM的就是TCP协议而匹配SOCK_DGRAM就是UDP协议。所以我们指定了第二个参数第三个就可以简单的设置为0。不过为了严谨我们最好还是把具体协议写出来比如我们的例子中的TCP协议的宏名称IPPROTO_TCP。数据的“地址”从数据封装的模型我们可以看到数据是怎么从应用程序传递到互联网的。我们说过数据的传送是通过socket进行的。但是socket只描述了协议类型。要让数据正确的传送到某个地方必须添加那个地方的sockaddr地址同样要能接受网络上的数据必须有自己的sockaddr地址。可见在网络上传送的数据包是socket和sockaddr共同“染指”的结果。他们共同封装和指定了一个数据包的网络协议IP和IP地址传输协议TCP/UDP和端口号。网络字节和本机字节的相互转换sockaddr结构中的IP地址sin_addr.s_addr和端口号sin_port将被封装到网络上传送的数据包中所以它的结构形式需要保证是网络字节形式。我们这里用到的函数是htons()和htonl()这些缩写的意思是h: host主机本机n: network网络to: to转换s: short16位2字节常用于端口号l: long, 32位4字节常用于IP地址“反过来”的函数也是存在的ntohs()和ntohl()。动作与持续行为本节最后的一个概念可以跟计算机无关。作为动词有些可以描述动作有些是描述一重持续的行为状态的就如同一般动词和be动词一样。扯到C来说我们可以把持续行为封装到函数内部只留出动作的接口。事实上构造函数中的bind()和listen()就是这种描述持续状态的行为函数。