Linux应用开发---网络通信

Linux应用开发—网络通信

1 网络通信概述

  Linux下的网络编程,我们一般称为 socket 编程,socket 是内核向应用层提供的一套网络编程接口,我们可以基于socket接口开发自己的网络相关应用程序。

1.1 socket 简介

  套接字(socket)是Linux下的一种进程间通信机制(socket IPC),使用 socket IPC 可以使得在不同主机上的应用程序之间进行通信(网络通信),也可以是同一台主机上的不同应用程序。socket IPC 通常使用客户端-服务器这种模式完成通信,多个客户端可以同时连接到服务器中,与服务器完成数据交互。
  内核向应用层提供了 socket 接口,我们只需要调用 socket 接口开发我们的应用程序即可。socket 是应用层与 TCP/IP 协议通信的中间软件抽象层,它是一组接口,它把复杂的 TCP/IP 协议隐藏在接口后面。因此,我们无需深入理解 TCP/UDP 等复杂的 TCP/IP 协议,socket 已经为我们封装好了,我们只需要遵循 socket 的规定去编程,写出的程序自然遵循 TCP/UDP 标准。

1.2 IP 和 端口

所有的数据传输,都要包含三个元素:源、目的和长度。
在这里插入图片描述

在网络通信中我们使用 “IP 和 端口” 来表示源和目的。

1.3 网络传输中的 2 个对象:server 和 client

  我们访问网站时会涉及到2个对象:网站服务器和浏览器。网站服务器平时安静地呆着,浏览器主动发起数据请求。网站服务器、浏览器可以抽象为 2 个软件的概念:server 程序、client 程序。
在这里插入图片描述

1.4 计算机网络体系结构

计算机网络体系结构分为3种:OSI体系结构(七层),TCP/IP体系结构(四层),五层体系结构。

  • OSI体系结构:概念清楚,理论也比较完整,但是它既复杂又不实用。
  • TCP/IP体系结构:TCP/IP是一个四层体系结构,得到了广泛的运用。
  • 五层体系结构:为了方便学习,折中OSI体系结构和TCP/IP体系结构,综合二者的优点,这样既简洁,又能将概念讲清楚。
    请添加图片描述

TCP/IP与OSI最大的不同在于:OSI是一个理论上的网络通信模型,而TCP/IP则是实际运行的网络协议。

1.4.1 五层网络体系结构概述
  1. 应用层:它是体系结构中的最高层,直接为用户的应用进程(例如电子邮件、文件传输和终端仿真)提供服务。在因特网中的应用层协议很多,如支持万维网应用的 HTTP 协议,支持电子邮件的 SMTP 协议,支持文件传送的 FTP 协议,DNS,POP3,SNMP,Telnet 等等。

  2. 运输层:负责向两个主机中进程之间的通信提供服务。运输层主要使用以下两种协议:

    • 传输控制协议 TCP:面向连接的,数据传输的单位是报文段,能够提供可靠的交付。
    • 用户数据包协议 UDP:无连接的,数据传输的单位是用户数据报,不保证提供可靠的交付,只能提供“尽最大努力交付”。
  3. 网络层:负责将被称为数据包(datagram)的网络层分组从一台主机移动到另一台主机。

  4. 链路层:因特网的网络层通过源和目的地之间的一系列路由器路由数据报。

  5. 物理层:在物理层上所传数据的单位是比特。物理层的任务就是透明地传送比特流。

我们并不需要具体理解这些层,我们只需要使用“运输层”编写应用程序。使用“运输层”时,可以选择 TCP 协议,也可以选择 UDP 协议。

1.4.2 TCP 和 UDP 区别
  1. TCP是面向连接的协议,而UDP是无连接的协议。
  2. TCP提供可靠的数据传输,UDP则不提供可靠性保证。
  3. TCP提供了可靠性保证,它的数据传输速度相对较慢。UDP没有额外的操作,所以传输速度相对较快。
1.4.3 为何存在UDP协议

  既然 TCP 可以提供可靠的数据服务,而 UDP 却不提供,那我们是否可以只用 TCP 呢?
  答案是否定的,举个例子:视频通话时,使用 UDP,偶尔的丢包、偶尔的花屏时可以忍受的;如果使用 TCP,每个数据包都要确保可靠传输,当它出错时就重传,这会导致后续的数据包被阻滞,视频效果反而不好。

1.4.4 TCP/UDP 网络通信交互图

在这里插入图片描述

在这里插入图片描述

2 TCP实现

2.1 服务端

#include <sys/types.h> 
#include <sys/socket.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <signal.h>#define SERVERPORT 8888
#define BACKLOG    10int main(int argc,char* argv[])
{int iSocketServer;int iSocketClient;struct sockaddr_in tSocketServerAddr;struct sockaddr_in tSocketClientAddr;int iRet;int iAddrLen;int iClientNum = -1;int iRecvLen;unsigned char ucRecvBuf[1000];signal(SIGCHLD,SIG_IGN); //解决僵尸进程问题iSocketServer = socket(AF_INET,SOCK_STREAM,0); //打开套接字if(iSocketServer == -1){printf("socket error!\n");return -1;}memset(&tSocketServerAddr, 0x0, sizeof(struct sockaddr_in)); //清零//填充变量tSocketServerAddr.sin_family = AF_INET;tSocketServerAddr.sin_addr.s_addr = htonl(INADDR_ANY);tSocketServerAddr.sin_port = htons(SERVERPORT);//将地址与套接字进行关联、绑定iRet = bind(iSocketServer,(const struct sockaddr *)&tSocketServerAddr,sizeof(struct sockaddr));if(iRet == -1){printf("bind error!\n");return -1;}iRet = listen(iSocketServer,BACKLOG); //监听客户端连接if(iRet == -1){printf("listen error!\n");return -1;}while (1){iAddrLen = sizeof(struct sockaddr);iSocketClient = accept(iSocketServer,(struct sockaddr *)&tSocketClientAddr,&iAddrLen); //接受连接if(iSocketClient != -1){iClientNum++;printf("Get connect from client %d : %s\n",iClientNum,inet_ntoa(tSocketClientAddr.sin_addr));if(!fork()){/********子进程源码**********/while (1){/********接受客户端发来的数据并显示出来**********/iRecvLen = recv(iSocketClient,ucRecvBuf,999,0);if(iRecvLen <= 0){close(iSocketClient);return -1;}else{ucRecvBuf[iRecvLen] = '\0';printf("Get Msg From Client %d: %s\n",iClientNum,ucRecvBuf);}}}}}close(iSocketServer); //关闭套接字return 0;
}

2.2客户端

#include <sys/types.h> 
#include <sys/socket.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <netinet/in.h>
#include <arpa/inet.h>#define SERVERPORT 8888int main(int argc,char* argv[])
{int iSocketClient;struct sockaddr_in tSocketServerAddr;int iRet;unsigned char ucSendBuf[1000];int iSendLen;if(argc != 2){printf("Usage:\n");printf("%s <server_ip>\n", argv[0]);return -1;}iSocketClient = socket(AF_INET,SOCK_STREAM,0); //打开套接字if(iSocketClient == -1){printf("socket error!\n");return -1;}memset(&tSocketServerAddr, 0x0, sizeof(struct sockaddr_in)); //清零//填充变量tSocketServerAddr.sin_family = AF_INET;tSocketServerAddr.sin_port = htons(SERVERPORT);if (0 == inet_aton(argv[1], &tSocketServerAddr.sin_addr)){printf("invalid server_ip\n");return -1;}iRet = connect(iSocketClient,(const struct sockaddr *)&tSocketServerAddr,sizeof(struct sockaddr)); //连接服务器if(iRet == -1){printf("connect error!\n");return -1;}while (1){if (fgets(ucSendBuf, 999, stdin)){iSendLen = send(iSocketClient, ucSendBuf, strlen(ucSendBuf), 0); //发送数据if (iSendLen <= 0){close(iSocketClient);return -1;}}}close(iSocketClient); //关闭套接字return 0;
}

2.3 实验

  1. 编译代码:
gcc -o server ./server.c
gcc -o client ./client.c

在这里插入图片描述

  1. 运行服务端:
./server

在这里插入图片描述

  1. 查看本机可用ip:
ifconfig

在这里插入图片描述

  1. 运行客户端
./client 192.168.217.128

在这里插入图片描述
并且可以看到服务端连接到客户端。
在这里插入图片描述

  1. 发送数据

在这里插入图片描述
在服务器端显示接收到的内容。
在这里插入图片描述

3 UDP实现

3.1 服务端

#include <sys/types.h> 
#include <sys/socket.h>
#include <stdio.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>#define SERVERPORT 8888int main(int argc,char* argv[])
{int iSocketServer;struct sockaddr_in tSocketServerAddr;struct sockaddr_in tSocketClientAddr;int iRet;int iAddrLen;int iRecvLen;unsigned char ucRecvBuf[1000];iSocketServer = socket(AF_INET, SOCK_DGRAM, 0);//打开套接字if (iSocketServer == -1){printf("socket error!\n");return -1;}memset(&tSocketServerAddr, 0x0, sizeof(struct sockaddr_in)); //清零//填充变量tSocketServerAddr.sin_family = AF_INET;tSocketServerAddr.sin_addr.s_addr = htonl(INADDR_ANY);tSocketServerAddr.sin_port = htons(SERVERPORT);//将地址与套接字进行关联、绑定iRet = bind(iSocketServer,(const struct sockaddr *)&tSocketServerAddr,sizeof(struct sockaddr));if(iRet == -1){printf("bind error!\n");return -1;}while (1){iAddrLen = sizeof(struct sockaddr);iRecvLen = recvfrom(iSocketServer,ucRecvBuf,999,0,(struct sockaddr *)&tSocketClientAddr,&iAddrLen);if(iRecvLen > 0){ucRecvBuf[iRecvLen] = '\0';printf("Get Msg From %s : %s\n", inet_ntoa(tSocketClientAddr.sin_addr), ucRecvBuf);}}   close(iSocketServer);return 0;
}

3.2 客户端

#include <sys/types.h> 
#include <sys/socket.h>
#include <stdio.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>#define SERVERPORT 8888int main(int argc,char* argv[])
{int iSocketClient;struct sockaddr_in tSocketServerAddr;int iRet;unsigned char ucSendBuf[1000];int iSendLen;if (argc != 2){printf("Usage:\n");printf("%s <server_ip>\n", argv[0]);return -1;}iSocketClient = socket(AF_INET, SOCK_DGRAM, 0); //打开套接字if (iSocketClient == -1){printf("socket error!\n");return -1;}memset(&tSocketServerAddr, 0x0, sizeof(struct sockaddr_in)); //清零//填充变量tSocketServerAddr.sin_family = AF_INET;tSocketServerAddr.sin_port = htons(SERVERPORT);if (0 == inet_aton(argv[1], &tSocketServerAddr.sin_addr)){printf("invalid server_ip\n");return -1;}iRet = connect(iSocketClient,(const struct sockaddr *)&tSocketServerAddr,sizeof(struct sockaddr)); //连接服务器if(iRet == -1){printf("connect error!\n");return -1;}while (1){if (fgets(ucSendBuf, 999, stdin)){iSendLen = send(iSocketClient, ucSendBuf, strlen(ucSendBuf), 0); //发送数据if (iSendLen <= 0){close(iSocketClient);return -1;}}}close(iSocketClient);return 0;
}

3.3 实验

  1. 编译代码:
gcc -o server ./server.c
gcc -o client ./client.c

在这里插入图片描述

  1. 运行服务端
./server

在这里插入图片描述

  1. 运行客户端
./client 192.168.217.128

在这里插入图片描述

  1. 发送数据

在这里插入图片描述
在服务器端显示接收到的内容。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/456705.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

新数据不影响原来的数据

问题描述 新数据修改时&#xff0c;原来的数据也会受影响 const obj1 ref({ name: slx, age: 20 })const obj2 obj1obj2.value.name hhhhconsole.log(obj1, obj1.value)console.log(obj2, obj2.value)解决方法 (仅适用于对象 在这段代码中&#xff0c;obj1 和 obj2 指向同…

Android9~Android13 某些容量SD卡被格式化为内部存储时容量显示错误问题的研究与解决方案

声明:原创文章,禁止转载! Android9~Android13 某些容量SD卡被格式化为内部存储时容量显示错误问题的研究与解决方案 分析Android11 系统对于EMMC/UFS作为内部存储、SD卡被格式化为内部存储、SD卡/U盘被格式化为便携式存储的不同处理 一.现象描述 实测Android9 Android10 A…

【Qt】常见问题

1.存在未解析的标识符 将build文件夹删掉重新编译。 2.左侧项目目录栏无法删除已添加项目 打开目标项目上一级的pro文件&#xff0c;将目标文件名字注释或者删除掉&#xff0c;最后保存&#xff0c;qt就会自动更新&#xff0c;将该项目隐藏掉。 3.在qt creator下添加槽函数…

mmdetection使用自己的voc数据集训练模型实战

一.自己数据集整理 将labelimg格式数据集进行整理 1.1. 更换图片后缀为jpg import os import shutilroot_path/media/ai-developer/imgfileos.listdir(root_path)for img in file:if img.endswith(jpeg) or img.endswith(JPG) or img.endswith(png):img_pathos.path.join(root…

【PTA选择题/基础夯实/期末复习】链表文件操作

2-1 对于一个头指针为head的带头结点的单链表&#xff0c;判定该表为空表的条件是&#xff08;&#xff09;。 A.headNULL B.head→nextNULL C.head→nexthead D.head!NULL 2-2 链表不具有的特点是&#xff08;&#xff09;。 A.可随机访问任一元素 B.插入、删除不需要移…

Express框架介绍—node.js

Express—Node.js 官网传送门(opens new window) 基于 Node.js 平台&#xff0c;快速、开放、极简的 Web 开发框架 Express 是用于快速创建服务器的第三方模块。 Express 初体验 基本使用 安装 Express&#xff1a; npm install express创建服务器&#xff0c;监听客户端请…

PyTorch 2.2 中文官方教程(十九)

使用 RPC 进行分布式管道并行 原文&#xff1a;pytorch.org/tutorials/intermediate/dist_pipeline_parallel_tutorial.html 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 作者&#xff1a;Shen Li 注意 在github中查看并编辑本教程。 先决条件&#xff1a; PyTorc…

Java13常用类3:Date类

4. Date类1&#xff1a;jdk1.8 之前 jdk1.8 之前的API&#xff1a; System类的currentTimeMillis()&#xff1a;获取当前系统时间两个Date类&#xff1a; SimpleDateFormate类&#xff1a;用于格式化、解析Calendar日历类的使用&#xff1a;抽象类 4.1 两个Date类 时间戳&…

《动手学深度学习(PyTorch版)》笔记6.1

注&#xff1a;书中对代码的讲解并不详细&#xff0c;本文对很多细节做了详细注释。另外&#xff0c;书上的源代码是在Jupyter Notebook上运行的&#xff0c;较为分散&#xff0c;本文将代码集中起来&#xff0c;并加以完善&#xff0c;全部用vscode在python 3.9.18下测试通过&…

Python循环语句——for循环的基础语法

一、引言 在Python编程的世界中&#xff0c;for循环无疑是一个强大的工具。它为我们提供了一种简洁、高效的方式来重复执行某段代码&#xff0c;从而实现各种复杂的功能。无论你是初学者还是资深开发者&#xff0c;掌握for循环的用法都是必不可少的。在本文中&#xff0c;我们…

第三百一十回

文章目录 1. 概念介绍2. 实现方法2.1 hintText2.2 labelText2.3 controller 3. 示例代码4. 内容总结 我们在上一章回中介绍了"如何在输入框中处理光标"相关的内容&#xff0c;本章回中将介绍如何添加输入框默认值.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1.…

【C++从0到王者】第四十一站:特殊类的设计

文章目录 一、设计一个类&#xff0c;不能被拷贝1.C98方法2.C11方法 二、设计一个类&#xff0c;只能在堆上创建对象1.析构函数私有化2.构造函数私有化 三、请设计一个类&#xff0c;只能在栈上创建对象四、设计一个类不能被继承1.C98方式2.C11方式 五、设计一个类&#xff0c;…