测试Linux下socket每次发送数据和接受数据的最大值

经试验发现,客户端一次发送数据大小有个上线,本机试验数据最大是384KB,超出最大值的时候,客户端会阻塞到发送语句哪里等待服务端接受数据。服务多也是一样,有一个最大的一次性接受数据的上线,本机试验最大接受数据是64KB,但是这个服务端好像不确定具体接受多少

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <pthread.h>

#define MAXBUF 1024 * 1024

void* recvsocket(void *arg)
{
	int st = *(int *)arg;
	char s[1024];

	while (1)
	{
		memset(s, 0, sizeof(s));
		int rc = recv(st, s, sizeof(s), 0);
		if (rc <= 0)
			break;
		printf("%s\n", s);
	}
	return NULL;
}

void* sendsocket(void *arg)
{
	int st = *(int *)arg;
	char s[1024];
	char *p = malloc(MAXBUF);//1M

	while (1)
	{
		memset(s, 0, sizeof(s));
		read(STDIN_FILENO, s, sizeof(s));
		int rc = send(st, p, MAXBUF, 0);
		printf("rc = %d\n", rc);
	}

	return NULL;
}


int main(int arg, char *args[])
{
	if (arg < 3)
	{
		return -1;
	}

	int port = atoi(args[2]);
	int st = socket(AF_INET, SOCK_STREAM, 0);//初始化socket

	struct sockaddr_in addr; // 定义一个IP地址的结构
	memset(&addr, 0, sizeof(addr));
	addr.sin_family = AF_INET;// 设置结构地址类型为TCP/IP地址
	addr.sin_port = htons(port); // 制定一个端口号:8080,htons:将short类型从host字节类型转到net字节类型
	// 将字符类型的IP地址转化为int,赋给addr结构
	//addr.sin_addr.s_addr = inet_addr("127.0.0.1");
	addr.sin_addr.s_addr = inet_addr(args[1]);

	//
	if (connect(st, (struct sockaddr *)&addr, sizeof(addr)) == -1)
	{
		printf("connect failed %s\n", strerror(errno));
		return EXIT_FAILURE;
	}

	pthread_t thrd1, thrd2;

	pthread_create(&thrd1, NULL, recvsocket, &st);
	pthread_create(&thrd2, NULL, sendsocket, &st);
	pthread_join(thrd1, NULL);
	//pthread_join(thrd2, NULL);// 加上这一句话会导致线程如法退出,主线程会一直等待子线程,子线程一直是退步出来的

	close(st);
	return EXIT_SUCCESS;
}


#define MAXBUF 1024 * 128

struct ps
{
	int st;
	pthread_t *thr;
};

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

int status = 0;

void* recvsocket(void *arg)
{
	struct ps *p = (struct ps *)arg;

	int st = p->st;
	char s[1024];
	char *pp = malloc(MAXBUF);
	while (1)
	{
		char buf[100];
		read(STDIN_FILENO, buf, sizeof(buf));
		memset(s, 0, sizeof(s));
		int rc = recv(st, pp, MAXBUF, 0);
		printf("rc = %d\n", rc);
		if (rc <= 0)//如果recv返回小于等于0,代表socket已经关闭或者出错了
			break;
		//printf("%s\n", s);
	}
	pthread_mutex_lock(&mutex);
	status = 0;
	pthread_mutex_unlock(&mutex);
	pthread_cancel(*(p->thr));
	return NULL;
}

void* sendsocket(void *arg)
{
	int st = *(int *)arg;
	char s[1024];

	while (1)
	{
		memset(s, 0, sizeof(s));
		read(STDIN_FILENO, s, sizeof(s));
		send(st, s, strlen(s), 0);
	}

	return NULL;
}


int main(int arg, char *args[])
{
	if (arg < 2)
	{
		return -1;
	}
	int port = atoi(args[1]);
	int st = socket(AF_INET, SOCK_STREAM, 0); //初始化socket

	int on = 1;
	if (setsockopt(st, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) == -1)
	{
		printf("setsockopt failed %s\n", strerror(errno));
		return EXIT_FAILURE;
	}

	struct sockaddr_in addr; // 定义一个IP地址结构
	memset(&addr, 0, sizeof(addr));
	addr.sin_family = AF_INET; // 将addr结构的属性定位为TCP/IP地址
	addr.sin_port = htons(port); // 将本地字节顺序转化为网络字节顺序
	addr.sin_addr.s_addr = htonl(INADDR_ANY); // INADDR_ANY代表这个server上所有的地址

	// 将ip与server程序绑定
	if (bind(st, (struct sockaddr *) &addr, sizeof(addr)) == -1)
	{
		printf("bind failed %s\n", strerror(errno));
		return EXIT_FAILURE;
	}

	// server端开始listen
	if (listen(st, 20) == -1)
	{
		printf("listen failed %s\n", strerror(errno));
		return EXIT_FAILURE;
	}
	//printf("listen success\n");
	int client_st = 0;
	struct sockaddr_in client_addr; // 表示client端的IP地址

	pthread_t thrd1, thrd2;
	while (1)
	{
		memset(&client_addr, 0, sizeof(client_addr));
		socklen_t len = sizeof(client_addr);
		client_st = accept(st, (struct sockaddr*) &client_addr, &len);

		pthread_mutex_lock(&mutex);
		status++;
		pthread_mutex_unlock(&mutex);
		if (status > 1)
		{
			close(client_st);
			continue;
		}
		if (client_st == -1)
		{
			printf("accept failed %s\n", strerror(errno));
			return EXIT_FAILURE;
		}
		printf("accept by %s\n", inet_ntoa(client_addr.sin_addr));
		struct ps ps1;
		ps1.st = client_st;
		ps1.thr = &thrd2;
		pthread_create(&thrd1, NULL, recvsocket, &ps1);
		pthread_detach(thrd1);//设置线程为可分离
		//pthread_create(&thrd2, NULL, sendsocket, &client_st);
		//pthread_detach(thrd2);//设置线程为可分离
	}
	close(st);
	return EXIT_SUCCESS;
}



  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 首先,需要在ESP8266上运行一个socket服务器程序,用于接收来自Python客户端的数据。可以使用Micropython或Arduino IDE编写此程序。以下是一个简单的Micropython示例: ```python import socket # 设置WiFi连接 ssid = 'your_ssid' password = 'your_password' wlan = network.WLAN(network.STA_IF) wlan.active(True) wlan.connect(ssid, password) # 创建socket服务器 server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) server_socket.bind(('', 1234)) # 绑定IP和端口 server_socket.listen(1) # 启动监听 # 接收客户端连接并处理数据 while True: client_socket, addr = server_socket.accept() print('Got a connection from %s' % str(addr)) data = client_socket.recv(1024) print('Received: %s' % data) client_socket.close() ``` 然后,可以编写Python客户端程序,与ESP8266建立连接并发送数据。以下是一个简单的Python示例: ```python import socket # 连接ESP8266服务器 server_ip = 'esp8266_ip' server_port = 1234 client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) client_socket.connect((server_ip, server_port)) # 发送数据给ESP8266 data = b'Hello ESP8266!' client_socket.send(data) # 关闭客户端socket client_socket.close() ``` 在运行Python客户端程序之前,需要确保ESP8266已经连接到WiFi网络,并且socket服务器已经启动。 ### 回答2: Python使用socket通信与ESP8266模块进行数据的发送和接收是一种非常常见且强大的方式。Python作为一种高级语言,提供了丰富的库和函数,使得实现socket通信变得相对容易。 要使用Python与ESP8266进行通信,你需要完成以下几个步骤: 1. 在ESP8266模块上运行一个类似于WebSocket的服务器。这可以使用Micropython或Arduino IDE编写一个简单的代码脚本来实现。该服务器将监听一个特定的端口(例如8000),并等待来自Python脚本的连接请求。 2. 在Python脚本中,你需要使用Python的socket模块来建立与ESP8266服务器的连接。你需要指定ESP8266服务器的IP地址和端口号。使用socket的connect函数连接到ESP8266服务器。 3. 一旦连接建立,你可以使用socket的send函数向ESP8266发送数据。你可以发送字符串、整数或字节数据。在ESP8266服务器端,你需要解析接收到的数据并进行相应的处理。 4. ESP8266收到数据后,可以使用它的WiFi模块将数据发送到其他设备或执行其他操作。你可以根据自己的需求来设计ESP8266的功能。 5. 在Python脚本中,你可以使用socket的recv函数来接收来自ESP8266的数据。你可以通过自己的逻辑处理来解析接收到的数据。 使用Python的socket模块与ESP8266进行通信可以实现很多功能,比如远程控制和数据传输等。但需要注意的是,确保你的ESP8266与网络连接正常,并且你的Python脚本与ESP8266之间的网络通路是可达的。 ### 回答3: Python是一种流行的编程语言,与ESP8266-01S一起使用非常方便。ESP8266-01S是一款低功耗的Wi-Fi模块,可以通过Wi-Fi连接到互联网。下面是如何使用Python的socket通信与ESP8266-01S进行数据的发送和接收的简要说明。 首先,要确保ESP8266-01S正确连接到电脑,并已正确设置Wi-Fi连接。接下来,在Python中使用socket模块来创建一个TCP套接字。首先导入socket模块: ```python import socket ``` 然后,使用`socket.socket()`方法来创建套接字对象: ```python s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) ``` 这里使用的是TCP套接字,`socket.AF_INET`表示使用IPv4协议。接下来,我们需要设置服务器地址和端口号: ```python host = 'ESP8266的IP地址' port = ESP8266的端口号 ``` 接着,使用`connect()`方法来建立与ESP8266-01S的连接: ```python s.connect((host, port)) ``` 现在,我们可以发送数据给ESP8266-01S了。可以使用`sendall()`方法来发送数据: ```python s.sendall('Hello ESP8266-01S'.encode()) ``` 这是一个简单的例子,向ESP8266-01S发送了一条Hello消息。注意要将字符串转换为字节码。然后,我们可以使用`recv()`方法来接收ESP8266-01S发送回来的数据: ```python data = s.recv(1024) print(data.decode()) ``` 这里,`recv()`方法的参数是缓冲区的大小。最后,不要忘记关闭套接字连接: ```python s.close() ``` 以上是使用Python的socket通信与ESP8266-01S进行数据发送和接收的基本步骤。根据实际需要,还可以使用其他socket方法来实现更复杂的通信功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值