TCP通信——基于C语言连接
设计两个程序分别作为服务器和客户端,互相进行连接
- 服务器
/*************************************************************************************************************************
*
* file name: tcp_server.c
* author : Dazz
* date : 2024/6/5
* function : 设计一个程序作为服务器,对客户端进行连接
* note : ./a.out port addr
*
* CopyRight (c) 2024 Dazz_24@163.com All Right Reseverd
*
* **************************************************************************************************************************/
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <errno.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/udp.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
// TCP服务器代码 ./xxx port
int main(int argc, char const *argv[])
{
// 判断参数有效性
if (argc != 2)
{
fprintf(stderr, "argument is invaild,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 1.创建TCP套接字
int tcp_socket = socket(AF_INET, SOCK_STREAM, 0);
if (tcp_socket == -1)
{
fprintf(stderr, "tcp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 2.绑定自身的IP地址和端口
struct sockaddr_in host_addr;
host_addr.sin_family = AF_INET; // 协议族,是固定的
host_addr.sin_port = htons(atoi(argv[1])); // 目标端口,必须转换为网络字节序
host_addr.sin_addr.s_addr = htonl(INADDR_ANY); // 目标地址 INADDR_ANY 这个宏是一个整数,所以需要使用htonl转换为网络字节序
bind(tcp_socket, (struct sockaddr *)&host_addr, sizeof(host_addr));
// 3.设置监听 队列最大容量是5
listen(tcp_socket, 5);
// 4.等待接受客户端的连接请求
struct sockaddr_in client;
socklen_t client_len = sizeof(client);
int connect_fd = accept(tcp_socket, (struct sockaddr *)&client, &client_len); // 会阻塞
char buf[128] = {0}; // 接收缓存区
// 5.说明双方建立连接,此时可以接收数据
// while(1)
//{
// 接收客户端的数据
read(connect_fd, buf, sizeof(buf));
printf("recv from [%s],data is = %s\n", inet_ntoa(client.sin_addr), buf);
bzero(buf, sizeof(buf)); // 清空缓存区
// }
// read(connect_fd, buf, sizeof(buf));
// printf("recv from [%s],data is = %s\n", inet_ntoa(client.sin_addr), buf);
// read(connect_fd, buf, sizeof(buf));
// 关闭套接字文件
close(tcp_socket);
return 0;
}
- 客户端
/**************************************************************************************************************************************
*
* 字节序:数据以字节流的方式进行传输,底层都是采用二进制,字节流的顺序是由架构决定的,现在假设使用X86架构,是采用小端存储
* 网络字节序:
* 本地字节序:
*
* 大端存储:低地址存储高字节 当数据超过1个字节的时候才需要区分大端还是小端 假设int型 0x12345678 0x12 | 0x34 | 0x56 | 0x78
* 小端存储:低地址存储低字节 当数据超过1个字节的时候才需要区分大端还是小端 假设int型 0x12345678 0x78 | 0x56 | 0x34 | 0x12
*
* 设备A采用X86架构,所以设备A采用小端存储 待发送的数据 :0x12345678
* 设备B采用ARM架构,所以设备B采用大端存储 待接收的数据 :?
*
* 为了统一发送数据的格式,所以互联网传输的数据统一采用大端方式,为了方便开发,linux系统提供了转换的接口:htonl、htons、ntohl、ntohs
*
* h :host 主机/本地
* to: 转换
* n :net 网络
* l :long 长整型
* s :short 短整型
*
* htons:把本地字节序的一个短整型转换为网络字节序
* htonl:把本地字节序的一个长整型转换为网络字节序
* ntohs:把网络字节序的一个短整型转换为本地字节序
* ntohl:把网络字节序的一个长整型转换为本地字节序
* ***********************************************************************************************************************************/
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <errno.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/udp.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char *argv[])
{
// 检查参数有效性
if (argc != 3)
{
fprintf(stderr, "argument is invaild ,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 1.创建UDP套接字
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 2.需要先绑定服务器的端口和地址
struct sockaddr_in host_addr;
host_addr.sin_family = AF_INET; // 协议族,是固定的
host_addr.sin_port = htons(atoi(argv[1])); // 目标端口,必须转换为网络字节序
// host_addr.sin_addr.s_addr = inet_addr(argv[2]); // 目标地址 "192.168.64.xxx" 已经转换为网络字节序 INADDR_ANY
host_addr.sin_addr.s_addr = htonl(INADDR_ANY); // 目标地址 "192.168.64.xxx" 已经转换为网络字节序 INADDR_ANY
bind(udp_socket, (struct sockaddr *)&host_addr, sizeof(host_addr));
// 3.调用recvfrom等待接收数据,并且接收客户端的网络信息
char buf[128] = {0};
struct sockaddr_in client;
socklen_t client_len = sizeof(client);
while (1)
{
recvfrom(udp_socket, buf, sizeof(buf), 0, (struct sockaddr *)&client, &client_len); // 默认会阻塞
printf("client ip is [%s],time is [%s]\n", inet_ntoa(client.sin_addr), buf);
bzero(buf, sizeof(buf));
}
return 0;
}
合集:
网络编程
分类:
网络编程 / TCP
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY