send()、sendto()和recv()、recvfrom()的使用

send()、sendto()和recv()、recvfrom()的使用udp通讯中的sendto()需要在参数里指定接收方的地址/端口,recvfrom()则在参数中存放接收发送方的地址/端口,与之对应的send()和recv()则不需要如此,但是在调用send()之前,需要为套接字指定接收方的地址/端口(这样该函数才知道要把数据发往哪里),在调用recv()之前,可以为套接字指定发送方的地址/端口,这样该函数就只接收指定的发送方的数据,当然若不指定也可,该函数就可以

大家好,又见面了,我是你们的朋友全栈君。

本文收录于微信公众号「 LinuxOK 」,ID为:Linux_ok,关注公众号第一时间获取更多技术学习文章。

udp通讯中的sendto()需要在参数里指定接收方的地址/端口,recvfrom()则在参数中存放接收发送方的地址/端口,与之对应的send()和recv()则不需要如此,但是在调用send()之前,需要为套接字指定接收方的地址/端口(这样该函数才知道要把数据发往哪里),在调用recv()之前,可以为套接字指定发送方的地址/端口,这样该函数就只接收指定的发送方的数据,当然若不指定也可,该函数就可以接收任意的地址的数据。(这些内容前面文章udp通讯中的connect()和bind()函数
有详细讲过)

这4个函数的使用比较简单,但在一个实例中,遇到一个小问题。
实现功能: udp服务器创建一个套接字接收客户端的连接,连接成功后,服务器再创建一个套接字与客户端进行数据交互,要求尽量使用connect()和recv()、send()函数。

udp服务器代码:

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

#define BUFSZ	1024
#define PORT	6567

int main(void)
{
	int srv_sd, cli_sd;
	int new_sd;
	int ret;
	struct sockaddr_in svr_addr, cli_addr;
	socklen_t addrlen = sizeof(struct sockaddr_in);
	char buf[BUFSZ] = {};
	
	//创建套接字
	if ((srv_sd = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
	{
		perror("socket");
		exit(EXIT_FAILURE);
	}
	
	//为服务器套接字绑定端口
	svr_addr.sin_family = AF_INET;
	svr_addr.sin_port = htons(PORT);
	svr_addr.sin_addr.s_addr = 0;
	if ((ret = bind(srv_sd, (struct sockaddr* )&svr_addr, addrlen)) < 0)
	{
		perror("bind");
		exit(EXIT_FAILURE);
	}
	
	//接收客户端的连接
	ret = recvfrom(srv_sd, buf, BUFSZ, 0, (struct sockaddr* )&cli_addr, &addrlen);
	if (ret < 0)
	{
		perror("recvfrom");
		exit(EXIT_FAILURE);
	}
	printf("server output msg:\n");
	printf("client IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(cli_addr.sin_addr), ntohs(cli_addr.sin_port), buf);
	close(srv_sd);
	
	//创建与客户端数据交互的套接字
	cli_sd = socket(AF_INET, SOCK_DGRAM, 0);
	if (cli_sd < 0)
	{
		perror("socket");
		exit(EXIT_FAILURE);
	}
	
	//为新套接字绑定地址信息
	svr_addr.sin_family = AF_INET;
	svr_addr.sin_port = htons(6666);
	svr_addr.sin_addr.s_addr = 0;
	if ((ret = bind(cli_sd, (struct sockaddr* )&svr_addr, addrlen)) < 0)
	{
		perror("bind");
		exit(EXIT_FAILURE);
	}

	//为新套接字指定目的地址,接下来的数据交互将可以采用recv()和send()
	if ((ret = connect(cli_sd, (struct sockaddr* )&cli_addr, addrlen)) < 0)
	{
		perror("connect");
		exit(EXIT_FAILURE);
	}
	
	//服务器先发数据再收数据
	while (1)
	{
		memset(buf, 0, BUFSZ);
		printf("ple input: ");
		fgets(buf, BUFSZ, stdin);
		//sendto(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, addrlen);
		send(cli_sd, buf, BUFSZ, 0);
		
		ret = recv(cli_sd, buf, BUFSZ, 0);
		printf("server output msg:\n");
		printf("client IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(cli_addr.sin_addr), ntohs(cli_addr.sin_port), buf);
	}
	
	close(cli_sd);
	return 0;
}

udp客户端代码:

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

#define BUFSZ	1024
#define PORT	6567

int main(int argc, char *argv[])
{
	int sd;
	struct sockaddr_in svr_addr, cli_addr;
	int ret;
	socklen_t addrlen = sizeof(struct sockaddr_in);
	char buf[BUFSZ] = {};
	
	//创建套接字
	if ((sd = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
	{
		perror("socket");
		exit(EXIT_FAILURE);
	}
	
	//为套接字绑定本地地址信息
	cli_addr.sin_family = AF_INET;
	cli_addr.sin_port = htons(9693);
	cli_addr.sin_addr.s_addr = 0;
	if ((ret = bind(sd, (struct sockaddr* )&cli_addr, addrlen)) < 0)
	{
		perror("bind");
		exit(EXIT_FAILURE);
	}
	
	//为套接字指定目的地址信息,接下来的与服务器的数据交互就可以使用
	svr_addr.sin_family = AF_INET;
	svr_addr.sin_port = htons(PORT);
	svr_addr.sin_addr.s_addr = inet_addr("192.168.1.166");
	if ((ret = connect(sd, (struct sockaddr* )&svr_addr, addrlen)) < 0)
	{
		perror("connect");
		exit(EXIT_FAILURE);
	}
	
	//数据交互
	while (1)
	{			
		memset(buf, 0, BUFSZ);
		printf("ple input: ");
		fgets(buf, BUFSZ, stdin);
		//sendto(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, addrlen);
		send(sd, buf, BUFSZ, 0);

		memset(buf, 0, BUFSZ);
		//ret = recvfrom(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, &addrlen);
		ret = recv(sd, buf, BUFSZ, 0);
		printf("client output msg:\n");
		printf("server IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(cli_addr.sin_addr), ntohs(cli_addr.sin_port), buf);
	}
	close(sd);
	return 0;
}

运行结果:
这里写图片描述

这里写图片描述

这里写图片描述

观察程序流程,可以得出:
这里写图片描述

客户端发起连接给服务器,服务器接收到后创建新的套接字并调用connect()函数为该套接字指定目标地址信息,这个目标地址信息虽然确实是客户端的,但是客户端的目标地址却是服务器,那么服务器新的套接字的目标地址不是客户端而是服务器,所以服务器发出的数据还是自己收到。

程序的问题出现在客户端,客户端创建了套接字后,就立即为其制定目标(服务器)的地址信息,而这个目标地址信息并非作为接下来数据交互的地址,所以应该把为客户端指定目标地址操作放在服务器创建新的sd之后返回数据到客户端之后,但是注意,客服端创建完套接字后不能马上为其connect()以指定目的地址信息,那么就发数据给服务器时就要使用sendto()、接收数据则是用recvfrom(),流程图改为:
这里写图片描述
客户端代码实现为:

int main(int argc, char *argv[])
{
	int sd;
	struct sockaddr_in svr_addr, cli_addr;
	int ret;
	socklen_t addrlen = sizeof(struct sockaddr_in);
	char buf[BUFSZ] = {};
	
	//创建套接字
	if ((sd = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
	{
		perror("socket");
		exit(EXIT_FAILURE);
	}
	
	cli_addr.sin_family = AF_INET;
	cli_addr.sin_port = htons(9693);
	cli_addr.sin_addr.s_addr = 0;
	if ((ret = bind(sd, (struct sockaddr* )&cli_addr, addrlen)) < 0)
	{
		perror("bind");
		exit(EXIT_FAILURE);
	}
	
	//为sendto()函数的使用指定目标地址
	svr_addr.sin_family = AF_INET;
	svr_addr.sin_port = htons(PORT);
	svr_addr.sin_addr.s_addr = inet_addr("192.168.1.166");
	
	//发送数据
	memset(buf, 0, BUFSZ);
	printf("ple input: ");
	fgets(buf, BUFSZ, stdin);
	sendto(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, addrlen);
	
	//接收数据,此时svr_addr的地址信息是服务端新建的专为数据交互使用的sd
	memset(buf, 0, BUFSZ);
	ret = recvfrom(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, &addrlen);
	printf("client output msg:\n");
	printf("server IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(cli_addr.sin_addr), ntohs(cli_addr.sin_port), buf);
	
	//为套接字绑定目标地址,目标地址是服务端专为数据交互使用的sd
	if ((ret = connect(sd, (struct sockaddr* )&svr_addr, addrlen)) < 0)
	{
		perror("connect");
		exit(EXIT_FAILURE);
	}
	
	//数据交互
	while (1)
	{			
		memset(buf, 0, BUFSZ);
		printf("ple input: ");
		fgets(buf, BUFSZ, stdin);
		//sendto(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, addrlen);
		send(sd, buf, BUFSZ, 0);

		memset(buf, 0, BUFSZ);
		//ret = recvfrom(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, &addrlen);
		ret = recv(sd, buf, BUFSZ, 0);
		printf("client output msg:\n");
		printf("server IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(cli_addr.sin_addr), ntohs(cli_addr.sin_port), buf);
	}
	close(sd);
	return 0;
}

运行结果:
这里写图片描述

结论: connect()和send()、recv()三个函数的搭配使用并不能说一定能代替sendto()、recvfrom(),具体使用还要依据代码场景。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/163414.html原文链接:https://javaforall.net

(0)
上一篇 2022年7月23日 下午6:46
下一篇 2022年7月23日 下午6:46


相关推荐

  • DOS命令:chcp

    DOS命令:chcpchcp 是命令提示符中的一个可执行命令 在命令提示符中运行 可以返回活动代码页的编码 还可以设置当前的活动代码页 chcp 查看官方帮助文档对 CHCP 的解释说明 1 在命令提示符中输入 chcp 命令 显示当前活动代码页编码 2 设置活动代码页 首先我们输入 chcp 命令然后指定活动代码页 437 点击回车 就可以设置活动代码页注意 936 代表中文简体 437 代表美国英语

    2026年3月16日
    2
  • cursor之java入门+Spring ai入门

    cursor之java入门+Spring ai入门

    2026年3月15日
    3
  • 临时表创建_临时表的创建方式

    临时表创建_临时表的创建方式临时表创建//Anhighlightedblock两种临时表的语法:createglobaltemporarytable临时表名oncommitpreserve|deleterows用preserve时就是SESSION级的临时表,用delete就是TRANSACTION级的临时表一、SESSION级临时表1、建立临时表Sql代码createglobal…

    2025年6月30日
    6
  • 国产操作系统: 盘点8款国产Linux桌面操作系统[通俗易懂]

    国产操作系统: 盘点8款国产Linux桌面操作系统[通俗易懂]2014年4月8日起,美国微软公司停止了对WindowsXPSP3操作系统提供服务支持,这引起了社会和广大用户的广泛关注和对信息安全的担忧。在这种背景下,国家出于计算机安全考虑,加大了针对操作系统开发力度,各软件企业纷纷开发自主操作系统。国产操作系统多以Linux为基础二次开发,今天我们为大家盘点一下二零一八年值得用户使用的国产Linux桌面操作系统。1.深度操作系统…

    2022年5月13日
    92
  • linux防ddos攻击工具 DoS Deflate

    linux防ddos攻击工具 DoS DeflateDoSDeflate 是一个轻量级阻止拒绝服务攻击的 bashshell 脚本 我们可以根据自己需要修改特定参数 来达到目的 安装 卸载都很简单 分别执行下面三步就可以了 安装 wgethttp www inetbase com scripts ddos install shchmod0700i sh install sh 卸载 wgethttp www inetbase com scripts ddos uninstall ddoschm

    2026年3月26日
    3
  • Airflow使用入门指南

    Airflow使用入门指南Airflow 能做什么 Airflow 是一个工作流分配管理系统 通过有向非循环图的方式管理任务流程 设置任务依赖关系和时间调度 Airflow 独立于我们要运行的任务 只需要把任务的名字和运行方式提供给 Airflow 作为一个 task 就可以 安装和使用最简单安装在 Linux 终端运行如下命令 需要已安装好 python2 x 和 pip pipinstallai air

    2026年3月20日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号