本文主要介绍Linux C++ 基础Socket网络编程。 大部分知识来自于网站:https://www.geeksforgeeks.org/socket-programming-cc/
Socket编程状态图
从图中可以看到,服务端这边需要处理四步才能进入等待连接的状态,而客户端只要两步。
Socket编程中各函数简单解析
本解析仅为自己理解所用,可能有些纰漏,有则改之。 原文中的知识总结得比我更好,尽量参考原文,我的理解仅做辅助之用。
服务端
先说服务端。服务端需要指定好端口并监听,所以需要bind()绑定好端口,需要listen()进入监听状态,然后通过accept()阻塞等待客户端的消息。
引用表:
- #include <sys/socket>
- socket()
- setsockopt()
- bind()
- listen()
- accept()
- #include <netinet/in.h>
- struct sockaddr_in
- #include <unistd.h>
- read()
- #include <arpa/inet.h>
- inet_pton()
socket()
这个函数是用来创建一个socket,3个参数中,需要特别传的就是前两个。返回一个socket编号,是个int值。
int sockfd = socket(domain, type, protocol)
domain: IPV4 用 AF_INET, IPV6 用 AF_INET6 type: TCP 用 SOCK_STREAM, UDP 用 SOCK_UGRAM
setsockopt()
这个函数用来给上面那个socket()函数返回的socket设置属性,作为服务端,为了方便? 可以设置重用地址和端口号。
int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen); 为了重用地址和端口号,需要这么做:
- level传SOL_SOCKET,代表你这次设置的属性值是给哪个模块用的
- optname传SO_REUSEADDR|SO_REUSEPORT,代表你打算同时设置这两个属性
- optval传一个int*指针,指向某一个数字
- optlen传sizeof()上面的optval
C++ socket很多函数都需要你再传一个length长度,以确定你真正想传给这个函数的数据是多长。
sockaddr_in
那么地址在socket编程中是怎么表示的呢?是使用struct sockaddr_in来定义的。 用的时候需要设置3个值:sin_family, sin_addr的s_addr, sin_port
- sin_family:和前面socket()的domain一样,AF_INET
- sin_addr.s_addr:这个属性是将我们的点分十进制ip地址转化为一个数字,需要使用专门的函数来处理,比如inet_pton()
- sin_port: 指定端口号,但是不是直接传一个int数字,需要用htons转成16进制的数字
bind()
这个函数用来给socket绑定地址信息。
上一节已经说了地址怎么设置,这一节讲socket绑定地址。
int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
照例将sockFd和address关联即可。
注意 这里的addr的类型是struct sockaddr * 而不是struct sockaddr_in *。 struct sockaddr的结构里并没有提供存放ip地址,端口号的属性,所以需要用struct sockaddr_in来强制类型转换。 在文档中,作者说struct sockaddr和struct sockaddr_in长度一定是一样的,所以一定可以强制类型转换,让大家不要担心。 https://www.gta.ufrj.br/ensino/eel878/sockets/sockaddr_inman.html 注意,后面的addrlen是一个值,不是指针。
listen()
int listen(int sockfd, int backlog); 这个函数将socket切换为被动模式,进入监听状态。第二个参数backlog指定消息等待队列的最大长度。
accept()
int new_socket= accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen); 函数执行之后,socket就会等待客户端的连接。当连接建立之后,返回一个用于通信的新的socket,这个新的socket用于客户端与服务端之间的通信。
注意:accept()的第三个参数socklen_t *addrlen和bind()的第三个参数socklen_t addrlen不一样,accept需要一个指针。 有人说是accept()的参数是双向参数,会更新地址的长度值,但也有人说是在accept()函数里,它不知道int能不能存的下这个长度,万一长度特别大就不好存了,为了统一存储结构,仅传一个指针即可。
send()
send()用于通过socket来给对方发送消息。
read()
read()函数是放在#include <unistd.h>中的。函数通过一个char数组来存储接收到的消息
客户端
再说服务端。客户端需要指定好ip地址和端口号,然后发起连接。
connect()
int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
客户端也要创建一个socket,指定sockaddr_in然后强转。这个socket不仅负责发起连接,也负责发送,接收数据。
示例代码(没有错误处理)
目前我只想学习这些socket底层api,所以不想浪费过多精力去记住api的返回值可能意味着什么错误,仅专注于能正常实现一个最简单的server和client。
服务器server端
server.cpp
#include <iostream>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <string.h>
using namespace std;
#define PORT 8000
int main() {
int sockFd, newSockFd, valread;
int opt = 1;
char buffer[1024] = {0};
char* helloFromServer = "hello from server";
struct sockaddr_in address;
sockFd = socket(AF_INET, SOCK_STREAM, 0);
setsockopt(sockFd, SOL_SOCKET, SO_REUSEADDR|SO_REUSEPORT, &opt, sizeof(opt));
address.sin_family = AF_INET;
address.sin_addr.s_addr = INADDR_ANY;
address.sin_port = htons(PORT);
int addrlen = sizeof(address);
bind(sockFd, (struct sockaddr*)&address, addrlen);
listen(sockFd, 3);
newSockFd = accept(sockFd, (struct sockaddr*)&address, (socklen_t*)&addrlen);
read(newSockFd, buffer, 1024);
printf("receive: %s\n", buffer);
send(newSockFd, helloFromServer, strlen(helloFromServer), 0);
printf("server sent message\n");
return 0;
}
客户端client端
client.cpp
#include <iostream>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#define PORT 8000
using namespace std;
int main() {
int sockFd = 0;
char buffer[1024] = {0};
char* helloFromClient = "hello from client";
struct sockaddr_in address;
address.sin_family = AF_INET;
inet_pton(AF_INET, "127.0.0.1", &address.sin_addr.s_addr);
address.sin_port = htons(PORT);
sockFd = socket(AF_INET, SOCK_STREAM, 0);
connect(sockFd, (struct sockaddr*)&address, sizeof(address));
send(sockFd, helloFromClient, strlen(helloFromClient), 0);
printf("client sent\n");
read(sockFd, buffer, 1024);
printf("read message:%s\n", buffer);
return 0;
}
编译、运行和目标输出
在linxu命令行下,分别输入:
g++ -o server server.cpp
g++ -o client client.cpp
然后开两个控制台,分别输入:
./server
./client
目标输出为: server输出:
receive: hello from client
server sent message
client输出:
client sent message
receive: hello from server