首页 > 其他分享 >TCP通信——基于C语言连接

TCP通信——基于C语言连接

时间:2024-06-06 09:12:04浏览次数:11  
标签:addr TCP C语言 host socket include buf 连接 字节

设计两个程序分别作为服务器和客户端,互相进行连接

  • 服务器
/*************************************************************************************************************************
 *
 *	file name:	tcp_server.c
 *	author	 :  Dazz
 *	date	 :  2024/6/5
 *	function :  设计一个程序作为服务器,对客户端进行连接
 *  note     :  ./a.out   port   addr
 *
 *	CopyRight (c)  2024   [email protected]   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;
}

标签:addr,TCP,C语言,host,socket,include,buf,连接,字节
From: https://www.cnblogs.com/Dazz24/p/18234392

相关文章

  • SQL Server如何判断哪些会话/连接是长连接?
    一般数据库中可能存在长连接或短连接会话。长连接是相对于通常的短连接而说的,也就是长时间保持客户端与服务端的连接状态。如果不从应用程序入手,如何判断SQLServer数据库中哪些会话是长连接呢?在SQLServer中有连接与会话的概念,一般而言,连接是物理概念,而会话则是逻辑上的概念。两......
  • 使用 Winsock 实现简单的 TCP 服务器和客户端教程
    使用Winsock实现TCP服务器和客户端在这篇博客中,我们将介绍如何使用WinsockAPI在Windows上实现简单的TCP服务器和客户端。Winsock是Windows套接字扩展,提供了网络编程所需的API。通过这两个示例代码,我们将展示如何初始化Winsock库、创建套接字、进行连接以......
  • TCP协议
    目录传输控制协议(TCP)TCP协议的特点TCP协议的可靠性TCP的连接机制传输控制协议(TCP) 传输控制协议(TCP)用于作为包交换计算机通信网络中的主机以及此类网络的互连系统中的高度可靠的主机对主机协议。TCP协议的特点 TCP协议是面向连接的,端到端可靠的协议,该协议被设计适应支持多网......
  • python 连接sqlite简单示例
    importsqlite3#连接到SQLite数据库#数据库文件是test.db,如果文件不存在,会自动在当前目录创建#如果文件已存在,则连接到该数据库conn=sqlite3.connect('test.db')#创建一个游标对象#通过游标对象来执行SQL语句cursor=conn.cursor()#创建一个表cursor.execut......
  • 完全指南:C语言学习资源汇总
    C语言是编程学习的基石,无论是为了职业发展还是个人兴趣,掌握C语言都是技术生涯的重要一步。为了帮助初学者和有经验的程序员更好地学习和深化对C语言的理解,我们汇总了一系列优秀的书籍和在线资源。这些资源将帮助你从基础知识到高级概念,全面掌握C语言。推荐书籍1.《C语言入门......
  • python 连接sqlite数据库
     首先要打开右边的Database,点击加号,选择图中的选项。  然后找到下载驱动,因为我的已经下载,就不提示了。在File类型后有三个点选择,找到自己创建的数据库。 之后就可以看到了。 实话来说,python确实比Java写的代码少,而且感觉跟容易理解,实现起来也是很直接。 ......
  • C语言数据结构实现-单链表表基本操作
    链表插入元素同顺序表一样,向链表中增添元素,根据添加位置不同,可分为以下3种情况:插入到链表的头部(头节点之后),作为首元节点;插入到链表中间的某个位置;插入到链表的最末端,作为链表中最后一个数据元素;虽然新元素的插入位置不固定,但是链表插入元素的思想是固定的,只需做以下两步操......
  • tcp客户端通信
    网络编程客户端通信代码/***************************************************************************************filename:1.c*author: [email protected]*date:2024/06/05*function: tcp客户端通信*note :none*Copy......
  • 基于调用C语言作为reference_model的UVM对全加器的验证
    代码来自于,本篇文章只对其中的细节做探讨UVM的基本教程-CSDN博客首先DUT为加法器,但是舒服不仅是加数和被加数,还有ready和valid,输出也不止是和,还有valid和ready。valid代表数据有效,ready表示已经准备好发送或者接收。加法器的设计语言中,共有三个状态,INITIAL/WAIT/SEND,rst为1......
  • http1.1 的默认长连接 Connection: keep-alive 与 TCP KeepAlive 之间区别
     HTTP长连接,也称为HTTP持久连接(HTTPPersistentConnection)或HTTP连接重用,是一种在HTTP协议中实现的机制。在传统的HTTP通信中,每个HTTP请求和响应都会伴随着TCP连接的建立和关闭,这在高并发场景下会增加网络开销和延迟。而HTTP长连接则允许客户端和服务器在完成......