标签:
socket是进程通信的一种方式,通过调用一些API可以实现进程间通信,建立连接以及收发信息的过程如下图所示:
这些函数的用法如下:
1、int socket(int protocolFamily, int type, int protocol); 返回描述符sockfd
调用socket创建之后,返回的描述符存在于协议族空间中,但没有一个具体的地址,必须要通过bind才行
2、int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
如, IPV4:
struct sockaddr_in { sa_family_t sin_family; /* address family: AF_INET */ in_port_t sin_port; /* port in network byte order */ struct in_addr sin_addr; /* internet address */ }; struct in_addr{ uint32_t s_addr; // address in network byte order };
3、int listen(int sockfd, int backlog);服务器监听函数
listen函数将socket变为被动类型的, 等待客户连接请求
4、int connect(int sockfd, const struct *addr, socklen_t addrlen);
5、int accept(int sockfd, struct sockaddr* addr, socklen_t *addrlen);
accept()成功返回一个SOCKET描述符,表示接收到的套接字的描述符。否则返回值INVALID_SOCKET。
6、ssize_t send(int sockfd, const void *buf, ssize_t len, int flags);
send将自己的数据copy到内核的send buffer,返回值为[-1,size]:
7、ssize_t recv(int sockfd, void *buf, ssize_t len, int flags);
各参数意义基本与send相同,recv操作将内核中的数据拷贝到应用程序内部
返回值是[-1,size]:
8、int close(int fd);
一个echo小例子,客户端向服务器发送什么服务器就给客户端返回什么信息:
1、服务器代码:
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<errno.h> #include<sys/types.h> // 基本系统数据类型,是系统的基本系统数据类型的头文件 #include<sys/socket.h> #include<netinet/in.h> // 互联网地址族 #include<arpa/inet.h> //IP地址转换函数inet_pton #include<unistd.h> //close #include<iostream> int main(int argc, char** argv){ int socketfd, bindfd, connectfd; char buffer[4096]; struct sockaddr_in serverAddress; int sendSize, recvSize; //printf("================== create socket ======================\n"); // create socket socketfd = socket(AF_INET, SOCK_STREAM, 0); if(socketfd == -1){ printf("Create socket error:%s(errno:%d)\n", strerror(errno),errno); exit(0); } //printf("================== set address ========================\n"); //set server address memset(&serverAddress, 0, sizeof(serverAddress)); serverAddress.sin_family = AF_INET; serverAddress.sin_port = htons(800); //serverAddress.sin_addr.s_addr = htonl(127.0.0.1); inet_pton(AF_INET, "127.0.0.1", &serverAddress.sin_addr); std::cout << serverAddress.sin_addr.s_addr << ‘\n‘; //printf("================== bind address ========================\n"); // bind address to the socket bindfd = bind(socketfd, (struct sockaddr*)&serverAddress, sizeof(serverAddress)); if(bindfd == -1){ printf("bind socket error:%s(errno:%d)\n", strerror(errno),errno); exit(0); } //printf("================== listen ========================\n"); if(listen(socketfd, 10) == -1){ printf("listen socket error:%s(errno:%d)\n",strerror(errno),errno); exit(0); } printf("================== waiting connect ========================\n"); while(1){ sleep(2); // recvive a connect and accept connectfd = accept(socketfd, (struct sockaddr*)NULL, NULL); if(connectfd == -1){ printf("connet socket error:%s(errno:%d)\n",strerror(errno),errno); continue; } // receive data recvSize = recv(connectfd, buffer, 4096, 0); if(recvSize == -1){ printf("recvive data error:%s(errno:%d)\n",strerror(errno),errno); continue; } printf("%s\n", buffer); // send data sendSize = send(connectfd, buffer, 4096, 0); if(sendSize == -1){ printf("send data error:%s(errno:%d)\n",strerror(errno),errno); continue; } close(connectfd); } close(socketfd); }
2、客户端代码:
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<errno.h> #include<sys/types.h> // 基本系统数据类型,是系统的基本系统数据类型的头文件 #include<sys/socket.h> #include<netinet/in.h> // 互联网地址族 #include<arpa/inet.h> //IP地址转换函数inet_pton #include<unistd.h> //close #include<iostream> int main(){ int socketfd, connectfd; int sendSize, recvSize; struct sockaddr_in serverAddress; char sendBuf[4096]; char recvBuf[4096]; printf("================== create socket ========================\n"); socketfd = socket(AF_INET, SOCK_STREAM, 0); if(socketfd == -1){ printf("create socket error:%s(error%d)\n", strerror(errno),errno); exit(0); } memset(&serverAddress, 0, sizeof(serverAddress)); serverAddress.sin_family = AF_INET; serverAddress.sin_port = htons(800); //serverAddress.sin_addr.s_addr = htonl("127.0.0.1"); inet_pton(AF_INET, "127.0.0.1", &serverAddress.sin_addr); std::cout << serverAddress.sin_addr.s_addr << ‘\n‘; connectfd = connect(socketfd, (struct sockaddr*)&serverAddress, sizeof(serverAddress)); if(connectfd == -1){ printf("connect server error:%s(error(%d))\n", strerror(errno), errno); exit(0); } printf("send message to server:\n"); //std::cin >> sendBuf; fgets(sendBuf, 4096, stdin); //printf("%s\n", sendBuf); sendSize = send(socketfd, sendBuf, strlen(sendBuf), 0); if(sendSize == -1){ printf("send data error:%s(error%d)\n", strerror(errno), errno); exit(0); } printf("wait echo from server:\n"); sleep(2); recvSize = recv(socketfd, recvBuf, sizeof(recvBuf), 0); if(recvSize < 0){ printf("recvive echo error:%s(error%d)\n", strerror(errno), errno); exit(0); } printf("%s", recvBuf); close(socketfd); }
在这过程中遇到的几个问题:
1、(客户端)errno 111:connection refused
这个问题说明客户端没有找到应该连接的端口,需要做到:
由于我客户端和服务器端口号不匹配,就出现了这个问题
2、(服务器)errno 14:bad address
accept()函数的第二个参数指的是一个接收返回结果的缓存区,表示的是本次连接的客户端的地址,如果无所谓客户端地址可以写为NULL,而一旦给这个参数赋了非NULL得值,但是这个缓存区又不可写的话,就会出现bad address报错
3、(服务器)errno 107: transport endpoint is not connected.
这个问题,是由于我服务器端的recv和send函数的第一个参数写的本地socket描述符造成的,事实上,这应该是已连接的socket描述符
标签:
原文地址:http://www.cnblogs.com/Chilly2015/p/5790787.html