Linux TCP/UDP CS模型
Linux TCP/UDP CS模型
TCP Server/TCP Client
以下是一个使用epoll和getopt的TCP服务器示例。这个服务器会监听指定端口上的连接,并将接收到的数据包内容原样返回给发送者。
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/epoll.h>
#include <sys/socket.h>
#include <unistd.h>
#include<stdio.h>
#include <stdlib.h>
#include<string.h>
#include <errno.h>
#include <getopt.h>
#define BUF_SIZE 1024
#define EPOLL_EVENTS_NUM 10
int main(int argc, char *argv[]) {
int port = 0;
char *bind_ip = NULL;
int opt;
while ((opt = getopt(argc, argv, "p:b:")) != -1) {
switch (opt) {
case 'p':
port = atoi(optarg);
break;
case 'b':
bind_ip = optarg;
break;
default:
fprintf(stderr, "Usage: %s [-p port] [-b bind_ip]\n", argv[0]);
exit(1);
}
}
if (port == 0) {
fprintf(stderr, "Please specify a port with -p option\n");
exit(1);
}
int sockfd;
struct sockaddr_in server_addr, client_addr;
socklen_t client_addr_len = sizeof(client_addr);
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0) {
perror("socket");
exit(1);
}
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
if (bind_ip != NULL) {
if (inet_pton(AF_INET, bind_ip, &server_addr.sin_addr) <= 0) {
perror("inet_pton");
exit(1);
}
} else {
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
}
if (bind(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
perror("bind");
exit(1);
}
if (listen(sockfd, 10) < 0) {
perror("listen");
exit(1);
}
int epollfd = epoll_create1(0);
if (epollfd < 0) {
perror("epoll_create1");
exit(1);
}
struct epoll_event event;
event.events = EPOLLIN;
event.data.fd = sockfd;
if (epoll_ctl(epollfd, EPOLL_CTL_ADD, sockfd, &event) < 0) {
perror("epoll_ctl");
exit(1);
}
struct epoll_event events[EPOLL_EVENTS_NUM];
char buffer[BUF_SIZE];
while (1) {
int n = epoll_wait(epollfd, events, EPOLL_EVENTS_NUM, -1);
for (int i = 0; i < n; i++) {
if (events[i].events & EPOLLERR || events[i].events & EPOLLHUP || !(events[i].events & EPOLLIN)) {
fprintf(stderr, "epoll error\n");
close(events[i].data.fd);
continue;
}
if (events[i].data.fd == sockfd) {
int connfd = accept(sockfd, (struct sockaddr *)&client_addr, &client_addr_len);
if (connfd < 0) {
perror("accept");
exit(1);
}
event.events = EPOLLIN;
event.data.fd = connfd;
if (epoll_ctl(epollfd, EPOLL_CTL_ADD, connfd, &event) < 0) {
perror("epoll_ctl");
exit(1);
}
} else {
ssize_t count = recv(events[i].data.fd, buffer, BUF_SIZE, 0);
if (count < 0) {
perror("recv");
exit(1);
} else if (count == 0) {
printf("Client disconnected\n");
close(events[i].data.fd);
continue;
}
ssize_t send_count = send(events[i].data.fd, buffer, count, 0);
if (send_count < 0) {
perror("send");
exit(1);
}
}
}
}
close(sockfd);
close(epollfd);
return 0;
}
要编译这个程序,请使用以下命令:
gcc -o tcp_server tcp_server.c
然后运行它,指定要监听的端口和绑定的接口(可选):
./tcp_server -p <port> -b <bind_ip>
这个服务器会一直运行,直到你手动终止它。要测试这个服务器,你可以使用nc
(netcat)工具发送TCP数据包。例如,如果服务器正在监听端口12345,你可以运行以下命令发送一个数据包:
echo "Hello, World!" | nc -u localhost 12345
服务器会将接收到的数据包内容原样返回给发送者。
在C语言中,创建一个TCP客户端主要涉及以下几个步骤:
- 创建套接字(socket)。
- 设置服务器的地址信息。
- 连接到服务器(connect)。
- 发送和接收数据。
- 关闭套接字。
以下是一个使用epoll和getopt的TCP客户端示例。这个客户端会连接到指定的服务器,并发送一条消息。然后,它会等待服务器返回的数据包,并将数据包内容打印到屏幕上。
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/epoll.h>
#include <sys/socket.h>
#include <unistd.h>
#include<stdio.h>
#include <stdlib.h>
#include<string.h>
#include <errno.h>
#include <getopt.h>
#define BUF_SIZE 1024
#define EPOLL_EVENTS_NUM 10
int main(int argc, char *argv[]) {
int port = 0;
char *server_ip = NULL;
int opt;
while ((opt = getopt(argc, argv, "p:s:")) != -1) {
switch (opt) {
case 'p':
port = atoi(optarg);
break;
case 's':
server_ip = optarg;
break;
default:
fprintf(stderr, "Usage: %s [-p port] [-s server_ip]\n", argv[0]);
exit(1);
}
}
if (port == 0 || server_ip == NULL) {
fprintf(stderr, "Please specify both port and server_ip with -p and -s options\n");
exit(1);
}
int sockfd;
struct sockaddr_in server_addr;
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0) {
perror("socket");
exit(1);
}
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
if (inet_pton(AF_INET, server_ip, &server_addr.sin_addr) <= 0) {
perror("inet_pton");
exit(1);
}
if (connect(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
perror("connect");
exit(1);
}
int epollfd = epoll_create1(0);
if (epollfd < 0) {
perror("epoll_create1");
exit(1);
}
struct epoll_event event;
event.events = EPOLLIN;
event.data.fd = sockfd;
if (epoll_ctl(epollfd, EPOLL_CTL_ADD, sockfd, &event) < 0) {
perror("epoll_ctl");
exit(1);
}
struct epoll_event events[EPOLL_EVENTS_NUM];
char buffer[BUF_SIZE];
printf("Please enter the message to send: ");
fgets(buffer, BUF_SIZE, stdin);
size_t len = strlen(buffer);
if (buffer[len - 1] == '\n') {
buffer[len - 1] = '\0';
}
ssize_t send_count = send(sockfd, buffer, len, 0);
if (send_count < 0) {
perror("send");
exit(1);
}
while (1) {
int n = epoll_wait(epollfd, events, EPOLL_EVENTS_NUM, -1);
for (int i = 0; i < n; i++) {
if (events[i].events & EPOLLERR || events[i].events & EPOLLHUP || !(events[i].events & EPOLLIN)) {
fprintf(stderr, "epoll error\n");
close(events[i].data.fd);
continue;
}
ssize_t count = recv(sockfd, buffer, BUF_SIZE, 0);
if (count < 0) {
perror("recv");
exit(1);
} else if (count == 0) {
printf("Connection closed by server\n");
close(sockfd);
close(epollfd);
exit(0);
}
printf("Received message from server: %s\n", buffer);
break;
}
}
return 0;
}
要编译这个程序,请使用以下命令:
gcc -o tcp_client tcp_client.c
然后运行它,指定要监听的端口和服务器的IP地址:
./tcp_client -p <port> -s<server_ip>
这个客户端会连接到指定的服务器,并发送一条消息。然后,它会等待服务器返回的数据包,并将数据包内容打印到屏幕上。如果服务器关闭连接,客户端会打印一条消息并退出。
UDP Server/UDP Client
在C语言中实现一个UDP服务器,你需要执行以下步骤:
- 创建一个UDP套接字(socket)。
- 绑定套接字到一个本地地址和端口。
- 接收客户端发送的数据(recvfrom)。
- 发送响应数据给客户端(sendto)。
- 关闭套接字。
以下是一个使用epoll和getopt的UDP服务器示例。这个服务器会监听指定端口上的数据包,并将接收到的数据包内容原样返回给发送者。
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/epoll.h>
#include <sys/socket.h>
#include <unistd.h>
#include<stdio.h>
#include <stdlib.h>
#include<string.h>
#include <errno.h>
#include <getopt.h>
#define BUF_SIZE 1024
#define EPOLL_EVENTS_NUM 10
int main(int argc, char *argv[]) {
int port = 0;
int opt;
while ((opt = getopt(argc, argv, "p:")) != -1) {
switch (opt) {
case 'p':
port = atoi(optarg);
break;
default:
fprintf(stderr, "Usage: %s [-p port]\n", argv[0]);
exit(1);
}
}
if (port == 0) {
fprintf(stderr, "Please specify a port with -p option\n");
exit(1);
}
int sockfd;
struct sockaddr_in server_addr, client_addr;
socklen_t client_addr_len = sizeof(client_addr);
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd < 0) {
perror("socket");
exit(1);
}
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
perror("bind");
exit(1);
}
int epollfd = epoll_create1(0);
if (epollfd < 0) {
perror("epoll_create1");
exit(1);
}
struct epoll_event event;
event.events = EPOLLIN;
event.data.fd = sockfd;
if (epoll_ctl(epollfd, EPOLL_CTL_ADD, sockfd, &event) < 0) {
perror("epoll_ctl");
exit(1);
}
struct epoll_event events[EPOLL_EVENTS_NUM];
char buffer[BUF_SIZE];
while (1) {
int n = epoll_wait(epollfd, events, EPOLL_EVENTS_NUM, -1);
for (int i = 0; i < n; i++) {
if (events[i].events & EPOLLERR || events[i].events & EPOLLHUP || !(events[i].events & EPOLLIN)) {
fprintf(stderr, "epoll error\n");
close(events[i].data.fd);
continue;
}
ssize_t count = recvfrom(sockfd, buffer, BUF_SIZE, 0, (struct sockaddr *)&client_addr, &client_addr_len);
if (count < 0) {
perror("recvfrom");
exit(1);
}
ssize_t send_count = sendto(sockfd, buffer, count, 0, (struct sockaddr *)&client_addr, client_addr_len);
if (send_count < 0) {
perror("sendto");
exit(1);
}
}
}
close(sockfd);
close(epollfd);
return 0;
}
要编译这个程序,请使用以下命令:
gcc -o udp_server udp_server.c
然后运行它,指定要监听的端口:
./udp_server -p <port>
这个服务器会一直运行,直到你手动终止它。要测试这个服务器,你可以使用nc
(netcat)工具发送UDP数据包。例如,如果服务器正在监听端口12345,你可以运行以下命令发送一个数据包:
echo "Hello, World!" | nc -u localhost 12345
服务器会将接收到的数据包内容原样返回给发送者。
以下是一个使用epoll和getopt的UDP客户端示例。这个客户端会向指定的服务器发送数据包,并接收服务器返回的数据包。
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/epoll.h>
#include <sys/socket.h>
#include <unistd.h>
#include<stdio.h>
#include <stdlib.h>
#include<string.h>
#include <errno.h>
#include <getopt.h>
#define BUF_SIZE 1024
#define EPOLL_EVENTS_NUM 10
int main(int argc, char *argv[]) {
int port = 0;
char *server_ip = NULL;
int opt;
while ((opt = getopt(argc, argv, "p:s:")) != -1) {
switch (opt) {
case 'p':
port = atoi(optarg);
break;
case 's':
server_ip = optarg;
break;
default:
fprintf(stderr, "Usage: %s [-p port] [-s server_ip]\n", argv[0]);
exit(1);
}
}
if (port == 0 || server_ip == NULL) {
fprintf(stderr, "Please specify both port and server_ip with -p and -s options\n");
exit(1);
}
int sockfd;
struct sockaddr_in server_addr;
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd < 0) {
perror("socket");
exit(1);
}
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
if (inet_pton(AF_INET, server_ip, &server_addr.sin_addr) <= 0) {
perror("inet_pton");
exit(1);
}
int epollfd = epoll_create1(0);
if (epollfd < 0) {
perror("epoll_create1");
exit(1);
}
struct epoll_event event;
event.events = EPOLLIN;
event.data.fd = sockfd;
if (epoll_ctl(epollfd, EPOLL_CTL_ADD, sockfd, &event) < 0) {
perror("epoll_ctl");
exit(1);
}
struct epoll_event events[EPOLL_EVENTS_NUM];
char buffer[BUF_SIZE];
printf("Please enter the message to send: ");
fgets(buffer, BUF_SIZE, stdin);
size_t len = strlen(buffer);
if (buffer[len - 1] == '\n') {
buffer[len - 1] = '\0';
}
ssize_t send_count = sendto(sockfd, buffer, len, 0, (struct sockaddr *)&server_addr, sizeof(server_addr));
if (send_count < 0) {
perror("sendto");
exit(1);
}
while (1) {
int n = epoll_wait(epollfd, events, EPOLL_EVENTS_NUM, -1);
for (int i = 0; i < n; i++) {
if (events[i].events & EPOLLERR || events[i].events & EPOLLHUP || !(events[i].events & EPOLLIN)) {
fprintf(stderr, "epoll error\n");
close(events[i].data.fd);
continue;
}
ssize_t count = recvfrom(sockfd, buffer, BUF_SIZE, 0, NULL, NULL);
if (count < 0) {
perror("recvfrom");
exit(1);
}
printf("Received message from server: %s\n", buffer);
break;
}
}
close(sockfd);
close(epollfd);
return 0;
}
要编译这个程序,请使用以下命令:
gcc -o udp_client udp_client.c
然后运行它,指定要监听的端口和服务器的IP地址:
./udp_client -p <port> -s<server_ip>
这个客户端会向指定的服务器发送一条消息,然后等待服务器返回的数据包。如果收到服务器返回的数据包,它会将数据包内容打印到屏幕上。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· 【自荐】一款简洁、开源的在线白板工具 Drawnix