[JAVAee]网络编程-套接字Socket

目录

基本概念

发送端与接收端

请求与响应

​编辑客户端与服务器

Socket套接字 

分类

数据报套接字

流套接字传输模型  

UDP数据报套接字编程

DatagramSocket API

DatagramPacket API

InetSocketAddress API

示例一:

示例二:

TCP流数据报套接字编程

ServerSocket API

Socket API

示例一:

 


网络编程指的是,网络上的主机的不同进程通过编程的方式实现网络通信.同一主机下只要满足不同进程间的通信就可以成为"网络通信"

基本概念

发送端与接收端

在网络通信中:

作为发送数据的进程称为"发送端",发送端主机即网络通信中的"源主机" 

作为接收数据的进程称为"接收端",接收端主机即网络通信中的"目的主机"

注意:网络通信中的发送端与接收端都是相对的.

请求与响应

一般来说,一次网络通信中设计到两次数据传输:

  • 第一次:A端向B端发送的请求
  • 第二次:B端向A端发送的响应

客户端与服务器

服务器:在网络通信下,提供服务的一端.(服务可以指:响应一定的要求)

客户端:获取服务的一端

Socket套接字 

Socket套接字,是由系统提供用于网络通信的技术,是基于TCP/IP协议的网络通信的基本操作单元.基于Socket套接字的网络程序开发就是网络编程.

分类

套接字根据传输层协议主要分成:

  • 数据报套接字:使用传输层UDP协议(User Datagram Protocol)用户数据报协议
  • 流套接字:使用传输层TCP协议(Transmission Control Protocol)传输层控制协议
  • 原始套接字:用于自定义传输层协议

数据报套接字

数据报固定每次传输的字节,更像是写信,有来有回的.

流套接字传输模型  

面对的是字节流.

打电话一般,接通后就可以无节制的传输.

UDP数据报套接字编程

DatagramSocket API

构造方法

方法签名方法说明
DatagramSocket()创建一个UDP数据报套接字的Socket,绑定到本机任意一个随机端口
(一般用于客户端)
DatagramSocket(int
port)
创建一个UDP数据报套接字的Socket,绑定到本机指定的端口(一般用
于服务端)

常用方法 

方法方法说明
void
receive(DatagramPacket p)
从此套接字接收数据报(如果没有接收到数据报,该方法会阻
塞等待)
void send(DatagramPacket
p)
从此套接字发送数据报包(不会阻塞等待,直接发送)
void close()关闭此数据报套接字

DatagramPacket API

构造方法

方法签名方法说明
DatagramPacket(byte[]
buf, int length)
构造一个DatagramPacket以用来接收数据报,接收的数据保存在
字节数组(第一个参数buf)中,接收指定长度(第二个参数
length)
DatagramPacket(byte[]
buf, int length,
SocketAddress address)
构造一个DatagramPacket以用来发送数据报,发送的数据为字节
数组(第一个参数buf)中,从0到指定长度(第二个参数
length)。address指定目的主机的IP和端口号

常用方法 

方法签名方法说明
InetAddress
getAddress()
从接收的数据报中,获取发送端主机IP地址;或从发送的数据报中,获取
接收端主机IP地址
int getPort()从接收的数据报中,获取发送端主机的端口号;或从发送的数据报中,获
取接收端主机端口号
byte[] getData()获取数据报中的数据

InetSocketAddress API

InetSocketAddress是ScketAddress的一个子类,用来包装IP与端口号

方法方法说明
InetSocketAddress(InetAddress addr, int port)创建一个Socket地址,包含IP地址和端口号

示例一:

客户端像服务器发出请求,但服务器无响应版本

服务器:

public class UdpServer {

    private DatagramSocket socket= null;

    public UdpServer(int port) throws SocketException {//构造方法
        this.socket = new DatagramSocket(port);
    }

    public void start() throws IOException {//作为启动服务器的方法
        while(true){//因为不知道什么时候客户端会发送请求
            //作为服务器,需要不停的接收客户端的请求
            //创建packet
            byte[] bytes = new byte[1024];
            DatagramPacket packet = new DatagramPacket(bytes,bytes.length);//用bytes作为接收,使用的长度为bytes的长度

            System.out.println("等待接收数据中...");
            socket.receive(packet);//还没收到之前会进行阻塞等待
            //此处的版本没有作出响应
            //我们可以打印出收到的packet中的数据看看有什么东西

            System.out.println("IP: " + packet.getAddress().getHostAddress());
            System.out.println("端口号: " + packet.getPort());
            System.out.printf("文本数据为: " + new String(packet.getData()));
            System.out.println("原始数据为: " + Arrays.toString(packet.getData()));
        }
    }
    public static void main(String[] args) throws IOException {
        UdpServer udpServer = new UdpServer(1024);
        udpServer.start();
    }
}

客户端:

方法一:
public class UdpClient {
    public static void main(String[] args) throws IOException {
        //创建Socket
        DatagramSocket socket = new DatagramSocket();//创建一个socket,端口号为系统随机分配
        //构建Packet
        byte[] bytes = "Hello World".getBytes();//字符串转换成byte再塞进数组
        SocketAddress address = new InetSocketAddress("localhost",1024);//目的IP为本地地址,端口号为1024
        DatagramPacket packet = new DatagramPacket(bytes,bytes.length,address);//构建packet
        socket.send(packet);//发送
        System.out.println("发送完成");
    }
}
方法二:
public class UdpClient {
    private DatagramSocket socket = null;//socket
    private String serverIp;
    private int serverPort;

    public UdpClient(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 text = scanner.next();
            if(text.equals("exit")){
                System.out.println("再见");
                break;
            }
            //需要用InetAddress将字符串钟的IP转换成地址形式
            //SocketAddress address = new InetSocketAddress("localhost",1024);//也可以创建一个实例进行包装IP与端口号
            //此处的长度是字节的长度噢,注意单位
            DatagramPacket packet = new DatagramPacket(text.getBytes(),text.getBytes().length,InetAddress.getByName(serverIp),serverPort);
            socket.send(packet);
            System.out.println("发送成功");
        }
    }
    public static void main(String[] args) throws IOException {
        UdpClient client = new UdpClient("127.0.0.1",1024);
        client.start();
    }

先启动服务器后启动客户端发送.

记得打开IDEA可以同时运行两个进程的选项噢!

服务器接收到的信息为:

示例二:

做一个服务器对客户端有响应的版本

简单的英汉翻译

服务器:

public class UdpServerResponse{
    private DatagramSocket socket= null;

    public UdpServerResponse(int port) throws SocketException {//构造方法
        this.socket = new DatagramSocket(port);
    }

    public void start() throws IOException {//启动服务器
        while(true){
            byte[] bytes = new byte[1024];
            DatagramPacket receivePacket = new DatagramPacket(bytes,bytes.length);//创建包来接收

            System.out.println("等待接收数据中...");
            socket.receive(receivePacket);//接收包
            String request = new String(receivePacket.getData(),0,receivePacket.getLength());//根据接收到的包转换成字符串
            String response = process(request);//对请求进行分析
            //记得是getSocketAddress噢里面通常包含了IP与端口号
            DatagramPacket sendPacket = new DatagramPacket(response.getBytes(),response.getBytes().length,receivePacket.getSocketAddress());
            socket.send(sendPacket);//对客户端作出响应
            System.out.println("客户端IP: " + receivePacket.getAddress());
            System.out.println("客户端端口号: " + receivePacket.getPort());
            System.out.println("收到的文本: " + request);
            System.out.println("返回的文本: " + response);
        }
    }

    public String process(String request){//解析请求,看看要做什么
        //这里就做一个英汉翻译吧
        HashMap<String,String> map = new HashMap<>();
        map.put("人","human");
        map.put("猫","cat");
        map.put("狗","dog");
        return map.getOrDefault(request,"查阅失败");
    }

    public static void main(String[] args) throws IOException {
        UdpServerResponse udpServerResponse = new UdpServerResponse(1024);
        udpServerResponse.start();
    }
}

客户端:

public class UdpClientResponse {
    private DatagramSocket socket = null;
    private String serverIp;
    private int serverPort;

    public UdpClientResponse(String serverIp,int serverPort) throws SocketException {
        this.serverIp = serverIp;
        this.serverPort = serverPort;
        socket = new DatagramSocket();
    }

    public void start() throws IOException {
        System.out.println("客户端启动");
        Scanner scanner = new Scanner(System.in);
        while (true) {
            System.out.print("输入: ");
            String request = scanner.next();
            if(request.equals("exit")){
                System.out.println("再见!");
                break;
            }
            //根据请求创建包
            DatagramPacket sendPacket = new DatagramPacket(request.getBytes(),request.getBytes().length, InetAddress.getByName("127.0.0.1"),1024);
            socket.send(sendPacket);
            System.out.println("发送成功");
            DatagramPacket receivePacket = new DatagramPacket(new byte[1024],1024);//创建接收包
            socket.receive(receivePacket);
            String receive = new String(receivePacket.getData(),0,receivePacket.getLength());
            System.out.println(receive);
        }
    }

    public static void main(String[] args) throws IOException {
        UdpClientResponse udpClientResponse = new UdpClientResponse("127.0.0.1",1024);
        udpClientResponse.start();
    }
}

服务器的打印

客户端的打印

TCP流数据报套接字编程

ServerSocket API

创建TCP服务端的API

构造方法

构造方法方法说明
ServerSocket(int port)创建一个服务端流套接字Socket,并绑定到指定端口

方法

方法签
方法说明
Socket
accept()
开始监听指定端口(创建时绑定的端口),有客户端连接后,返回一个服务端Socket
对象,并基于该Socket建立与客户端的连接,否则阻塞等待
void
close()
关闭此套接字

Socket API

用来建立链接后保存对方的信息

构造方法:

方法方法说明
Socket(String host, int
port)
创建一个客户端流套接字Socket,并与对应IP的主机上,对应端口的
进程建立连接

常用方法 

方法签名方法说明
InetAddress getInetAddress()返回套接字所连接的地址
InputStream getInputStream()返回此套接字的输入流
OutputStream getOutputStream()返回此套接字的输出流

在TCP协议中的连接还分为长连接与短链接.

  • 短连接:每次接收到数据并返回响应后,都关闭连接,即是短连接。也就是说,短连接只能一次收发数据
  • 长连接:不关闭连接,一直保持连接状态,双方不停的收发数据,即是长连接。也就是说,长连接可以多次收发数据

示例一:

一请求一响应

此处为长连接(把代码里的while(true)去掉就是短连接啦!只进行一次请求响应)

服务器:

对于服务器来说,每次与客户端连接后会创建一个socket来暂时存储客户端的信息数据

断开连接后,记得要将这个存储客户端数据的socket进行close释放掉

在服务器进程中,一个客户端socket会占用文件描述符的一个位置,一个服务器可能会要与成千上万个客户端进行通信,不释放就会将文件描述符的位置沾满造成泄露.

而服务器的serverSocket的生命周期与整个进程相当,且只有一个.所以可以不进行释放

使用线程池,用多线程的方式来运行服务器达到同时与多个客户端进行通信的功能.

public class TcpServer {
    private ServerSocket socket = null;
     public TcpServer(int port) throws IOException {
         socket = new ServerSocket(port);
     }

     public void start() throws IOException {
         //尝试链接
         ExecutorService threadPool = Executors.newCachedThreadPool();//创建一个线程池,一个线程对应一个客户端进行通信
         while (true) {
             Socket clientSocket = socket.accept();//会阻塞等待接受
             threadPool.submit(() -> {//向线程提供任务
                 try {
                     processConnect(clientSocket);
                 } catch (IOException e) {
                     e.printStackTrace();
                 }
             });
         }
     }
     public void processConnect(Socket clientSocket) throws IOException {
         System.out.println("已与客户端进行链接-" + clientSocket.getInetAddress() + clientSocket.getPort());
         try(InputStream inputStream = clientSocket.getInputStream();
             OutputStream outputStream = clientSocket.getOutputStream()){
             while(true){
                 Scanner scanner = new Scanner(inputStream);//读
                 PrintWriter printWriter = new PrintWriter(outputStream);//写
                 if(!scanner.hasNext()){//客户端不再传输数据就断开链接
                     System.out.println("结束");
                     break;
                 }
                 String request = scanner.next();//接收请求
                 String response = process(request);//处理请求
                 printWriter.println(response);//向客户端写回响应
                 printWriter.flush();//记得写回后进行刷新缓冲区
                 System.out.println("响应:" + clientSocket.getInetAddress() + clientSocket.getPort() + "文本: "+ response);
             }
         } catch (IOException e) {
             e.printStackTrace();
         }finally {
             clientSocket.close();//记得要关闭
         }
     }

     public String process(String request){
         HashMap<String,String> map = new HashMap<>();
         map.put("人","human");
         map.put("猫","cat");
         map.put("狗","dog");
         return map.getOrDefault(request,"查阅失败");
     }

    public static void main(String[] args) throws IOException {
        TcpServer tcpServer = new TcpServer(1024);
        tcpServer.start();
    }
}

客户端:

public class TcpClient {
    private Socket socket = null;
    private String serverIp;
    private int serverPort;

    public TcpClient(String serverIp,int serverPort) throws IOException {
        socket = new Socket(serverIp,serverPort);//客户端随机分配端口号
        this.serverIp = serverIp;
        this.serverPort = serverPort;
    }

    public void start(){
        try(InputStream inputStream = socket.getInputStream();
            OutputStream outputStream = socket.getOutputStream()) {
            //创建流对象进行写与读
            while(true){
                Scanner scanner = new Scanner(System.in);//用来写入
                PrintWriter printWriter = new PrintWriter(outputStream);//包装output流对象
                String request = scanner.next();//写请求
                if(request.equals("exit")){
                    System.out.println("结束与服务器连接");
                    break;
                }
                //把请求放到流对象中写出去
                printWriter.println(request);
                printWriter.flush();//刷新缓冲区
                Scanner responseScanner = new Scanner(inputStream);
                String response = responseScanner.next();//读服务器的响应
                System.out.println(response);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) throws IOException {
        TcpClient tcpClient = new TcpClient("127.0.0.1",1024);
        tcpClient.start();
    }
}

服务器打印:

 

客户端打印:

 

 


 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Jupyter Notebook 遇上 NebulaGraph,可视化探索图数据库

在之前的《手把手教你用 NebulaGraph AI 全家桶跑图算法》中&#xff0c;除了介绍了 ngai 这个小工具之外&#xff0c;还提到了一件事有了 Jupyter Notebook 插件: https://github.com/wey-gu/ipython-ngql&#xff0c;可以更便捷地操作 NebulaGraph。 本文就手把手教你咋在 J…

设备管理平台:采用以可靠性为中心的维护策略的优势

在如今的工业领域&#xff0c;以可靠性为中心的维护策略正逐渐成为企业数字化转型的核心。无论是混合还是离散自动化应用&#xff0c;优化维护和工作流程实践已经成为提高利润、降低停机时间、增强运营和生产性能的不可或缺的一环。在这个过程中&#xff0c;设备管理系统与物联…

汽车BOOTLOADER开发经历

鄙人参与电动汽车BOOTLOADER开发近三年&#xff0c;从完全没有这方面的基础到参与国内外大小知名或不知名车企的电动车三大件的BOOTLOADER开发&#xff0c;总结了以下一点学习心得。 1.熟悉基本术语含义 诊断、寻址方式、FBL、擦除、驱动 2.熟悉国际标准、UDS服务格式 汽车的…

Redis BitMap/HyperLogLog/GEO/布隆过滤器案例

面试问题&#xff1a; 抖音电商直播&#xff0c;主播介绍的商品有评论&#xff0c;1个商品对应了1系列的评论&#xff0c;排序展现取前10条记录用户在手机App上的签到打卡信息&#xff1a;1天对应1系列用户的签到记录&#xff0c;新浪微博、钉钉打卡签到&#xff0c;来没来如何…

我开源的 c#+wpf 模仿网易云音乐播放器

MusicApp 介绍 gitee地址&#xff1a;https://gitee.com/liu_guo_feng/music-app 我开源的 c#wpf 模仿网易云音乐播放器 项目页面功能完成列表 首页(待完善) 每日推荐音乐 歌单详情 带播放列表 歌词页(待完善) 换肤功能(待完善) 系统托盘 … 预览 仅供学习使用 不作任何商业用…

类图的6种关系和golang应用

文章目录 1. 依赖和关联1.1 依赖&#xff08;Dependency&#xff09;概念类图示例代码示例 1.2 关联&#xff08;Association&#xff09;概念类图示例代码示例 2. 组合和聚合&#xff08;特殊的关联关系&#xff09;2.1 聚合&#xff08;Aggregation&#xff09;概念类图示例代…

9、Kubernetes核心技术 - Volume

目录 一、概述 二、卷的类型 三、emptyDir 四、hostPath 五、NFS 5.1、master服务器上搭建nfs服务器 5.2、各个slave节点上安装nfs客户端 5.3、创建Pod 六、PV和PVC 6.1、PV 6.1.1、PV资源清单文件示例 6.1.2、PV属性说明 6.1.3、PV的状态 6.2、PVC 6.2.1、PVC资…

Chapter 13: Network Programming | Python for Everybody 讲义笔记_En

文章目录 Python for Everybody课程简介Network ProgrammingNetworked programsHypertext Transfer Protocol - HTTPThe world’s simplest web browserRetrieving an image over HTTPRetrieving web pages with urllibReading binary files using urllibParsing HTML and scra…

如何实现对主机的立体监控?

主机监控是保证系统稳定性和性能的重要环节之一&#xff0c;那应该如何实现对主机的立体监控&#xff1f; 本期EasyOps产品使用最佳实践&#xff0c;我们将为您揭晓&#xff1a; 主机应该如何分组和管理&#xff1f; 主机监控应该关注哪些关键性指标&#xff1f; 背 景 通…

利用线程池多线程并发实现TCP两端通信交互,并将服务端设为守护进程

文章目录 实现目标实现步骤封装日志类封装线程池封装线程封装锁封装线程池 TCP通信的接口和注意事项accept TCP封装任务客户端Client.hppClient.cc 服务端Server.hpp Server.cc实现效果 守护进程服务端守护进程化 实现目标 利用线程池多线程并发实现基于TCP通信的多个客户端与…

vue3 excel 导出功能

1.安装 xlsx 库 npm install xlsx2.创建导出函数 src/utils/excelUtils.js import * as XLSX from xlsx;const exportToExcel (fileName, datas, sheetNames) > {// 创建工作簿const wb XLSX.utils.book_new()for (let i 0; i < datas.length; i) {let data datas…

Q-Tester 3.8:适用于开发、生产和售后的诊断测试软件

Q-Tester是一款简易使用的诊断测试软件&#xff0c;同时也是一款基于ODX&#xff08;ASAM MCD-2D/ISO 22901-1&#xff09;国际标准的工程诊断仪&#xff0c;通过该诊断仪可实现与ECU控制之间的数据交互。这一方案的优势是&#xff0c;在功能方面确定并完成相关开发工作后&…

文章采集伪原创发布工具-147采集

在当今信息爆炸的时代&#xff0c;企业和个人都意识到了获取高质量、原创的内容的重要性。然而&#xff0c;手动撰写大量的原创内容是一项耗时费力的任务。为了解决这个问题&#xff0c;我向您介绍一款颠覆性的数据采集工具——147采集。 147采集是一款专业且高效的数据采集软件…

【干货】商城系统的重要功能特性介绍

电子商务的快速发展&#xff0c;商城系统成为了企业开展线上销售的重要工具。一款功能强大、用户友好的商城系统能够有效提升企业的销售业绩&#xff0c;提供良好的购物体验。下面就商城系统的重要功能特性作一些简单介绍&#xff0c;帮助企业选择合适的系统&#xff0c;打造成…

软件测试面试【富途面经分享】

目录 一面面经&#xff08;1h&#xff09; 二面面经 一面面经&#xff08;1h&#xff09; 一、对白盒黑盒灰盒测试的理解 答&#xff1a; 1、黑盒测试就当整个程序是个黑盒子&#xff0c;我们看不到它里面做了什么事情&#xff0c;只能通过输入输出看是否能得到我们所需的来…

Linux初识网络基础

目录 网络发展 认识“协议 ” 网络协议 OSI七层模型&#xff1a; TCP/IP五层&#xff08;或四层&#xff09;模型 网络传输基本流程 网络传输流程图&#xff1a; 数据包封装和封用 网络中的地址 认识IP地址&#xff1a; 认识MAC地址&#xff1a; 网络发展 1.独立…

【MySQL】增删查改基础

文章目录 一、创建操作1.1 单行插入1.2 多行插入1.3 插入否则替换更新1.4 替换replace 二、查询操作2.1 select查询2.2 where条件判断2.3 order by排序2.4 limit筛选分页结果 三、更新操作四、删除操作4.1 删除一列4.2 删除整张表数据 五、插入查询结果 CRUD : Create(创建), R…

vue去掉所有输入框两边空格,封装指令去空格,支持Vue2和Vue3,ElementUI Input去空格

需求背景 就是页面很多表单输入框&#xff0c;期望在提交的时候&#xff0c;都要把用户两边的空格去掉 ❌使用 vue 的指令 .trim 去掉空格 中间会输入不了空格&#xff0c; 比如我想输入 你好啊 中国, 这中间的空格输入不了&#xff0c;只能变成 你好啊中国 ❌在提交的时候使用…

leetcode每日一练-第278题-第一个错误的版本

一、思路 二分查找——因为它可以快速地将版本范围缩小一半&#xff0c;从而更快地找到第一个坏版本。 二、解题方法 维护一个左边界 left 和一个右边界 right&#xff0c;在每一步循环中&#xff0c;我们计算中间版本 mid&#xff0c;然后检查它是否是坏版本。如果是坏版本…

router和route的区别

简单理解为&#xff0c;route是用来获取路由信息的&#xff0c;router是用来操作路由的。 一、router router是VueRouter的实例&#xff0c;通过Vue.use(VueRouter)和VueRouter构造函数得到一个router的实例对象&#xff0c;这个对象中是一个全局的对象&#xff0c;他包含了所…