javaSE————网络编程套接字

news2025/4/19 15:52:24

网络编程套接字~~~~~

好久没更新啦,蓝桥杯爆掉了,从今天开始爆更嗷;

1,网络编程基础

为啥要有网络编程呢,我们进行网络通信就是为了获取丰富的网络资源,说实话真的很神奇,想想我们躺在床上,通过网络,访问到世界上成千上万的人做好的网络资源,我们甚至能看到一辈子都看不到的景色,这些资源本质上都是二进制资源或者是文本资源,我们通过网络可以让很多人来访问,这个就是网络编程,当年的网络就是个看报的,谁能想到发展到这样呢?

网络编程的概念:网络上的主机,通过不同的进程,以编程的方式实现网络通信;

同一个主机下的不同进程实现网络通信也是网络编程;

网络编程的基本概念:

接收端:数据接收端进程,目标主机;

发送端:数据发送端进程,源主机;

收发端:发送接收两端;

请求:请求数据的发送;

响应:响应数据的发送;

就像我们去餐厅点一份蛋炒饭就是请求,厨师给我们拿来蛋炒饭就是回应;

服务端:提供服务的一方,返回服务资源;

客户端:获取服务的一方;


2,Socket套接字

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

Socket套接字主要针对传输层协议划分为三类:

我们这里就学两个核心协议

1,TCP,有连接,可靠传输,面向字节流,全双工;

2,UDP,无连接,不可靠传输,面向数据报,全双工;

这里的有连接和无连接都是抽象的概念,虚拟上的连接;

TCP协议保存了对端的信息,A和B通信,A和B先建立连接,A保存了B的信息,B也保存了A的信息,他们彼此都知道谁与自己建立的连接,而UDP不保存对方的信息;

可靠传输和不可靠传输是什么呢?

网络上,数据的传输是会发生丢包的情况的,可能会受到外界的干扰,那么可靠传输就能避免这样的丢包吗,事实上是不能的,可靠传输只是保证尽可能提高传输成功的概率,如果出现了丢包,也能感知到,不可靠传输的话就是把数据发送之后就不管了,那么这样的话我们就使用可靠传输就好了呀,还要UDP干嘛呢,凡是必有代价,UDP的话速度会更快;

面向字节流和面向数据报呢?

面向字节流:读写数据的时候,以字节为单位;

面向数据报:读写数据的时候,以一个数据报为单位;

全双工和半双工呢?

全双工:一个通信联络,能读也能写;

半双工:只支持单向通信,要么读,要么写;

我们使用Socket api来进行网络编程,我们之前提到计算机中的“文件”有狭义和广义的概念,硬件设备可以抽象成文件,统一管理;电脑上的网卡就是Socket文件;


3,UDP数据报套接字编程

我们先来介绍UDP需要的API:

DatagramSocket

UDP的Socket用于接收和发送UDP数据报;

构造方法:

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

方法:

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

DatagramPacket

datagramPacket是UDP socket发送和接收的数据报;

构造方法:

方法签名方法说明
DatagramPacket(byte[] bytes, int length)

构造一个DatagramPacket用来接收数据报,接收的数据呢保存在字节数组中,接收指定的长度

DatagramPacket(byte[] bytes, int offset, int length, SocketAddress address)构造一个DatagramPacket用来发送数据报,发送的数据为字节数组,从0开始,到指定长度,address指定制定目的主机的IP和端口号

方法:

方法签名方法说明
InetAddress  getAddress()

从接收的数据报中,获取发送端主机IP;或者从发送的数据报中,获取接收端主机IP;

int getPort()从接收的数据报中,获取接收端主机的端口号;或者从发送的数据报中,或缺发送端主机的端口号;
byte[] getData()获取数据报中的数据; 
getSocketAddress()直接获取到当前数据报的IP和端口号;

 

讲完方法了,我们来模拟一个回显服务器,这里就是模拟,回显服务器就是自问自答;

我们来用UDP模拟一下:

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.Socket;
import java.net.SocketException;

public class UdpEchoServer {
    private DatagramSocket socket = null;
    public UdpEchoServer(int port) throws SocketException {
        socket = new DatagramSocket(port);//指定端口号,让服务器来使用
    }
    public void start() throws IOException {
        //启动服务器
        while(true){
            //循环一次,就意味着处理了一次请求
            //1读取请求并解析
            DatagramPacket packet = new DatagramPacket(new byte[4090],4090);
            socket.receive(packet);
            String request = new String(packet.getData(),0,packet.getLength());
            //2根据请求计算响应
            String response = process(request);
            //3把响应返回给客户端
            DatagramPacket packet1 = new DatagramPacket(response.getBytes(),response.getBytes().length
            ,packet.getSocketAddress());
            socket.send(packet1);
            //4打印日志
            System.out.printf("[%s:%d] req: %s, resp: %s\n",
                    packet.getAddress().toString(), packet.getPort(), request, response);
        }
    }

    public String process(String request){
        return request;
    }

    public static void main(String[] args) throws IOException {
        UdpEchoServer udpEchoServer = new UdpEchoServer(9090);
        udpEchoServer.start();
    }
}

模拟服务器:难点用注释标注了,看不明白就来私信我! 

接下来实现客户端代码:

import java.io.IOException;
import java.net.*;
import java.util.Scanner;

public class UdpEchoClient {
    private DatagramSocket socket = null;
    private String SeverIp;
    private int SeverPort;
    public UdpEchoClient(String IP,int port) throws SocketException {
        this.SeverIp = IP;
        this.SeverPort = port;
        socket = new DatagramSocket();
    }

    public void start() throws IOException {
        Scanner scanner = new Scanner(System.in);
        while(true){
            System.out.println("请输入要发送的内容");
            while(!scanner.hasNext()){
                break;
            }
            String request = scanner.nextLine();
            DatagramPacket datagramPacket = new DatagramPacket(request.getBytes(),request.getBytes().length
            , InetAddress.getByName(SeverIp),SeverPort);
            socket.send(datagramPacket);

            DatagramPacket datagramPacket1 = new DatagramPacket(new byte[4090],4090);
            socket.receive(datagramPacket1);

            String response = new String(datagramPacket1.getData(),0,datagramPacket1.getLength());
            System.out.println(response);
        }
    }

    public static void main(String[] args) throws IOException {
        UdpEchoClient udpEchoClient = new UdpEchoClient("127.0.0.1",9090);
        udpEchoClient.start();
    }

}

我们把两个代码的都运行下 

客户端

服务器


4,TCP流套接字编程

接下来我们来实现TCP的网络编程,还是先来学习一下方法:

SeverSocket

SeverSocket呢是创建TCP服务端Socket的ApI;

构造方法:

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

方法:

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

Socket

socket是客户端Socket,或服务端中接收到客户端建立连接(accept)的请求后,返回的服务端Socket;

构造方法:

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

方法:

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

服务端

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;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

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("启动服务器");
        ExecutorService executorService = Executors.newCachedThreadPool();
        while(true){
            Socket socketClient = serverSocket.accept();
            executorService.submit(()->{
                try {
                    processConnection(socketClient);
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            });
        }
    }

    public void processConnection(Socket socketClient) throws IOException {
        System.out.println("客户端上线"+ socketClient.getInetAddress()+socketClient.getPort());
        try(InputStream inputStream = socketClient.getInputStream();
            OutputStream outputStream = socketClient.getOutputStream()){
            Scanner scanner = new Scanner(inputStream);
            PrintWriter printWriter = new PrintWriter(outputStream);
            while (true){
                if(!scanner.hasNext()){
                    System.out.println("客户端下线"+ socketClient.getInetAddress()+socketClient.getPort());
                    break;
                }
                String request = scanner.nextLine();
                String response = process(request);
                printWriter.println(response);
                printWriter.flush();
                System.out.printf("[%s:%d] req: %s, resp: %s\n", socketClient.getInetAddress(), socketClient.getPort(),
                        request, response);
            }
        }
    }

    public String process(String request){
        return request;
    }

    public static void main(String[] args) throws IOException {
        TcpEchoServer tcpEchoServer = new TcpEchoServer(9090);
        tcpEchoServer.start();
    }
}


这里引入线程池来处理多个请求,避免其他请求发来处理不了的情况;

客户端:
 

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

public class TcpEchoClient {
    private Socket socket = null;
    public TcpEchoClient(String IP,int port) throws IOException {
        socket = new Socket(IP,port);
    }

    public void start() throws IOException {
        Scanner scanner = new Scanner(System.in);
        try(InputStream inputStream = socket.getInputStream();
            OutputStream outputStream = socket.getOutputStream()){
            Scanner scanner1 = new Scanner(inputStream);
            PrintWriter printWriter = new PrintWriter(outputStream);

            while(true){
                System.out.println("请输入要发送的信息");
                String request = scanner.nextLine();
                printWriter.println(request);
                printWriter.flush();
                String response = scanner1.nextLine();
                System.out.println(response);
            }
        }
    }

    public static void main(String[] args) throws IOException {
        TcpEchoClient tcpEchoClient = new TcpEchoClient("127.0.0.1",9090);
        tcpEchoClient.start();
    }
}

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

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

相关文章

FreeRTOS二值信号量详解与实战教程

FreeRTOS二值信号量详解与实战教程 📚 作者推荐:想系统学习FreeRTOS嵌入式开发?请访问我的FreeRTOS开源学习库,内含从入门到精通的完整教程和实例代码! 1. 二值信号量核心概念解析 二值信号量(Binary Semaphore)是Fre…

赛灵思 XCVU440-2FLGA2892E XilinxFPGA Virtex UltraScale

XCVU440-2FLGA2892E 属于 Xilinx Virtex UltraScale 系列,是面向高端应用的旗舰 FPGA 器件。该系列产品以出色的高并行处理能力、丰富的逻辑资源和高速互联能力闻名,广泛用于 高性能计算、数字信号处理等对计算能力和带宽要求极高的场景。采用先进的 20n…

Spring Cloud Alibaba微服务-微服务介绍和搭建

1. 课程介绍 单体服务中有订单,用户,库存, 两个缺陷: a. 是以应用的维度进行负载均衡,资源占用大 b. 当其中一个模块宕机,整个应用就不能用了; nacos;ribbon,loadBa…

KALI安装JAVA8和切换JDK版本

一、安装JDK1.8 1、直接使用下面的地址下载java 1.8: https://repo.huaweicloud.com/java/jdk/8u202-b08/jdk-8u202-linux-x64.tar.gz 2、建立目录,将下载的jdk的安装包复制过去并进行解压 sudo mkdir -p /usr/local/java cp jdk-8u202-linux-x64.t…

今日行情明日机会——20250417

指数目前在区间内缩量震荡 2025年4月17日涨停主要行业方向分析 一、核心主线方向 化工(产能优化涨价预期) • 涨停家数:11家(最强方向)。 • 代表标的: ◦ 红宝丽(2连板)&#xff…

一篇讲完自动化测试基础-Python【万字详细讲解】12

✨博客主页: https://blog.csdn.net/m0_63815035?typeblog 💗《博客内容》:.NET、Java.测试开发、Python、Android、Go、Node、Android前端小程序等相关领域知识 📢博客专栏: https://blog.csdn.net/m0_63815035/cat…

极限编程(XP)简介及其价值观与最佳实践

目录 一、什么是极限编程(XP)二、极限编程的核心价值观1. 沟通2. 简单3. 反馈4. 勇气 三、极限编程的12个最佳实践1. 结对编程2. 40小时工作制3. 简单设计4. 代码规范5. 测试驱动开发(TDD)6. 系统隐喻7. 持续集成8. 重构9. 客户在…

四层板的蛇形走线技巧:原理、策略与应用

在四层板的设计过程中,蛇形走线是一种常见且重要的布线方式。它能够满足特定的设计需求,如调整信号线长度、实现等长布线等,但如果使用不当,也可能会带来一些负面影响,如增加信号衰减、引入电磁干扰等。以下将详细探讨…

面向对象—有理数类的设计

目录 1.代码呈现 1.1编写toString、equals方法 1.2测试代码 1.3有理数类的代码 2.论述题 3.有理类设计 1.代码呈现 1.1编写toString、equals方法 (1)toString方法 Overridepublic String toString(){if(this.v20){return "Undefined";}return this.v1 "/…

408数据结构绪论刷题001

答案:D 解析: • A选项:数据元素是组成数据对象的基本单位 ,它只是数据的基本个体,不能完整定义数据结构,所以A选项错误。 • B选项:数据对象是性质相同的数据元素的集合,仅仅描述…

Leetcode 3359. 查找最大元素不超过 K 的有序子矩阵【Plus题】

1.题目基本信息 1.1.题目描述 给定一个大小为 m x n 的二维矩阵 grid。同时给定一个 非负整数 k。 返回满足下列条件的 grid 的子矩阵数量: 子矩阵中最大的元素 小于等于 k。 子矩阵的每一行都以 非递增 顺序排序。 矩阵的子矩阵 (x1, y1, x2, y2) 是通过选择…

文件系统 软硬连接

🌻个人主页:路飞雪吖~ 🌠专栏:Linux 目录 一、理解文件系统 🌠磁盘结构 二、软硬连接 🌟软硬链接 🌠软链接: 🌠硬链接: 🌟理解软硬链接的应…

计算机视觉——JPEG AI 标准发布了图像压缩新突破与数字图像取证的挑战及应对策略

概述 今年2月,经过多年旨在利用机器学习技术开发一种更小、更易于传输和存储且不损失感知质量的图像编解码器的研究后,JPEG AI国际标准正式发布。 来自JPEG AI官方发布流,峰值信噪比(PSNR)与JPEG AI的机器学习增强方法…

Oracle 19c部署之数据库软件安装(二)

在完成了Oracle Linux 9的初始化配置之后,我们准备安装Oracle 19c数据库软件。 Oracle数据库支持两种主要的安装方式:图形化安装和静默安装。这两种方法各有优缺点,选择哪种取决于你的具体需求、环境配置以及个人偏好。 图形化安装 图形化安…

在Vmware15(虚拟机免费) 中安装纯净win10详细过程

一、软件备选 1. VMware15.5.1 网盘下载地址 链接: https://pan.baidu.com/s/1y6GLJ2MG-1tomWblt3otsg?pwdim8e 提取码: im8e 2. windows镜像下载 去官网下载ios包 链接:https://www.microsoft.com/zh-cn/software-download/windows10 二、在VMware15.5.1下安装w…

[Spark]深入解密Spark SQL源码:Catalyst框架如何优雅地解析你的SQL

本文内容组织形式 总结具体例子执行语句解析层优化层物理计划层执行层 猜你喜欢PS 总结 先写个总结,接下来会分别产出各个部分的源码解析,Spark SQL主要分为以下五个执行部分。 具体例子 接下来举个具体的例子来说明 执行语句 SELECT name, age FR…

基于Flask的漏洞挖掘知识库系统设计与实现

基于Flask的漏洞挖掘知识库系统设计与实现 一、系统架构设计 1.1 整体架构 本系统采用经典的三层Web架构,通过Mermaid图展示的组件交互流程清晰呈现了以下核心模块: 前端展示层:基于Bootstrap5构建响应式界面业务逻辑层:Flask…

ECharts散点图-散点图8,附视频讲解与代码下载

引言: ECharts散点图是一种常见的数据可视化图表类型,它通过在二维坐标系或其它坐标系中绘制散乱的点来展示数据之间的关系。本文将详细介绍如何使用ECharts库实现一个散点图,包括图表效果预览、视频讲解及代码下载,让你轻松掌握…

Langchain-构建向量数据库和检索器

向量数据库安装 pip install langchain-chroma 文档》向量存储》向量数据库。 和0416 提示词工程相同。 初始化 import osfrom langchain_chroma import Chroma from langchain_community.chat_message_histories import ChatMessageHistory from langchain_core.documents im…

首席人工智能官(Chief Artificial Intelligence Officer,CAIO)的详细解析

以下是**首席人工智能官(Chief Artificial Intelligence Officer,CAIO)**的详细解析: 1. 职责与核心职能 制定AI战略 制定公司AI技术的长期战略,明确AI在业务中的应用场景和优先级,推动AI与核心业务的深度…