当前位置: 首页 > news >正文

【Java EE初阶 --- 网络原理】网络编程

乐观学习,乐观生活,才能不断前进啊!!!

我的主页:optimistic_chen

我的专栏:c语言 ,Java

欢迎大家访问~
创作不易,大佬们点赞鼓励下吧~

文章目录

  • 前言
  • 网络编程原理
    • UDP Socket编程准备
      • 服务端
      • 客户端
    • TCP Socket编程准备
      • 服务端
      • 客户端
      • 多个客户端同时发起请求
  • 完结

前言

在上篇博客提到网络的发展史,我们知道实现网络通信的原理,那么实现网络通信能给人类发展带来怎么样的驱动力呢?这篇博客将带了解网络通信的最大优势,网络编程带来的丰富的网络资源。

网络编程原理

⽹络编程:指⽹络上的主机,通过不同的进程,以编程的⽅式实现⽹络通信(或称为⽹络数据传输)

因为应用层要把数据交给传输层,传输层就提供了一个接口socket,这个接口是程序员关注的重中之重;至于更底层的数据链路层的接口,我们基本不关心,它们基本都是操作系统实现完成的。
而在传输层中有两个核心协议:TCP/UDP

协议
TCP有连接可靠传输面向字节流全双工
UDP无连接不可靠传输面向数据报全双工

· 有无连接,指的是一个抽象概念,逻辑上有无连接,也就是说,双方主机都保存在对方的信息
· 是否可靠传输,网络传输中,数据是非常容易出现丢失的情况,因为光电信号容易抽到外界干扰。可靠传输只是尽可能的提高传输成功的概率,不可靠传输把数据一发就不再管了。
· 面向字节流,读写数据时,以字节为单位;面向数据报, 读写数据时,以一个数据报为单位。
· 全双工支持双向通信(能读也能写);半双工,只支持单向通信(要么读要么写)
网卡是由Socket文件操作,所以操作网卡和操作普通文件差不多;Socket文件就相当于“网卡的遥控器”

UDP Socket编程准备

DatagramSocket是UDP Socket,用于 发送和接收UDP数据报(工具)

打开文件

方法名
DatagramSocket()创建⼀个UDP数据报的Socket,绑定到本机任意⼀个随机端⼝(⼀般⽤于客⼾端)
DatagramSocket(int port)创建⼀个UDP数据报的Socket,绑定到本机指定的端⼝(⼀般⽤于服务端)

读写文件

方法名
void receive(DatagramPacket p )从此套接字接收(读)数据报(如果没有接收到数据报,该⽅法会阻塞等待)
void send(DatagramPacket p)从此套接字发送(写)数据报(不会阻塞等待,直接发送)

关闭文件

方法名
void close()关闭此数据报套接字

DatagramPacket 是UDP Socket(单位),是 发送和接收的数据报(数据)

构造方法
DatagramPacket(byte[] buf, int length)构造⼀个DatagramPacket以⽤来接收数据报,接收的数据保存在字节数组(第⼀个参数buf)中,接收指定⻓度(第⼆个参数length)
DatagramPacket(byte[] buf, int offset, int length,SocketAddress address)构造⼀个DatagramPacket以⽤来发送数据报,发送的数据为字节数组(第⼀个参数buf)中,从0到指定⻓度(第⼆个参数length)。address指定⽬的主机的IP和端⼝号
方法
InetAddress getAddress()从接收的数据报中,获取发送端主机IP地址;或从发送的数据报中,获取接收端主机IP地址
int getPort()从接收的数据报中,获取发送端主机的端⼝号;或从发送的数据报中,获取接收端主机端⼝号

服务端


import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;
import java.nio.charset.StandardCharsets;public class UdpEchoServer {private DatagramSocket socket=null;public UdpEchoServer(int port) throws SocketException {//指定端口号,让服务器使用socket=new DatagramSocket(port);//socket对象代表网卡对象//读文件代表从网卡收数据,写文件代表从网卡发数据}public  void start() throws IOException {System.out.println("启动服务器");while(true){//循环一次相当于处理一次请求//DatagramPacket表示一个UDP数据报,此处传入字节数组DatagramPacket requestpacket=new DatagramPacket(new byte[4096],4096);//1.读取请求并解析socket.receive(requestpacket);//输出型参数//将读取到的二进制数据转换为字符串String request=new String(requestpacket.getData(),0,requestpacket.getLength());//2.根据请求,计算响应(此处为回显服务器)String response=process(request);//3.把响应返回给客户端//根据request构造 DatagramPacket,发送给客户端DatagramPacket responsePacket=new DatagramPacket(response.getBytes(),response.getBytes().length,requestpacket.getAddress(),requestpacket.getPort());//此时UPD协议无连接,没有保存客户端信息,还不知道发给谁//还需要知道目的IP和目的端口号socket.send(responsePacket);System.out.printf("[%s:%d] req:%s,resp:%s\n",requestpacket.getAddress().toString(),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();}
}

在这里插入图片描述

客户端


import java.io.IOException;
import java.net.*;
import java.sql.SQLOutput;
import java.util.Scanner;public class UdpEchoClient {private DatagramSocket  socket = null;private String serverIP;private int serverPort;//客户端,构造方法要指定访问服务区的地址和端口号public UdpEchoClient(String serverIP,int serverPort) throws SocketException {this.serverIP=serverIP;this.serverPort=serverPort;socket=new DatagramSocket();}public void start() throws IOException {while(true){//客户端输入数据Scanner scanner=new Scanner(System.in);System.out.println("请输入要发送的内容");if(!scanner.hasNext()){break;}String request=scanner.next();//把数据发给服务器,要构造DatagramPacket对象//不但要构造数据,也要设置好服务器的IP和端口号DatagramPacket requestPacket=new DatagramPacket(request.getBytes(),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();}
}

在这里插入图片描述
在这里插入图片描述

TCP Socket编程准备

TCP核心就是面向字节流,读写的基本单位就是字节byte
ServerSocket是创建 TCP 服务端 Socket 的接口

构造方法
ServerSocket(int port)创建一个服务端Socket,并指定端口号
普通方法
Socket accept()联通连接的关键操作
void close()关闭套接字

Socket是客户端Socket,或者服务端接收到客户端建立连接的请求后,返回服务端Socket.但是无论是客户端还是服务端,都是双方建立联系以后,保存对端信息,与对方接收发数据的

构造方法
Socket(String host,int port)创建一个客户端Socket,并与对应IP的主机上,对应端口的进程建立联系
普通方法
InetAddress getInetAddress()返回Socket连接地址
InetStream getInputStream()返回Socket输入流(读)
InetStream getOutputStream()返回Socket输出流(写)

服务端

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;public class TcpEchoServer {private ServerSocket serversocket=null;public TcpEchoServer(int port) throws IOException {serversocket=new ServerSocket(port);}public void start() throws IOException {System.out.println("启动服务器");while(true){//TCP先处理客户端发来的连接Socket clientSocket=serversocket.accept();processConnection(clientSocket);}}private void processConnection(Socket clientSocket){System.out.printf("[%s:%d]客户端上线\n",clientSocket.getInetAddress(),clientSocket.getPort());try(InputStream inputStream=clientSocket.getInputStream();OutputStream outputStream=clientSocket.getOutputStream()) {Scanner scanner=new Scanner(inputStream);PrintWriter writer=new PrintWriter(outputStream);while(true){//读取请求if(!scanner.hasNext()) {System.out.printf("[%s:%d]客户端下线\n",clientSocket.getInetAddress(),clientSocket.getPort());break;}String request=scanner.next();//根据请求计算响应String response=process(request);//返沪响应到客户端writer.println(response);writer.flush();System.out.printf("[%s:%d] req:%s,resp:%s\n",clientSocket.getInetAddress(),clientSocket.getPort(),request,response);}} catch (IOException e) {e.printStackTrace();}}private String process(String request){return request;}public static void main(String[] args) throws IOException {TcpEchoServer server=new TcpEchoServer(9090);server.start();}
}

客户端

import jdk.jshell.SourceCodeAnalysis;import java.io.*;
import java.net.Socket;
import java.util.Scanner;public class TcpEchoClient {private Socket socket=null;public TcpEchoClient(String serverIP,int serverPort) throws IOException {socket=new Socket(serverIP,serverPort);}public void start(){Scanner scanner=new Scanner(System.in);try(InputStream inputStream=socket.getInputStream();OutputStream outputStream=socket.getOutputStream()){Scanner scanner1=new Scanner(inputStream);PrintWriter writer=new PrintWriter(outputStream);while(true){//输入数据String request=scanner.next();//发送给服务器writer.println(request);//这里只是把数据发送到“发送缓冲区(内存)”,没有到网卡writer.flush();//刷新缓冲区//读取服务器返回的响应String response=scanner1.next();//打印System.out.println(response);}} catch (IOException e) {e.printStackTrace();}}public static void main(String[] args) throws IOException {TcpEchoClient client=new TcpEchoClient("127.0.0.1",9090);client.start();}
}

多个客户端同时发起请求

面对多个客户端同时给服务器发送请求,我们要求第一时间响应,也就是说服务端要同时回应多个请求,那就刚好用到了多线程的理念,多个线程解决多个客户端请求。

首先,我们使服务器只要接收到请求,就去创建一个线程,响应请求

public void start() throws IOException {System.out.println("服务器启动!");while (true) {Socket clientSocket = serversocket.accept();Thread t = new Thread(() -> {processConnection(clientSocket);});t.start();}
}

但是呢,如果有超级多的客户端发起请求,难道我们要创建同样多的线程吗?这种方法太浪费CPU资源,为了避免巨大的资源开销,可以引入线程池

public void start() throws IOException {System.out.println("服务器启动!");ExecutorService service = Executors.newCachedThreadPool();while (true) {Socket clientSocket = serversocket.accept();service.submit(new Runnable() {@Overridepublic void run() {processConnection(clientSocket);}});}
}

完结


可以点一个免费的赞并收藏起来~
可以点点关注,避免找不到我~ ,我的主页:optimistic_chen
我们下期不见不散 ~ ~ ~

http://www.dtcms.com/a/284533.html

相关文章:

  • Vue3 + WebSocket
  • 基于现代R语言【Tidyverse、Tidymodel】的机器学习方法
  • 3.2 函数参数与返回值
  • .vscode 扩展配置
  • 浅析网络安全面临的主要威胁类型及对应防护措施
  • 【C++指南】C++ list容器完全解读(四):反向迭代器的巧妙实现
  • 如何做好DNA-SIP?
  • 【41】MFC入门到精通——MFC中 GetLBText()、GetWindowText()、SetWindowText区别
  • 扭蛋机小程序开发:开启线上娱乐新风尚
  • 分布式光伏发电系统中的“四可”指的是什么?
  • 教资科三【信息技术】— 学科知识: 第一章(信息技术基础)
  • 基于springboot+vue+mysql技术的实验室管理系统(源码+论文)
  • 044_设计模式入门(创建型 / 结构型 / 行为型)
  • 【解决方案】鸿蒙 / 矿鸿系统 Shell 无故退出问题(息屏导致)详解
  • Spatial Frequency Modulation for Semantic Segmentation。针对图像下采样造成信息丢失问题的解决思路
  • 深入理解 Spring Boot Starter 的生成机制
  • tcp/udp调试工具
  • Linux内核网络栈深度剖析:inet_connection_sock.c的服务器端套接字管理
  • LINUX例行性工作(计划任务)实验操作 ---at和crontab以及系统级别的计划任务
  • springboot跨域问题 和 401
  • 当下主流摄像头及其核心参数详解
  • 不同场景下git指令的搭配
  • pycharm回车、删除、方向键和快捷键等不能使用原因
  • TRAE Agent 在 SWE-bench Verified 上得分 75.2%,并已开源
  • opencv、torch、torchvision、tensorflow的区别
  • Linux717 SWAP扩容;逻辑卷条带化
  • 前端-HTML
  • 杰理AC70NN项目用脚本自定义添加.mk文件,直接链接进主Makefile脚本编译
  • 开通腾讯位置复位
  • 深入理解Collections.addAll方法