Day19-【Java SE进阶】网络编程

一、网络编程

1.概述
  • 可以让设备中的程序与网络上其他设备中的程序进行数据交互(实现网络通信的)。
  • java.net,*包下提供了网络编程的解决方案!

基本的通信架构
基本的通信架构有2种形式:CS架构(Client客户端/Server服务端)、BS架构(Browser浏览器/Server服务端)。

在这里插入图片描述

在这里插入图片描述
网络通信的关键三要素
在这里插入图片描述
IP地址

  • IP(Internet Protocol):全称”互联网协议地址”,是分配给上网设备的唯一标志。
  • IP地址有两种形式:IPV4、IPV6

在这里插入图片描述

  • IPv6:共128位,号称可以为地球每一粒沙子编号。
  • IPV6分成8段表示,每段每四位编码成一个十六进制位表示,数之间用冒号(:)分开。

在这里插入图片描述
IP域名

  • 公网IP:是可以连接互联网的IP地址;内网IP:也叫局域网IP,只能组织机构内部使用。
  • 192.168.开头的就是常见的局域网地址,范围即为192.168.0.0–192.168.255.255,专门为组织机构内部使用。
    在这里插入图片描述

特殊IP地址
127.0.0.1、localhost:代表本机IP,只会寻找当前所在的主机。
IP常用命令:
ipconfig:查看本机IP地址。
ping IP地址:检查网络是否连通。

1.1 InetAddress
  • 代表IP地址。
    -

端口

  • 标记正在计算机设备上运行的应用程序的,被规定为一个16 位的二进制,范围是 0~65535

分类

  • 周知端口:0~1023,被预先定义的知名应用占用(如:HTTP占用 80,FTP占用21)
  • 注册端口:1024~49151,分配给用户进程或某些应用程序。
  • 动态端口:49152到65535,之所以称为动态端口,是因为它 一般不固定分配某种进程,而是动态分配。
  • 注意:我们自己开发的程序一般选择使用注册端口,且一个设备中不能出现两个程序的端口号一样,否则出错。

通信协议

  • 网络上通信的设备,事先规定的连接规则,以及传输数据的规则被称为网络通信协议

开放式网络互联标准:OSI网络参考模型

  • OSI网络参考模型:全球网络互联标准。
  • TCP/IP网络模型:事实上的国际标准。
    在这里插入图片描述
1.2 重要知识点:传输层的2个通信协议
  • UDP(User Datagram Protocol):用户数据报协议;
  • TCP(Transmission ControlProtocol):传输控制协议。

UDP协议

  • 特点:无连接、不可靠通信。
  • 通讯效率高!语音通话 视频直播
  • 不事先建立连接,数据按照包发,一包数据包含:自己的IP、程序端口,目的地IP、程序端口和数据(限制在64KB内)等。
  • 发送方不管对方是否在线,数据在中间丢失也不管,如果接收方收到数据也不返回确认,故是不可靠的。

TCP协议

  • 特点:面向连接、可靠通信。
  • 通信效率相对不高 网页 文件下载 支付
  • TCP的最终目的:要保证在不可靠的信道上实现可靠的传输。
  • TCP主要有三个步骤实现可靠传输:三次握手建立连接,传输数据进行确认,四次挥手断开连接。
  • TCP协议:三次握手建立可靠连接
    • 可靠连接:确定通信双方,收发消息都是正常无问题的!(全双工)
    • 传输数据会进行确认,以保证数据传输的可靠性在这里插入图片描述
  • TCP协议:四次握手断开连接
    • 目的:确保双方数据的收发都已经完成!

在这里插入图片描述

在这里插入图片描述

2. UDP通信-快速入门

UDP通信

  • 特点:无连接、不可靠通信。
  • 不事先建立连接;发送端每次把要发送的数据(限制在64KB内)、接收端IP、等信息封装成一个数据包,发出去就不管了。
  • Java提供了一个java.net.DatagramSocket类来实现UDP通信。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
import java.net.*;public class Client {public static void main(String[] args) throws Exception {//1. 创建客户端对象(发韭菜出去的人)DatagramSocket socket = new DatagramSocket();//2. 创建数据包对象封装要发出去得数据(创建一个韭菜盒子)/* public DatagramPacket(byte buf[], int length,InetAddress address,int port)参数一:封装要发出去的数据。参数二:发送出去的数据大小(字节个数)参数三:服务端的IP地址(找到服务端主机)参数四:服务端程序的端口。*/byte[] bytes = "我是客户端,向您发送数据!".getBytes();DatagramPacket packet = new DatagramPacket(bytes,bytes.length,InetAddress.getLocalHost(),6666);//3. 开始正式发送这个数据包出去了socket.send(packet);System.out.println("客户端数据发送完毕~~~");socket.close();//释放资源!}
}
import java.net.DatagramPacket;
import java.net.DatagramSocket;public class Server {public static void main(String[] args) throws Exception {//1. 创建一个服务端对象(创建一个接韭菜的人) 注册端口DatagramSocket socket = new DatagramSocket(6666);//2. 创建一个数据包对象,用于接收数据的(创建一个韭菜盒子)byte[] buf = new byte[1024 * 64];DatagramPacket packet = new DatagramPacket(buf, buf.length);//3. 开始正式使用数据包来接收客户端发来的数据socket.receive(packet);//4. 从字节数组中,把接收到的数据直接打印出来// 接收多少就倒出多少 获取本次数据包接收了多少数据int len = packet.getLength();String s = new String(buf, 0, len);System.out.println(s);String hostAddress = packet.getAddress().getHostAddress();int port = packet.getPort();System.out.println(hostAddress + ":" + port);socket.close();//释放资源}
}

在这里插入图片描述
在这里插入图片描述
UDP通信 多发多收
edit configures->allow 多开 apply即可躲开client程序的

package com.jingwei;import java.net.*;
import java.util.Scanner;public class Client {public static void main(String[] args) throws Exception {//1. 创建客户端对象(发韭菜出去的人)DatagramSocket socket = new DatagramSocket();//可以分配端口 或者不填写自动分配//2. 创建数据包对象封装要发出去得数据(创建一个韭菜盒子)/* public DatagramPacket(byte buf[], int length,InetAddress address,int port)参数一:封装要发出去的数据。参数二:发送出去的数据大小(字节个数)参数三:服务端的IP地址(找到服务端主机)参数四:服务端程序的端口。*/Scanner sc = new Scanner(System.in);while (true) {System.out.println("请说");String msg = sc.nextLine();if("exit".equals(msg)) {System.out.println("退出成功!");socket.close();break;}byte[] bytes = msg.getBytes();DatagramPacket packet = new DatagramPacket(bytes,bytes.length,InetAddress.getLocalHost(),6666);//3. 开始正式发送这个数据包出去了socket.send(packet);}System.out.println("客户端数据发送完毕~~~");}
}
package com.jingwei;import java.net.DatagramPacket;
import java.net.DatagramSocket;public class Server {public static void main(String[] args) throws Exception {//1. 创建一个服务端对象(创建一个接韭菜的人) 注册端口DatagramSocket socket = new DatagramSocket(6666);//2. 创建一个数据包对象,用于接收数据的(创建一个韭菜盒子)byte[] buf = new byte[1024 * 64];DatagramPacket packet = new DatagramPacket(buf, buf.length);while (true) {//3. 开始正式使用数据包来接收客户端发来的数据socket.receive(packet);//4. 从字节数组中,把接收到的数据直接打印出来// 接收多少就倒出多少 获取本次数据包接收了多少数据int len = packet.getLength();String s = new String(buf, 0, len);System.out.println(s);String hostAddress = packet.getAddress().getHostAddress();int port = packet.getPort();System.out.println(hostAddress + ":" + port);System.out.println("-----------------------");}}
}
3. TCP通信
  • 特点:面向连接、可靠通信。
  • 通信双方事先会采用“三次握手”方式建立可靠连接,实现端到端的通信;底层能保证数据成功传给服务端。
  • Java提供了一个java.net.Socket类来实现TCP通信。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
一发一收

package com.jingwei;import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketAddress;public class SocketClient {public static void main(String[] args) throws IOException {System.out.println("服务端启动成功!");//1. 创建ServerSocket的对象,同时为服务端注册端口。ServerSocket serverSocket = new ServerSocket(8888);//2. 使用ServerSocket对象,调用accept方法,等待客户端的连接请求。Socket socket = serverSocket.accept();//3. 从socket通信管道中得到一个字节输入流InputStream is = socket.getInputStream();//4. 把原始的字节输入流包装成数据输入流DataInputStream dis = new DataInputStream(is);//5. 使用数据输入流读取客户端发送过来的消息String s = dis.readUTF();System.out.println(s);//其实我们也可以获取客户端的IP地址SocketAddress remoteSocketAddress = socket.getRemoteSocketAddress();System.out.println(remoteSocketAddress);dis.close();socket.close();}
}
package com.jingwei;import java.io.DataOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.Socket;public class SocketServer {public static void main(String[] args) throws IOException {//1. 创建Socket对象,并同时请求与服务器程序的连接。Socket socket = new Socket("127.0.0.1", 8888);//2. 从Socket通信管道中得到一个字节输出流,用来发数据给服务端程序OutputStream os = socket.getOutputStream();//3. 把低级的字节输出流包装成数据输出流DataOutputStream dos = new DataOutputStream(os);//4. 开始写数据出去了dos.writeUTF("在一起好吗?");dos.close();socket.close();//释放连接资源}
}

多发多收

package com.jingwei;import java.io.DataOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.Socket;
import java.util.Scanner;public class SocketServer {public static void main(String[] args) throws IOException {//1. 创建Socket对象,并同时请求与服务器程序的连接。Socket socket = new Socket("127.0.0.1", 8888);//2. 从Socket通信管道中得到一个字节输出流,用来发数据给服务端程序OutputStream os = socket.getOutputStream();//3. 把低级的字节输出流包装成数据输出流DataOutputStream dos = new DataOutputStream(os);Scanner sc = new Scanner(System.in);while (true) {System.out.println("请说:");//4. 开始写数据出去了String msg = sc.nextLine();if("exit".equals(msg)){dos.close();socket.close();break;}dos.writeUTF(msg);dos.flush();}//        dos.close();
//
//        socket.close();//释放连接资源}
}
package com.jingwei;import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketAddress;public class SocketClient {public static void main(String[] args) throws IOException {System.out.println("服务端启动成功!");//1. 创建ServerSocket的对象,同时为服务端注册端口。ServerSocket serverSocket = new ServerSocket(8888);//2. 使用ServerSocket对象,调用accept方法,等待客户端的连接请求。Socket socket = serverSocket.accept();//3. 从socket通信管道中得到一个字节输入流InputStream is = socket.getInputStream();//4. 把原始的字节输入流包装成数据输入流DataInputStream dis = new DataInputStream(is);while (true) {//5. 使用数据输入流读取客户端发送过来的消息try {String s = dis.readUTF();System.out.println(s);//其实我们也可以获取客户端的IP地址SocketAddress remoteSocketAddress = socket.getRemoteSocketAddress();System.out.println(remoteSocketAddress);} catch (IOException e) {System.out.println(socket.getRemoteSocketAddress()+"离线了!");dis.close();socket.close();break;}}
//        dis.close();
//        socket.close();}
}

在这里插入图片描述

目前我们开发的服务端程序,是否可以支持与多个客户端同时通信?

  • 不可以的。
  • 因为服务端现在只有一个主线程,只能处理一个客户端的消息,

在这里插入图片描述

package com.jingwei;import java.io.DataOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.Socket;
import java.util.Scanner;public class SocketServer {public static void main(String[] args) throws IOException {//1. 创建Socket对象,并同时请求与服务器程序的连接。Socket socket = new Socket("127.0.0.1", 8888);//2. 从Socket通信管道中得到一个字节输出流,用来发数据给服务端程序OutputStream os = socket.getOutputStream();//3. 把低级的字节输出流包装成数据输出流DataOutputStream dos = new DataOutputStream(os);Scanner sc = new Scanner(System.in);while (true) {System.out.println("请说:");//4. 开始写数据出去了String msg = sc.nextLine();if("exit".equals(msg)){dos.close();socket.close();break;}dos.writeUTF(msg);dos.flush();}//        dos.close();
//
//        socket.close();//释放连接资源}
}
package com.jingwei;import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;public class SocketClient {public static void main(String[] args) throws IOException {System.out.println("服务端启动成功!");//1. 创建ServerSocket的对象,同时为服务端注册端口。ServerSocket serverSocket = new ServerSocket(8888);//2. 使用ServerSocket对象,调用accept方法,等待客户端的连接请求。while (true) {Socket socket = serverSocket.accept();System.out.println("有人上线了"+socket.getRemoteSocketAddress());new ServerReaderThread(socket).start();}
//        dis.close();
//        socket.close();}
}
package com.jingwei;import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.Socket;public class ServerReaderThread extends Thread{private Socket socket;public ServerReaderThread(Socket socket) {this.socket = socket;}@Overridepublic void run() {try {InputStream inputStream = socket.getInputStream();DataInputStream dataInputStream = new DataInputStream(inputStream);while (true){try {String msg = dataInputStream.readUTF();System.out.println(msg);} catch (IOException e) {System.out.println(socket.getRemoteSocketAddress()+"下线了");dataInputStream.close();socket.close();break;}}} catch (IOException e) {throw new RuntimeException(e);}}
}

TCP通信-综合案例

  • 即时通信-群聊
  • 实现一个简易版的BS架构

题目:要求从浏览器中访问服务器
并立即让服务器响应一个很简单的网页给浏览器展示
网页内容就是“我666”

在这里插入图片描述

注意:服务器必须给浏览器响应HTTP协议规定的数据格式,否则浏览器不识别返回的数据

在这里插入图片描述

package com.jingwei;import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;public class SocketClient {public static void main(String[] args) throws IOException {System.out.println("服务端启动成功!");//1. 创建ServerSocket的对象,同时为服务端注册端口。ServerSocket serverSocket = new ServerSocket(8888);//2. 使用ServerSocket对象,调用accept方法,等待客户端的连接请求。while (true) {Socket socket = serverSocket.accept();System.out.println("有人上线了"+socket.getRemoteSocketAddress());
//            new ServerReaderThread(socket).start();new ServerWriterThread(socket).start();}
//        dis.close();
//        socket.close();}
}
package com.jingwei;import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintStream;
import java.net.Socket;public class ServerWriterThread extends Thread {private Socket socket;public ServerWriterThread(Socket socket) {this.socket = socket;}@Overridepublic void run() {try {OutputStream outputStream = socket.getOutputStream();PrintStream printStream = new PrintStream(outputStream);printStream.println("Http/1.1 200 OK");printStream.println("Content-Type: text/html;charset=utf-8");printStream.println();printStream.println("<!DOCTYPE html>");printStream.println("<html>");printStream.println("<head>");printStream.println("<title>Server Writer</title>");printStream.println("</head>");printStream.println("<body>");printStream.println("<h1>Server Writer</h1>");printStream.println("</body>");printStream.println("</html>");printStream.flush();printStream.close();socket.close();} catch (IOException e) {throw new RuntimeException(e);}}}

拓展知识
每次请求都开一个新线程,到底好不好?
高并发时,容易宕机!

在这里插入图片描述

package com.day19;import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.*;public class SocketClient {public static void main(String[] args) throws IOException {System.out.println("服务端启动成功!");//1. 创建ServerSocket的对象,同时为服务端注册端口。ServerSocket serverSocket = new ServerSocket(8888);//创建出一个线程池,负责处理通信管道的任务ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(20 * 2, 20 * 2, 0, TimeUnit.SECONDS,new ArrayBlockingQueue<>(8), Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());//2. 使用ServerSocket对象,调用accept方法,等待客户端的连接请求。while (true) {Socket socket = serverSocket.accept();System.out.println("有人上线了"+socket.getRemoteSocketAddress());
//            new ServerReaderThread(socket).start();//3、把这个客户端对应的socket通信管道,交给一个独立的线程负责处理。threadPoolExecutor.execute(new ServerWriterRunnable(socket));}
//        dis.close();
//        socket.close();}
}
package com.day19;import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintStream;
import java.net.Socket;public class ServerWriterRunnable implements Runnable {private Socket socket;public ServerWriterRunnable(Socket socket) {this.socket = socket;}@Overridepublic void run() {try {OutputStream outputStream = socket.getOutputStream();PrintStream printStream = new PrintStream(outputStream);printStream.println("Http/1.1 200 OK");printStream.println("Content-Type: text/html;charset=utf-8");printStream.println();printStream.println("<!DOCTYPE html>");printStream.println("<html>");printStream.println("<head>");printStream.println("<title>Server Writer</title>");printStream.println("</head>");printStream.println("<body>");printStream.println("<h1>Server Writer</h1>");printStream.println("</body>");printStream.println("</html>");printStream.flush();printStream.close();socket.close();} catch (IOException e) {throw new RuntimeException(e);}}}

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

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

相关文章

【新版】系统架构设计师 - 知识点 - 结构化开发方法

个人总结&#xff0c;仅供参考&#xff0c;欢迎加好友一起讨论 文章目录 架构 - 知识点 - 结构化开发方法结构化开发方法结构化分析结构化设计 数据流图和数据字典模块内聚类型与耦合类型 架构 - 知识点 - 结构化开发方法 结构化开发方法 分析阶段 工具&#xff1a;数据流图、…

从0到1实现RPC | 12 限流

在服务提供者provider端添加限流逻辑 限流&#xff1a;指定时间内请求数超过指定阈值时就抛出异常。 在ProviderInvoker的调用过程中&#xff0c;添加限流逻辑&#xff1a; 使用滑动窗口SlidingTimeWindow统计30s的请求数&#xff1b;每个服务service对应一个滑动窗口&#…

神经网络--反向传播算法推导

神经网络–反向传播算法推导 文章目录 神经网络--反向传播算法推导概述神经网络模型反向传导算法 概述 以监督学习为例&#xff0c;假设我们有训练样本集 ( x ( i ) , y ( i ) ) (x^{(i)},y^{(i)}) (x(i),y(i))&#xff0c;那么神经网络算法能提供一种复杂且非线性的假设模型 …

route路由命令、ip route命令、default默认路由(0.0.0.0 )

文章目录 3. route语法3.1 查看路由表3.1 参数解释 3.2 添加路由记录3.2.1 添加到达单个目标主机的路由3.2.2 添加到达目标网络的路由3.2.3 添加默认路由 3.3 删除路由记录 4. ip route4.1 查看路由4.1.1 不带条件4.1.2 带条件 4.2 添加路由4.3 删除路由4.4 清空路由表&#xf…

elasticsearch7安全配置--最低安全等级,用户名密码

上一篇博客在centos7上安装了elasticsearch7 接下来对elasticsearch进行安全方面的配置 minimal security 最低安全等级&#xff0c;用户名密码 首先开启xpack vim config/elasticsearch.yml xpack.security.enabled: true由于我是单机配置的&#xff0c;还加了如下配置 d…

JavaScript:事件循环机制(同步、异步)(单、多线程)

事件循环机制: 多进程和多线程 1. 进程&#xff1a;程序的一次执行, 它占有一片独有的内存空间 2. 线程&#xff1a; CPU的基本调度单位, 是程序执行的一个完整流程 3. 进程与线程 * 一个进程中一般至少有一个运行的线程: 主线程 * 一个进程中也可以同时运行多个线程, 我们…

ADC的认识

ADC介绍 Q:ADC是什么&#xff1f; A&#xff1a;全称&#xff1a;Analog-to-Digital Converter&#xff0c;指模拟/数字转换器 ADC的性能指标 量程&#xff1a;能测量的电压范围分辨率&#xff1a;ADC能辨别的最小模拟量&#xff0c;通常以输出二进制数的位数表示&#xf…

有限差分法求解一维、二维波动方程

差分格式方法是数值计算方法中微分以及偏微分导数的一种离散化方法。具体来说&#xff0c;它使用相邻两个或者多个数值点的差分来取代偏微分方程中的导数或偏导数。选择差分格式是离散化偏微分方程的第一步&#xff0c;通过这种离散化&#xff0c;我们可以将连续空间区域上的问…

面试通关秘籍:一面到终面的秘密

在当今竞争激烈的职场中&#xff0c;面试已经成为求职者和企业之间互相了解和选择的重要环节。面试过程常常被划分为多个阶段&#xff0c;包括一面&#xff08;初次面试&#xff09;、二面&#xff08;二次面试&#xff09;、三面&#xff08;深入面试&#xff09;以及终面&…

入门:多层感知器Multiple-Layer Perceiver, MLP

本文将简单介绍多层感知器&#xff08;MLP&#xff09;的基本概念、原理和应用。MLP是一种前馈人工神经网络&#xff0c;由多层节点组成&#xff0c;每层节点通过权重和偏置与下一层节点相连。MLP在许多领域都有广泛的应用&#xff0c;如分类、回归、自然语言处理等。 本文将分…

Windows不常见问题集

● 解决CACLS 禁止修改计算机名 管理员权限运行cmd&#xff1a;cacls %SystemRoot%\System32\netid.dll /grant administrators:f ● Excel 2010 AltTab組合鍵設置 HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Explorer&#xff0c;在該路徑建32字元DWO…

Spring Boot | Spring Boot 整合 “Servlet三大组件“ ( Servlet / Filter / Listene )

目录: Spring Boot 整合 "Servlet三大组件" &#xff1a;1. 使用 "组件注册" 的方式 "整合Servlet三大组件" ( 实际操作为 : 创建自定义的"三大组件"对象 结合刚创建"的自定义组件对象"来 将 XxxRegistrationBean对象 通过…