Linux下的TCP&UDP通信作业

2023-10-28

题目描述:

  • 一: 对于TCP传输,实现客户端发送某特定信息/字符时结束信息传输,断开连接。
  • 二:编写程序验证在同一台机器上TCP和UDP能不能同时占用同一个端口进行通信。

开始编程:

  • 使用vim创建并编辑几个源文件:
vim TCPserver.c
vim TCPclient.c
vim UDPserver.c 
vim UDPclient.c
  • 分别对应TCP和UDP的服务器端和客户端
    其中:

TCPserver.c


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

int main(int argc,char **argv){
    int sock,accetp_sock,sock_len,numbytes,i;
    struct sockaddr_in server_addr;
    struct sockaddr_in client_addr;
    char buff[BUFSIZ];
    
   
    server_addr.sin_family = AF_INET;
    //绑定端口
    server_addr.sin_port = htons(8220);
    //绑定本地ip(也就是0.0.0.0)
    //表示本机的所有ip(可能本机有多个网卡,这能代表所有网卡)
    server_addr.sin_addr.s_addr = INADDR_ANY;
    //清零
    bzero(&(server_addr.sin_zero),8);
    sock_len = sizeof(struct sockaddr_in);
    printf("start\n");
    sock = socket(AF_INET,SOCK_STREAM,0);
    //绑定
    while(bind(sock,(struct sockaddr *)&server_addr,sock_len)==-1);
    printf("bind success\n");
    while(listen(sock,10)==-1);
    printf("Listening..\n");
    printf("等待来自客户端的连接\n");
    //开始阻塞
    accetp_sock = accept(sock,(struct sockaddr *)&client_addr,&sock_len);
    printf("成功连接客户端\n");
    numbytes = send(accetp_sock,"\nWelecome to Cheney‘server\n",21,0);
    //开始阻塞
    while((numbytes = recv(accetp_sock,buff,BUFSIZ,0))>0){
        buff[numbytes] = '\0';
        printf("%s\n",buff);
        if(send(accetp_sock,buff,numbytes,0)<0){
            perror("write");
            return 1;
        }
    }
    close(accetp_sock);
    close(sock);
    return 0;
}   

TCPclient.c


/***********引入必要的文件**************/
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<errno.h>
#include<netdb.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>

int main(int argc,char **argv){
    int sockfd,numbers;
    char buf[BUFSIZ];
    struct sockaddr_in sockaddress;
    printf("start:\n");
    while((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1);
    printf("get sockfd:\n");
   
    sockaddress.sin_family = AF_INET;
    //绑定端口
    sockaddress.sin_port = htons(8220);
    //连接某个ip
    sockaddress.sin_addr.s_addr = inet_addr(argv[1]);
    //清零
    bzero(&(sockaddress.sin_zero),8);
    //连接
    while(connect(sockfd,(struct sockaddr*)&sockaddress,sizeof(struct sockaddr))==-1);
    printf("成功连接到服务器\n");
    //开始阻塞
    numbers = recv(sockfd,buf,BUFSIZ,0);
    buf[numbers]= '\0';
    printf("%s",buf);
    

    printf("输入要发送的信息:");
    scanf("%s",buf);
    numbers = send(sockfd,buf,sizeof(buf),0);
    //开始阻塞
    numbers = recv(sockfd,buf,BUFSIZ,0);
    buf[numbers]= '\0';
    printf("接收到回信:%s\n",buf);
    
    
    close(sockfd);
    return 0;
}

vim UDPserver.c

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

#define SERVER_PORT 8220
#define BUFF_LEN 1024

void handle_udp_msg(int fd)
{
    char buf[BUFF_LEN];  //接收缓冲区,1024字节
    socklen_t len;
    int count;
    struct sockaddr_in clent_addr;  //clent_addr用于记录发送方的地址信息
    while(1)
    {
        memset(buf, 0, BUFF_LEN);
        len = sizeof(clent_addr);
        count = recvfrom(fd, buf, BUFF_LEN, 0, (struct sockaddr*)&clent_addr, &len);  //recvfrom是拥塞函数,没有数据就一直拥塞
        if(count == -1)
        {
            printf("recieve data fail!\n");
            return;
        }
        printf("client:%s\n",buf);  //打印client发过来的信息
        memset(buf, 0, BUFF_LEN);
        sprintf(buf, "I have recieved %d bytes data!\n", count);  //回复client
        printf("server:%s\n",buf);  //打印自己发送的信息
        sendto(fd, buf, BUFF_LEN, 0, (struct sockaddr*)&clent_addr, len);  //发送信息给client,注意使用了clent_addr结构体指针

    }
}

int main(int argc, char* argv[])
{
    int server_fd, ret;
    struct sockaddr_in ser_addr; 
    //AF_INET:IPV4;SOCK_DGRAM:UDP
    server_fd = socket(AF_INET, SOCK_DGRAM, 0); 
    if(server_fd < 0)
    {
        printf("create socket fail!\n");
        return -1;
    }

    memset(&ser_addr, 0, sizeof(ser_addr));
    ser_addr.sin_family = AF_INET;
 
 
    ser_addr.sin_addr.s_addr = htonl(INADDR_ANY); 
    ser_addr.sin_port = htons(SERVER_PORT);  //端口号,需要网络序转换

    ret = bind(server_fd, (struct sockaddr*)&ser_addr, sizeof(ser_addr));
    if(ret < 0)
    {
        printf("socket bind fail!\n");
        return -1;
    }

    handle_udp_msg(server_fd);   //处理接收到的数据

    close(server_fd);
    return 0;
}

UDPclient.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#define SERVER_PORT 8220
#define BUFF_LEN 512
#define SERVER_IP "127.0.0.1"


void udp_msg_sender(int fd, struct sockaddr* dst)
{

    socklen_t len;
    struct sockaddr_in src;
    while(1)
    {
        char buf[BUFF_LEN] = "TEST UDP MSG!\n";
        len = sizeof(*dst);
        printf("client:%s\n",buf);  //打印自己发送的信息
        sendto(fd, buf, BUFF_LEN, 0, dst, len);
        memset(buf, 0, BUFF_LEN);
        recvfrom(fd, buf, BUFF_LEN, 0, (struct sockaddr*)&src, &len);  //接收来自server的信息
        printf("server:%s\n",buf);
        sleep(1);  //一秒发送一次消息
    }
}



int main(int argc, char* argv[])
{
    int client_fd;
    struct sockaddr_in ser_addr;

    client_fd = socket(AF_INET, SOCK_DGRAM, 0);
    if(client_fd < 0)
    {
        printf("create socket fail!\n");
        return -1;
    }

    memset(&ser_addr, 0, sizeof(ser_addr));
    ser_addr.sin_family = AF_INET;
    //作为客户端,要指定远端服务器的(ip, port)
    ser_addr.sin_addr.s_addr = inet_addr(SERVER_IP);//仅仅连接服务器IP
    ser_addr.sin_port = htons(SERVER_PORT);  //注意网络序转换
    udp_msg_sender(client_fd, (struct sockaddr*)&ser_addr);

    close(client_fd);

    return 0;
}

编辑完以上源文件之后用GCC将其编译为可执行代码:

gcc TCPserver.c -o TCPserver
gcc TCPclient.c -o TCPclient
gcc UDPserver.c -o UDPserver
gcc UDPclient.c -o UDPclient

执行完之后开始运行调试:

TCP 测试:
在这里插入图片描述

可见其可以正常运行,但用户无法连续发送多个消息以及控制何时结束通信,修改部分代码,使其增加退出功能。

这里的修改很简单:只需要将原发送语句增加一个while循环,并令其结束条件为 接受到用户输入‘q’。

char sendline[4096];
printf("输入'q'以结束通信\n");
while(1)
	{
		printf("输入要发送的信息:");
		fgets(sendline,sizeof(sendline),stdin);
		sendline[strlen(sendline)-1] = '\0';

		if(strncasecmp(sendline,"q",1) == 0) {
			printf("已退出\n");
			break;
		}
		
		int sentbytes = send(sockfd,sendline,strlen(sendline),0); 
		printf("sent:%d\n",sentbytes);
	}

修改后:
在这里插入图片描述
可见其满足题1要求,当程序接收到用户输入的‘q’就会自动断开与服务器的连接。

UDP 测试:
在这里插入图片描述

最后验证TCP和UDP两种不同的协议是否可以同时通过相同的端口号进行通信。
由于上述代码已经统一端口号为8220,所以不需修改只需要同时运行两个程序,观察能否同时进行通信即可,测试截图如下:
在这里插入图片描述

可见,TCP和UDP两种协议是可以同时在同一ip通过相同的端口号进行通信的。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Linux下的TCP&UDP通信作业 的相关文章

随机推荐

  • 向上管理(中高层核心能力的表现)

    向上管理 即在工作中为了让公司或上级以及自己取得更好的结果而下意识地配合上级一起工作的过程 在职业生涯中 向上管理其实也是工作能力的一部分 一项重要的职业技能 管理者不仅要做好向下管理 他们还要学会向上管理 1 向下管理 主要涉及的是管理者
  • 基于51单片机霍尔传感器测速(仿真+源程序)

    资料编号 196 下面是该资料仿真演示视频 196 基于51单片机霍尔传感器测速 仿真 源程序 全套资料 功能简介 51单片机计数测速转速测量 在仿真中等价于测量外部脉冲频率 如果修改输入脉冲的频率 在数码管上可实时显示当前频率 功能 霍尔
  • MTCNN人脸及特征点检测---代码应用详解(基于ncnn架构)

    本博记录为卤煮理解 如有疏漏 请指正 转载请注明出处 卤煮 非文艺小燕儿 本博地址 MTCNN人脸及特征点检测 代码应用详解 本文主要讲述当你拿到MTCNN的caffemodel后 如何使用它对一张图里的人脸进行检测和特征点标定 相当于一个
  • Qt信号和槽机制

    Qt信号和槽机制 什么是信号和槽 当某个事件发生 就执行一个操作 发生的事情就是信号 执行的操作就是槽 函数 给二者加上主体 信号发出者发出信号 信号接收者执行操作 将二者联系起来 松耦合 connect 函数 connect sender
  • 网络安全必备1000道面试题集锦(附答案)

    前言 以下为网络安全各个方向涉及的面试题 星数越多代表问题出现的几率越大 祝各位都能找到满意的工作 注 本套面试题 已整理成pdf文档 但内容还在持续更新中 因为无论如何都不可能覆盖所有的面试问题 更多的还是希望由点达面 查漏补缺 一 渗透
  • Error: @vitejs/plugin-vue requires vue (>=3.2.13) or @vue/compiler-sfc to be present in the dependen

    1 没有下载安装axios运行依赖 2 或者缺少这个库没有安装 npm i vue compiler sfc 3 node版本冲不冲突
  • Android Button、TextView等控件使用Toolbar中默认的返回图标

    Button backButton findViewById R id back button backButton setBackgroundResource R drawable abc ic ab back material R dr
  • 网站框架识别方法

    cms一般有dedecms 织梦 dzcms phpweb phpwind phpcms ecshop dvbbs siteweaver aspcms 帝国 zblog wordpress等 一般cms都有特定的文件 只需要识别特定的文件便
  • 运行jetty-maven-plugin时,出现错误

    ERROR Failed to execute goal org eclipse jetty jetty maven plugin 9 4 0 v20161208 run default cli on project kind perm w
  • 【漏洞复现】CVE-2023-22809 sudo提权漏洞

    一 前言 漏洞简介 Sudo中的sudoedit对处理用户提供的环境变量 如SUDO EDITOR VISUAL和EDITOR 中传递的额外参数存在缺陷 当用户指定的编辑器包含绕过sudoers策略的 参数时 拥有sudoedit访问权限的
  • 「iOS」swift 和 objectivec 获得对象的 class 或者 Type 的方法

    一 oc 中使用 oc 中非常简单 一行搞定 NSString str1 test str1 class 这里的 str1 class 就是获取对象 class 的方法 二 swift 中使用 时间紧 任务重 上代码 var str Str
  • Windows 找不到文件 ‘chrome‘。请确认文件名是否正确后,再试一次。

    Windows 找不到文件 chrome 请确认文件名是否正确后 再试一次 错误 当不运行 IDEA 项目 通过快捷键进入浏览器时 可能会出现以下错误 原因 未设置 chrome 路径 导致找不到路径报错 解决办法 1 在桌面上找到 chr
  • 解决qemu虚拟机中内存偏小的问题

    问题描述 最近测试部报了一个问题 云平台中设置大于4GB的内存并设置numa 启动linux2 6 32内核的客户机 之后操作系统中查看实际内存是1 9G 比设置内存小了大概2 1GB 使用版本信息如下 QEMU version 3 0 0
  • 替代MySQL半同步复制,Meta技术团队推出MySQL Raft共识引擎

    作者 Anirban Rahut Abhinav Sharma Yichen Shen Ahsanul Haque 原文链接 https engineering fb com 2023 05 16 data infrastructure m
  • 关于突破SQL注入限制的新思路

    http www bitscn com network protect 200804 139215 html 突然想我们是否可以用什么方法绕过SQL注入的限制呢 到网上考察了一下 提到的方法大多都是针对AND与 号和 号过滤的突破 虽然有点
  • Mac电脑使用小技巧

    Mac OS很多操作与Windows还是有较大的区别 需要一段时间适应 因此我整理了Mac OS的实用小技巧 希望帮助刚入手Mac的小伙伴轻松上手 一 使用预览直接修改图片大小 在遇到需要修改图片的大小 大家的惯有思维就是用Photosho
  • C语言标准库函数qsort(快速排序函数)

    文章目录 一 函数原型 二 函数解析 比较函数 三 手写快排 四 使用qsort 1 对int数组排序 2 对double数组排序 3 对char数组排序 4 对字符串排序 1 char s 2 char s 5 对结构体排序 1 一级排序
  • 简单一行代码解决流读取导致StringBuilder.toString()乱码问题

    今天做上传功能代码读取返回值得时候 StringBuilder toString 出现乱码的情况 在网上找了一堆乱七八糟的办法 繁琐不说 代码还好几行 后来看到String类构造自己就有办法解决这个问题 简单一行代码搞定 在这里记录一下 S
  • 变分推断(Variational Inference)解析

    一 什么是变分推断 假设在一个贝叶斯模型中 x x x为一组观测变量 z z z为一组隐变量 参数也看做随机变量 包含在 z
  • Linux下的TCP&UDP通信作业

    题目描述 一 对于TCP传输 实现客户端发送某特定信息 字符时结束信息传输 断开连接 二 编写程序验证在同一台机器上TCP和UDP能不能同时占用同一个端口进行通信 开始编程 使用vim创建并编辑几个源文件 vim TCPserver c v