[网络编程】网络编程的基础使用

系列文章目录

1、 初识网络


网络编程套接字

  • 系列文章目录
  • 前言
  • 一、TCP和UDP协议的引入
  • 二、UDP网络编程
    • 1.Java中的UDP
    • 2.UDP回显代码案例
    • 3.UDP网络编程的注意事项
  • 三、TCP网络编程
    • 1.TCP回显代码案例
    • 2.TCP多线程使用
  • 总结


前言

在学习完基础的网络知识后,完成跨主机通信的网络编程。


一、TCP和UDP协议的引入

注意:TCP协议并非TCP/IP模型。
TCP和UDP都属于传输层协议中的一种,为应用层提供服务。但是二者之间的特点和差异都很大,因此有了两套API分别应用于二者。
特点:
TCP:有连接,可靠传输,面向字节流,全双工
UDP:无连接、不可靠传输、面向数据报、全双工

其中有连接的意思是如同打电话一样,在电话接通以后才能相互通信;无连接就像发送短信,不需要“先接通”,允许直接发送。
可靠传输的意思是将要传输的数据尽可能传输给对方,不可靠传输的意思是在数据传输时不关心对方是否收到信息。
面向字节流,和文件操作中的字节流类似,读写操作十分灵活。
面向数据报:在UDP协议中,传输数据的基本单位时一个个UDP数据报,一次读写只能读写一个完整的UDP数据报。
全双工:在一条链路中能够进行双向通信;半双工:一条链路中只能进行单向通信。

二、UDP网络编程

1.Java中的UDP

在Java中,引入系统的API(称为Socket API)并进行封装,在Java中主要有两个类:DatagramSocket和DatagramPacket。
1)DatagramSocket:socke可以理解为是一种文件,socket文件可以视为对网卡的硬件设备进行的抽象化。在UDP中对于操作系统中socket概念的封装,针对该对象进行读写操作,也可以理解为对网卡硬件设备的操作。
2)DatagramPacket: 针对UDP数据报的一个抽象表示,一个DatagramPacket对象,就相当于一个UDP数据报。一次发送/接收就相当于传输了一个DatagramPacket对象。

2.UDP回显代码案例

所谓回显代码,即客户端请求什么内容,服务器就返回什么代码。回显代码中没有实现复杂的逻辑,能够最简单的反映UDP的服务器和客户端之间的互动情况。

  1. UDPEchoServer:在Server端,端口号是固定的,DaragramSocket 在构造方法中传入端口号。在服务器启动之后需要不断接收请求和返回请求值。
    在DatagramSocket中的receive方法负责接收客户端发送过来的请求, 将接收到的数据转换成字符串request,作为后面处理请求的参数。通过方法process运行之后返回的结果需要数据报进行封装,因此在结果返回之后,新建一个数据报对象并将字符串结果转换成字节流。通过socket.send方法发送。
public class UdpEchoServer {DatagramSocket socket = null;public UdpEchoServer(int port) throws SocketException {socket = new DatagramSocket(port);}public void start() throws IOException {System.out.println("服务器启动");while (true) {DatagramPacket requestPacket = new DatagramPacket(new byte[4096],4096);socket.receive(requestPacket);String request = new String(requestPacket.getData(),0,requestPacket.getLength());String response = this.process(request);DatagramPacket responsePacket = new DatagramPacket(response.getBytes(),0,response.getBytes().length,requestPacket.getSocketAddress());socket.send(responsePacket);System.out.printf("[%s:%d] req = %s,res = %s \n",requestPacket.getAddress(),requestPacket.getPort(),request,response);}}private String process(String request) {return request;}public static void main(String[] args) throws IOException {UdpEchoServer server = new UdpEchoServer(9090);server.start();}
}
  1. UDPEchoClient:在客户端,用户发起请求并接收返回的结果。对服务器发送的信息应该包括用户IP地址、用户端口号的信息。通过DatagramSocket类中的send方法发送请求。
    在start()方法中,新建字符串request来保存用户所发送的请求,由于网卡的的性质需要发送数据报格式。为了解决这个问题,我们需要new一个数据报requestPacket对请求内容进行封装。同时我们需要把字符串的数据转换成字节流的数据。
    INetAddress中的getByName方法中存储着服务器的IP地址以及服务器的端口号。
    在发送请求之后等待数据接收状态下,receive()方法进入阻塞状态。
    在接收到数据之后,将字节流数据转换成字符串的形式,用response接收。
public class UdpEchoClient {private DatagramSocket socket = null;private String serverIp;private int serverPort;public UdpEchoClient(String serverIp, int serverPort) throws SocketException {this.socket = new DatagramSocket();this.serverIp = serverIp;this.serverPort = serverPort;}public void start() throws IOException {System.out.println("客户端启动");Scanner scanner = new Scanner(System.in);while (true) {System.out.println("请输入要发送的请求:");String request = scanner.next();DatagramPacket requestPacket = new DatagramPacket(request.getBytes(),0,request.getBytes().length,InetAddress.getByName(serverIp),serverPort);socket.send(requestPacket);DatagramPacket responsePacket = new DatagramPacket(new byte[4096],4096);socket.receive(responsePacket);String response = new String(responsePacket.getData(),0, responsePacket.getLength());System.out.println(response);}}public static void main(String[] args) throws IOException {UdpEchoClient client = new UdpEchoClient("127.0.0.1",9090);client.start();}
}

最终,我们成功完成了UDP中基础的数据请求与通信。


3.UDP网络编程的注意事项

  1. 在上面的UDP编程中,主要可以分为以下几个步骤(如下图所示)
    在这里插入图片描述
  2. 进程的运行顺序是先启动服务器,再启动客户端。
  3. 在代码编写过程中,都没有涉及到close操作的原因:DatagreamSocket作为这个进程的对象和服务器程序的生命周期是一样的。如果在某个程序中,socket对象的生命周期和进程不一样,需要提前释放的话就需要进行close操作。

三、TCP网络编程

在TCP中的socket api中与UDP不同的是,在TCP中ServerSocket类是专门给服务器使用的。 Socket类是专门作用于客户端和服务器之间进行通信使用的。
由于TCP属于有连接和可靠传输,他们之间必须时刻保持通信状态。在ServerSocket中有accept方法接收客户端发来的请求。

1.TCP回显代码案例

我们知道TCP是面向字节流的,因此在TCP中的操作是以流对象进行操作的。

  1. 在Socket类中,通过两个方法:getInputStream和getOutpurStream 进行操作的。
  2. 在服务器中,start方法中通过accept同意客户端连接。接下来调用processConnection
  3. 使用Scanner读取InputStream流中的数据内容。
  4. 使用Scanner的好处是对于客户端传输进来的字节流,Scanner会自动将这些数据转换成String。
  5. 在方法start中,循环保持与客户端之间的通信。通过scanner.hasNext()方法判断是否还有输入,当客户端关闭了连接之后,服务器的输入流会收到一个EOF信号,从而判断出没有输入了,最终结束循环。
  6. 将回显的结果以response的结果以二进制的形式写入outputStream中。
  7. 最后,处理完client的请求之后,客户端下线之后,回收client对象,当下次重新有新的客户端时重新启用。
public void start() throws IOException {System.out.println("服务器启动!");//accept可以阻塞等待Socket socketClient = serverSocket.accept();processConnection(socketClient);}private void processConnection(Socket socketClient) throws IOException {System.out.printf("[%s:%d] 客户端上线\n", socketClient.getInetAddress(),socketClient.getPort());try(InputStream inputStream = socketClient.getInputStream();OutputStream outputStream = socketClient.getOutputStream()) {Scanner scanner = new Scanner(inputStream);while (true) {if (!scanner.hasNext()) {System.out.printf("[%s:%d] 客户端下线\n", socketClient.getInetAddress(),socketClient.getPort());break;}//接收客户端发送过来的请求String request = scanner.next();//处理请求String response = process(request);//把响应写回客户端outputStream.write(response.getBytes(),0,response.getBytes().length);//服务器打印日志System.out.printf("[%s:%d] req = %s resp = %s",socketClient.getInetAddress(),socketClient.getPort(),request,response);}} catch (IOException e) {throw new RuntimeException(e);}finally {socketClient.close();}}

在客户端方面,通过scannerNetWork读取inputStream中的数据。将请求的内容以二进制的形式写入outputStream中。当服务器关闭后,scannerNetWork传入服务器的EOF信号,代表着双方连接中断,因此可以退出循环结束进程。

public void start() {System.out.println("客户端启动!");Scanner scanner = new Scanner(System.in);try(InputStream inputStream = clientSocket.getInputStream();OutputStream outputStream = clientSocket.getOutputStream()) {Scanner scannerNetWork = new Scanner(inputStream);while (true) {System.out.println("输入要发送的数据");//1.读取数据String request = scanner.next();//2.将request作为请求发送到服务器中,通过\n标识请求内容request += "\n";outputStream.write(request.getBytes());//3.读取服务器返回的响应if (!scannerNetWork.hasNext()){break;}String response = scannerNetWork.next();System.out.println(response);}} catch (IOException e) {e.printStackTrace();}}

2.TCP多线程使用

在上面的TCP回显代码中,并不能使用于多线程的形势下。在服务器中,应当有线程池之类的多线程状态中,同时处理多个客户端发送过来的请求。
因此需要对服务器实现多线程解决问题。通过线程池的方式,while循环中不断接收客户端发来的连接请求,并将任务交予线程池解决。

public void start() throws IOException {System.out.println("服务器启动!");ExecutorService pool = Executors.newCachedThreadPool();//accept可以阻塞等待while (true) {Socket socketClient = serverSocket.accept();pool.submit(new Runnable() {@Overridepublic void run() {try{processConnection(socketClient);} catch (IOException e) {throw new RuntimeException(e);}}});}}

总结

在本文中主要讲解了TCP和UDP两个协议之间的区别;同时对UDP网络编程进行了最基础的创建。
源码☞本文源码

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

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

相关文章

元器件基础学习笔记——二极管基础

一、二极管基础 二极管是用半导体材料(硅、硒、锗等)制成的一种电子器件,具有单向导电性,是现代电子技术的基石。它在电子电路中扮演着至关重要的角色,通过与电阻、电容、电感等元器件的合理连接,能够实现整流、检波、限幅、稳压等…

Midjourney只需输入描述性文本,就能生成栩栩如生的图像

前言 Midjourney 是一款基于人工智能的图像生成工具,由 Midjourney 研究实验室开发。它最初在2022年3月面世,并在同年7月12日开始公开测试。Midjourney 的核心功能是能够根据用户提供的文本提示(prompt),利用其内置的…

【软件测试】--接口自动化测试

1. 接口自动化 1.1 概念 接口测试:是对系统或组件之间的接口进行测试,主要是校验数据的交换、传递和控制管理过程,以及相互逻辑依赖关系 自动化测试:是把以人为驱动的测试行为转化为机器执行的一种过程 接口自动化测试&#xff1…

【详细】Ubuntu下安装qt5

Ubuntu下安装qt5 一. QT安装环境准备1、判断gcc是否安装2、安装g3、安装clang编译器4、安装 clang 5、安装make6、安装make-guile7、安装cmake 二. QT5安装1、安装Qt5的组件2、安装Qt的开发工具3、安装qtcreator4、安装qt55、安装qt charts(可选) 三、安…

国服最强文字转音频?Fish Speech

官网文档与示例 Fish Speech V1.2 是一款领先的文本到语音 (TTS) 模型,使用 30 万小时的英语、中文和日语音频数据进行训练。我尝试用1066运行,但是质量不尽如人意,建议使用RTX系列的显卡进行推理。 使用结果展示 text """20…

WPF项目实战视频《三》(主要为客户端软件界面设计)

20.WPF项目实战(项目介绍) 1.WEB API:接口,如何获取数据(增删改查) 2.客户端 项目功能: 待办,备忘录,汇总,完成比例,设置系统主题等。 21.WPF项…

nodejs与npm版本对应表

Node.js — Node.js 版本 (nodejs.org)

Springboot集成微信公众号模板通知java

先看效果 1、微信模板消息官方文档 微信公众平台 2、申请微信公众平台接口测试账号 微信公众平台 3、创建3个实体 (1)、ConfigBean import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configurat…

python中,jsonpath提取数据的时候出现TypeError: ‘bool‘ object is not subscriptable怎么解决

json格式如下: { success: True, result: { codeInfo: { code: 0, msg: 成功 }, uploadToken: { resId: rzJRpo, endpoint: https://sit-api-ypsx-resource.ypsx-internal.com/r…

stm32入门-----DMA直接存储器存取(上——理论篇)

目录 前言 DMA 1.简介 2.存储器映像 3.DMA结构 4.数据宽度与对齐 5.DMA工作示例 前言 本期我们就开始学习DMA直接存储器存取,DMA是一个数据装运的小助手,执行数据的搬运处理,减少了CPU的负担,在stm32中担当重要的工作。在前…

pypi如何上传自己的代码记录

目录 一. 注册pypi账号并创建token 1. 注册pypi账号并创建token 2. Pypi账号注册 3. 邮箱验证 ​编辑 4. 重新生成恢复代码 5. 输入账号密码 ​编辑 6. 保存code并继续 ​编辑7. 输入一行即可,然后点击verify 8. 点击左方目录内的account setting&#xff…

基于微信小程序的高校排课系统 /基于微信小程序的排课管理系统/课程管理系统

摘 要 随着信息技术和网络技术的飞速发展,人类已进入全新信息化时代,传统管理技术已无法高效,便捷地管理信息。为了迎合时代需求,优化管理效率,各种各样的管理系统应运而生,各行各业相继进入信息管理时代&a…

爱快路由的dns强制客户端代理真是个强大的功能

大致情况是这样的:同事说在linux服务器/etc/resolv.conf上随便写个IP地址【不在线的】,dns地址也能解析,让我帮忙查查。 我看了下也感觉纳闷,试了下不光在服务器上,我本地的pc随便设置了个dns解析也是一样的。 通过wir…

解决Qt6 error: The kit needs to define a CMake tool to parse this project.

cmake对于Qt6来说很重要,所以学会cmake是必须的。 上述错误,就是我在Windows10下运行cmake项目总是报错的一个问题。 明明路径已经配好了,却总是报错。 具体原因可能和cmake的版本,以及是否设置为默认有关。另外也和QtCreator中…

飞凌全志T527开发板实现局域网内文件传输功能

之前玩开发板的时候,如果需要实现主机与开发板之间的文件传输,通常是通过挂载NFS的方式,而飞凌的OKT527板载WIFI,并且官方提供的镜像中已经将其成功驱动,那我们就可以通过WIFI连接家中的路由器的方式,实现局…

【Redis】Centos7 安装 redis(详细教程)

查看当前 Redis 版本: 当前的 redis 版本太老了,选择安装 Redis5。 一、使用 yum 安装 1、首先安装 scl 源 yum install centos-release-scl-rh 由于我之前已经安装过了,所以加载速度比较快,且显示已经安装成功,是最…

MongoDB整合SpringBoot

文章目录 SpringBoot整合MongoDB环境准备文档操作相关注解创建实体添加文档查询文档更新文档删除文档 小技巧:如何去掉_class属性 SpringBoot整合MongoDB https://docs.spring.io/spring-boot/docs/current/reference/html/data.html#data.nosql.mongodb.repositor…

prompt技巧(2) - 如何解决对话过程中角色反转问题

在一些角色扮演场景下,需要大模型扮演某个角色进行对话,但是在对话过程中偶尔会发生角色反转问题,如大模型扮演客户角色与销售人员进行对话,大模型有时会忘记自己身份,突然以销售人员的身份进行回答,这种情…

ubuntu上部署vue项目到ngixn中+SpringBoot项目+postgresql数据库

文章目录 前提1.Ubuntu上安装ngix2.部署Vue项目2.1上传vue项目2.2.配置 3.Ubuntu上安装Postgres4.部署springboot项目 前提 记一次在ubuntu部署前端vue和后端springboot项目,以及数据库postgresql的安装以及启动、停止等常用的命令。 1.Ubuntu上安装ngix 1、检查…

Cuppa CMS v1.0 任意文件读取漏洞(CVE-2022-25578)

前言 春秋云镜靶场是一个专注于网络安全培训和实战演练的平台,旨在通过模拟真实的网络环境和攻击场景,提升用户的网络安全防护能力和实战技能。这个平台主要提供以下功能和特点: 实战演练: 提供各种网络安全攻防演练场景&#…