【JAVA学习笔记】 68 - 网络——TCP编程、UDP编程

news2025/1/15 16:28:22

项目代码

https://github.com/yinhai1114/Java_Learning_Code/tree/main/IDEA_Chapter21/src

网络

一、网络相关概念

1.网络通讯

1.概念:两台设备之间通过网络实现数据传输

2.网络通信:将数据通过网络从一台设备传输到另一台设备

3. java.net包下提供了一系列的类或接口,供程序员使用,完成网络通信

1.

2.网络

概念:两台或多台设备通过一定物理设备连接起来构成了网络

2.根据网络的覆盖范围不同,对网络

进行分类:

局域网:覆盖范围最小,仅仅覆盖一个教室或一个机房

城域网:覆盖范围较大,可以覆盖个城市

广域网:覆盖范围最大,可以覆盖全国,甚至全球,万维网是广域网的代表

3.IP地址

1.概念:用于唯一标识网络中的每台计算机/主机

2.查看ip地址: ipconfig

3. ip地址的表示形式:点分十进制xx.xxx.xx

4.每一个十进制数的范围: 0~255

5. ip地址的组成=网络地址+主机地址,比如: 192.168.16.69

        对于IPV4:4个字节(32位)表示 1个字节的范围是0~255 

        

6. iIPv6是互联网工程任务组设计的用于替代IPv4的下一 代IP协议,其地址数量号称可以为全世界的每一粒沙子编上一 个地址。使用16机制表示

        

7.由于IPv4最大的问题在于网络地址资源有限,严重制约了互联网的应用和发展。IPv6的使用,不仅能解决网络地址资源数量的问题,而且也解决了多种接入设备连入互联的障碍

        

        

4.域名

1. www.baidu.com

2.好处:为了方便记忆,解决记ip的困难

3.概念:将ip地址映射成域名

5.端口号

1.概念:用于标识计算机上某个特定的网络程序

2.表示形式:以整数形式,范围0~ 65535 (两个字节表示)

3.0~1024已经被占用,比如ssh 22, ftp 21, smtp 25 http 80

        在网络开发中,不要使用0 - 1024的端口

4.常见的网络程序端口号:

        tomcat :8080

        mysql:3306

        oracle:1521

        sqlserver:1433

6.网络通讯协议

TCP/IP (Transmission Control Protocol/Internet Protocol)的简写,

中文译名为传输控制协议/因特网互联协议,又叫网络通讯协议,这个协议是Internet最基本的协议、Internet国际互联网络的基础,简单地说,就是由网络层的IP协议和传输层的TCP协议组成的。

TCP协议:传输控制协议

1.使用TCP协议前,须先建立TCP连接,形成传输数据通道

2.传输前,采用"三次握手"方式,是可靠的

        

3. TCP协议进行通信的两个应用进程:客户端、服务端

4.在连接中可进行大数据量的传输

5.传输完毕,需释放已建立的连接,效率低

UDP协议:

1.将数据、源、目的封装成数据包,不需要建立连接

2.每个数据报的大小限制在64K内,不适合传输大量数据

3.因无需连接,故是不可靠的

4.发送数据结束时无需释放资源(因为不是面向连接的),速度快

5.举例:厕所通知

二、InetAddress类

1.相关方法

1.获取本机InetAddress对象getL ocalHost

2.根据指定主机名/域名获取ip地址对象getByName

3.获取InetAddress对象的主机名getHostName

4.获取InetAddress对象的地址getHostAddress

public class API_ {
    public static void main(String[] args) throws UnknownHostException {

        //1. 获取本机的InetAddress 对象
        InetAddress localHost = InetAddress.getLocalHost();
        System.out.println(localHost);//DESKTOP-9UOSPK2/192.168.0.105

        //2. 根据指定主机名 获取 InetAddress对象
        InetAddress host1 = InetAddress.getByName("DESKTOP-9UOSPK2");
        System.out.println("host1=" + host1);//DESKTOP-9UOSPK2/192.168.0.105

        //3. 根据域名返回 InetAddress对象, 比如 www.baidu.com 对应
        InetAddress host2 = InetAddress.getByName("www.baidu.com");
        System.out.println("host2=" + host2);//www.baidu.com / 14.119.104.254

        //4. 通过 InetAddress 对象,获取对应的地址
        String hostAddress = host2.getHostAddress();//IP 110.242.68.4
        System.out.println("host2 对应的ip = " + hostAddress);//14.119.104.254

        //5. 通过 InetAddress 对象,获取对应的主机名/或者的域名
        String hostName = host2.getHostName();
        System.out.println("host2对应的主机名/域名=" + hostName); // www.baidu.com
    }
}

三、Socket

1.基本介绍

1.套接字(Socket)开发网络应用程序被厂泛采用,以至于成为事实上的标准。

2.通信的两端都要有Socket,是两台机器间通信的端点

3.网络通信其实就是Socket间的通信。

4. Socket允许程序把网络连接当成一个流,数据在两个Socket间通过I0传输。

5. 一般主动发起通信的应用程序属客户端,等待通信请求的为服务端

示意图:

2.TCP网络通信编程

基本介绍

1.基于客户端一服务端的网络通信

2.底层使用的是TCP/IP协议

3.应用场景举例:客户端发送数据,服务端接受并显示

4.基于Socket的TCP编程

           

应用案例1

1.编写一个服务器端,和一个客户端

2.服务器端在9999端口监听

3.客户端连接到服务器端,发送"hello,server”,然后退出

4.服务器端接收到客户端发送的信息,输出,并退出

服务端

public class SocketTCP01Server {
    public static void main(String[] args) throws IOException {
        //要求 该port端口没有其他服务在监听9999
        //ServerSocket可以对应多个socket //细节:这个ServerSocket 可以通过accept() 返回多个Socket[多个客户端连接服务器的并发]
        ServerSocket serverSocket = new ServerSocket(9999);
        System.out.println("服务端在9999端口监听,等待连接");
        //如果没有客户端连接9999程序会阻塞,等待连接
        //如果有客户端连接,则会返回socket对象,程序继续
        Socket socket = serverSocket.accept();
        System.out.println("服务端socket =" + socket.getClass());
        //io读取
        InputStream inputStream = socket.getInputStream();
        byte[] buf = new byte[1024];
        int readLen = 0;
        while ((readLen = inputStream.read(buf) )!= -1){
            System.out.println(new String(buf,0,readLen));
        }
        inputStream.close();
        socket.close();
        serverSocket.close();
    }
}

客户端

public class SocketTCP01Client {
    public static void main(String[] args) throws IOException {
        //解读: 连接本机的 9999端口, 如果连接成功,返回Socket对象
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        System.out.println("客户端 socket返回=" + socket.getClass());
        //2. 连接上后,生成Socket, 通过socket.getOutputStream()
        //   得到 和 socket对象关联的输出流对象
        OutputStream outputStream = socket.getOutputStream();
        //3. 通过输出流,写入数据到 数据通道
        outputStream.write("hello world".getBytes());
        //4. 关闭流对象和socket, 必须关闭
        outputStream.close();
        socket.close();
        System.out.println("客户端退出...");

    }
}

应用案例2

1.编写个服务端,和一个客户端

2.服务器端在9999端口监听

3.客户端连接到服务端,发送"hello, server"并接收服务器端回发的"hello,client, 再退出

4.服务器端接收到客户端发送的信息,输出,并发送"hello, client", 再退出

注意,在案例1的基础上更改,需要在服务端Out字节流完之后告诉客户端我发送完毕,否则程序不知道什么时候结束,需要设置一个结束标记socket.shutdownOutput()(案例1是因为客户端关闭了才结束的。)

服务端

public class SocketTCP02Server {
    public static void main(String[] args) throws IOException {
        //该port端口没有其他服务在监听9999
        //1.创建ServerSocket服务端
        //ServerSocket可以对应多个socket //细节:这个ServerSocket 可以通过accept() 返回多个Socket[多个客户端连接服务器的并发]
        ServerSocket serverSocket = new ServerSocket(9999);
        System.out.println("服务端在9999端口监听,等待连接");
        //2.获取socket管道
        //如果没有客户端连接9999程序会阻塞,等待连接
        //如果有客户端连接,则会返回socket对象,程序继续
        Socket socket = serverSocket.accept();
        System.out.println("服务端socket =" + socket.getClass());
        //3.回去客户端写入管道的数据
        InputStream inputStream = socket.getInputStream();
        //4.io读取
        byte[] buf = new byte[1024];
        int readLen = 0;
        while ((readLen = inputStream.read(buf) )!= -1){
            System.out.println(new String(buf,0,readLen));
        }
        socket.shutdownInput();//设置关闭读取标记
        //5.获取socket相关联的输出流
        OutputStream outputStream = socket.getOutputStream();
        outputStream.write("hello client by server".getBytes());

        //6.关闭流
        inputStream.close();
        outputStream.close();
        socket.close();
        serverSocket.close();
    }
}

 客户端

public class SocketTCP02Client {
    public static void main(String[] args) throws IOException {
        //解读: 连接本机的 9999端口, 如果连接成功,返回Socket对象
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        System.out.println("客户端 socket返回=" + socket.getClass());
        //2. 连接上后,生成Socket, 通过socket.getOutputStream()
        //   得到 和 socket对象关联的输出流对象
        OutputStream outputStream = socket.getOutputStream();
        //3. 通过输出流,写入数据到 数据通道
        outputStream.write("hello server by client ".getBytes());
        socket.shutdownOutput();//设置发送完毕标记
        //4.获取和socket关联的输入流,读取数据(字节),并显示
        InputStream inputStream = socket.getInputStream();
        byte[] buf = new byte[1024];
        int readLen = 0;
        while ((readLen = inputStream.read(buf)) != -1){
            System.out.println(new String(buf,0,readLen));
        }
        //5. 关闭流对象和socket, 必须关闭
        outputStream.close();
        socket.close();
        System.out.println("客户端退出...");
    }
}

应用案例3

1.编写一个服务端,和一个客户端

2.服务端在9999端口监听

3.客户端连接到服务端,发送"hello, server",并接收服务端回发的"hello,client",再退出

4.服务端接收到客户端发送的信息,输出,并发送"hello, client",再退出

服务端

public class SocketTCP03Server {
    public static void main(String[] args) throws IOException {
        //该port端口没有其他服务在监听9999
        //1.创建ServerSocket服务端
        //ServerSocket可以对应多个socket //细节:这个ServerSocket 可以通过accept() 返回多个Socket[多个客户端连接服务器的并发]
        ServerSocket serverSocket = new ServerSocket(9999);
        System.out.println("服务端在9999端口监听,等待连接");
        //2.获取socket管道
        //如果没有客户端连接9999程序会阻塞,等待连接
        //如果有客户端连接,则会返回socket对象,程序继续
        Socket socket = serverSocket.accept();
        System.out.println("服务端socket =" + socket.getClass());
        //3.转成字符流 获取客户端写入管道的数据
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        //4.io读取
        String line;
        while ((line =  bufferedReader.readLine())!= null){
            System.out.println(line + "\r\n");
        }
        socket.shutdownInput();//设置关闭读取标记

        //5.获取socket相关联的输出流 并转成字符流
        BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
        //6.写入数据
        bufferedWriter.write("hello client by server");
        bufferedWriter.newLine();//插入一个换行符,表示写入的内容结束,注意,要求对方使用readLine()!!!!
        bufferedWriter.flush();
        socket.shutdownOutput();//设置发送完毕标记

        //6.关闭流
        socket.close();
        serverSocket.close();
    }
}

客户端

public class SocketTCP03Client {
    public static void main(String[] args) throws IOException {
        //解读: 连接本机的 9999端口, 如果连接成功,返回Socket对象
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        System.out.println("客户端 socket返回=" + socket.getClass());

        //2. 连接上后,生成Socket, 通过socket.getOutputStream()
        //   得到 和 socket对象关联的输出流对象
        //3.将output转成writer 通过输出流,写入数据到 数据通道
        BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
        bufferedWriter.write("hello server by client ");
        bufferedWriter.newLine();//插入一个换行符,表示写入的内容结束,注意,要求对方使用readLine()!!!!
        bufferedWriter.flush();//如果使用的字符流,需要手动刷新,否则数据不会写入数据通道
        //4.关闭输出流
        socket.shutdownOutput();//设置发送完毕标记 即便是字符流而且没有显式的定义OutputStream也需要关闭

        //5.获取和socket关联的输入流,读取数据(字节),并显示
        //6.转成Reader
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        String line;
        while ((line = bufferedReader.readLine())!=null){
            System.out.println(line + "\r\n");
        }
        // socket.shutdownInput();
        //5. 关闭流对象和socket, 必须关闭

        bufferedReader.close();
        socket.close();
        System.out.println("客户端退出...");
    }
}

应用案例4

1.编写一个服务端,和一个客户端

2.服务器端在8888端口监听

3.客户端连接到服务端,发送一张图片e:\qie.png

4.服务器端接收到客户端发送的图片,保存到src下,发送"收到图片"再退出

5.客户端接收到服务端发送的"收到图片",再退出

6.该程序要求使用StreamUtils.java

客户端

public class SocketTCP03Client {
    public static void main(String[] args) throws IOException {
        //解读: 连接本机的 9999端口, 如果连接成功,返回Socket对象
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        System.out.println("客户端 socket返回=" + socket.getClass());

        //2. 连接上后,生成Socket, 通过socket.getOutputStream()
        //   得到 和 socket对象关联的输出流对象
        //3.将output转成writer 通过输出流,写入数据到 数据通道
        BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
        bufferedWriter.write("hello server by client ");
        bufferedWriter.newLine();//插入一个换行符,表示写入的内容结束,注意,要求对方使用readLine()!!!!
        bufferedWriter.flush();//如果使用的字符流,需要手动刷新,否则数据不会写入数据通道
        //4.关闭输出流
        socket.shutdownOutput();//设置发送完毕标记 即便是字符流而且没有显式的定义OutputStream也需要关闭

        //5.获取和socket关联的输入流,读取数据(字节),并显示
        //6.转成Reader
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        String line;
        while ((line = bufferedReader.readLine())!=null){
            System.out.println(line + "\r\n");
        }
        // com.yinhai.socket.shutdownInput();
        //5. 关闭流对象和socket, 必须关闭

        bufferedReader.close();
        socket.close();
        System.out.println("客户端退出...");
    }
}

服务端

public class SocketTCP04Server {
    public static void main(String[] args) throws Exception {
        ServerSocket serverSocket = new ServerSocket(8888);
        Socket socket = serverSocket.accept();
        InputStream inputStream = socket.getInputStream();
        BufferedInputStream bufferedInputStream = new BufferedInputStream(inputStream);
        //该方法已经读到了数组
        byte[] bytes = StreamUtils.streamToByteArray(bufferedInputStream);

        //将得到的byte数组写入到指定的路径
        String receptionPath = "src\\com\\yinhai\\socket\\upload\\serverReceptionFile\\mikuByServer.jpg";
        BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(new FileOutputStream(receptionPath));
        bufferedOutputStream.write(bytes);
        bufferedOutputStream.flush();


        BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
        bufferedWriter.write("Server已收到客户端传来的文件,文件创建在" + receptionPath);
        bufferedWriter.newLine();
        bufferedWriter.flush();
        socket.shutdownOutput();


        //关闭流
        bufferedOutputStream.close();
        socket.close();
        bufferedInputStream.close();
        bufferedWriter.close();

    }
}

netset指令 

1. netstat - an可以查看当前主机网络情况,包括端口监听情况和网络连接情况

2. netstat - an | more可以分页显示

3.要求在dos控制台下执行

说明:

(1) Listening表示某个端口在监听

(2)如果有一个外部程序(客户端)(外部地址)连接到该端口,就会显示一条连接信息

使用netstat -anb | more可以查看是哪个程序正在监听 (需要以管理员身份启动)

细节

当客户端连接到服务端后,实际上客户端也是通过一个端口和服务端进行通讯的,这个端口是TCP/IP来分配的

特别注意

对于 ObjectOutputStream,它本身包含了更多的信息,能够更好地表示对象的边界,因此不需要显式地调用 shutdownOutput

当你使用 BufferedWriter 写入数据时,并没有类似于换行符的标志来表示数据的结束,所以你需要通过 shutdownOutput 来显式关闭输出流。

当你使用 OutputStream 直接写入字节时,也需要调用 shutdownOutput 来表示数据的结束。

因为在Java中,ObjectOutputStream 是用于将对象序列化为字节流的类。这个类会将对象的数据以及对象的类型信息一并写入输出流中。在这个过程中,ObjectOutputStream 会使用一些特殊的标记和格式来标识不同的对象,并在读取时还原这些信息。

底层的工作原理涉及到Java序列化机制,以下是一些关键概念:

  1. 序列化: 将对象转换为字节流的过程称为序列化。在Java中,对象要想序列化,必须实现 Serializable 接口。

  2. 对象流的特殊处理: ObjectOutputStreamObjectInputStream 会处理对象的序列化和反序列化。它们会使用特殊的标记来表示对象的类型、字段等信息。

  3. 对象边界: 在对象流中,每个写入的对象都被看作是一个边界。ObjectOutputStream 会负责处理对象的序列化,包括将对象的类型信息写入流中,以便在反序列化时正确还原对象。

由于 ObjectOutputStream 在写入对象时已经包含了足够的信息来表示对象的边界,所以在正常情况下,不需要显式地调用 shutdownOutput 来关闭输出流。关闭输出流的目的是告知对方数据发送结束,但对于 ObjectOutputStream 来说,对象边界的信息已经足够表示数据的结束。

总体来说,这是Java序列化机制和 ObjectOutputStream 的设计,使得在对象流中不需要显式关闭输出流,而字节流和字符流需要显式关闭以确保对方能够正确判断数据的结束。

3.UDP网络通信编程

1.基本介绍

1.类DatagramSocket和DatagramPacket[数据包/数据报]实现了基于UDP协议网络程序。

2. UDP数据报通过数据报套接字DatagramSocket发送和接收,系统不保证UDP数据报一定能够安全送到目的地,也不能确定什么时候可以抵达。

3. DatagramPacket 对象封装了UDP数据报,在数据报中包含了发送端的IP地址和端口号以及接收端的IP地址和端口号。

4. UDP协议中每个数据报都给出了完整的地址信息,因此无须建立发送方和接收方的连接

2.UDP说明

1.没有明确的服务端和客户端,演变成数据的发送端和接收端

2.接收数据和发送数据是通过DatagramSocket对象完成

3.将数据封装到DatagramPacket对象/装包

4.当接收到DatagramPacket对象,需要进行拆包,取出数据

5. DatagramSocket可以指定在哪个端接收数据

3.基本流程

1.核心的两个类/对象DatagramSocket与DatagramPacket

2.建立发送端,接收端(没有服务端和客户端概念)

3.发送数据前,建立数据包/报DatagramPacket对象

4.调用DatagramSocket的发送、接收方法

5.关闭DatagramSocket

4.案例1 

1.编写一个接收端A,和一个发送端B

2.接收端A在9999端口等待接收数据(receive)

3.发送端A向接收端B发送数据"hello ,明天吃火锅~

4.接收端B接收到发送端A发送的数据,回复"好的,明天见"再退出

5.发送端接收回复的数据,再退出

接收端

public class UDPReceiverA {
    public static void main(String[] args) throws IOException {
        //创建一个DatagramSocket对象 准备在9999端口接收
        DatagramSocket datagramSocket = new DatagramSocket(9999);
        //2.构建数据包,准备接收数据,一个数据包最大为64k
        byte[] buf = new byte[1024];
        DatagramPacket datagramPacket = new DatagramPacket(buf, buf.length);
        //3.调用接收方法,通过网络传输的packet对象填充到packet对象内
        datagramSocket.receive(datagramPacket);//如果没有数据包发送过来,就会阻塞等待
        //4.把packet拆包并显示
        int length = datagramPacket.getLength();//实际接收到的长度
        byte[] data = datagramPacket.getData();//接受数据
        String s = new String(data, 0, length);
        System.out.println(s);
        //接受到后发送ok回去
        byte[] bytes = "OK,I get it and hello UDPSender".getBytes();

        DatagramPacket datagramPacket1 =
                new DatagramPacket(bytes, bytes.length, InetAddress.getByName("192.168.0.105"),9998);
        datagramSocket.send(datagramPacket1);
        //5.关闭资源
        datagramSocket.close();
    }
}

发送端

public class UDPSenderB {
    public static void main(String[] args) throws IOException {
        //1.创建DatagramSocket对象,准备发送和数据,该对象 可以接受和发送
        DatagramSocket datagramSocket = new DatagramSocket(9998);//准备在9998等别人的发送数据
        //2.将需要发送的数据封装到packet对象中
        byte[] bytes = "hello UDPReceiver".getBytes();
        //说明:封装的DatagramPacket对象(内容字节数组,data.Length ,主机(IP) ,端口)
        DatagramPacket datagramPacket =
                new DatagramPacket(bytes, bytes.length, InetAddress.getByName("192.168.0.105"), 9999);
        datagramSocket.send(datagramPacket);
        byte[] buf = new byte[1024];
        DatagramPacket datagramPacket1 = new DatagramPacket(buf, buf.length);
        //3.调用接收方法,通过网络传输的packet对象填充到packet对象内
        datagramSocket.receive(datagramPacket1);//如果没有数据包发送过来,就会阻塞等待
        //4.把packet拆包并显示
        int length = datagramPacket1.getLength();//实际接收到的长度
        byte[] data = datagramPacket1.getData();//接受数据
        String s = new String(data, 0, length);
        System.out.println(s);
        datagramSocket.close();
    }
}

四、本章作业

1.

服务端

public class Homework01Server {
    public static void main(String[] args) throws IOException {
        //该port端口没有其他服务在监听9999
        //1.创建ServerSocket服务端
        //ServerSocket可以对应多个socket //细节:这个ServerSocket 可以通过accept() 返回多个Socket[多个客户端连接服务器的并发]
        ServerSocket serverSocket = new ServerSocket(9999);
        System.out.println("服务端在9999端口监听,等待连接");
        //2.获取socket管道
        //如果没有客户端连接9999程序会阻塞,等待连接
        //如果有客户端连接,则会返回socket对象,程序继续


        //3.转成字符流 获取客户端写入管道的数据

        //4.io读取
        while (true) {
            Socket socket = serverSocket.accept();
            BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            String line;
            String answer = null;
            while ((line = bufferedReader.readLine()) != null) {
                if (line.equals("name")) {
                    answer = "My name is yinhai";
                } else if ("hobby".equals(line)) {
                    answer = "My favor to compile program";
                }else if ("exit".equals(line)){
                    answer = "exit";
                }
                else {
                    answer = "Sorry , about that cant understand with me";
                }
            }
            socket.shutdownInput();

            //5.获取socket相关联的输出流 并转成字符流
            BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
            //6.写入数据
            bufferedWriter.write(answer);
            bufferedWriter.newLine();//插入一个换行符,表示写入的内容结束,注意,要求对方使用readLine()!!!!
            bufferedWriter.flush();
            socket.shutdownOutput();//设置发送完毕标记
            if (answer.equals("exit")){
                socket.close();
                break;
            }
        }

        //6.关闭流

        serverSocket.close();
    }
}

客户端

public class Homework01Client {
    public static void main(String[] args) throws IOException {
        while (true) {
            Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
            Scanner scanner = new Scanner(System.in);
            System.out.println("请输入你的问题");
            String next = scanner.next();

            BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
            bufferedWriter.write(next);
            bufferedWriter.newLine();//插入一个换行符,表示写入的内容结束,注意,要求对方使用readLine()!!!!
            bufferedWriter.flush();//如果使用的字符流,需要手动刷新,否则数据不会写入数据通道
            //4.关闭输出流
            socket.shutdownOutput();//设置发送完毕标记 即便是字符流而且没有显式的定义OutputStream也需要关闭

            //5.获取和socket关联的输入流,读取数据(字节),并显示
            //6.转成Reader
            BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            String line;
            while ((line = bufferedReader.readLine()) != null) {
                System.out.println(line + "\r\n");
            }
            if(next.equals("exit")){
                socket.close();
                break;
            }
            // com.yinhai.socket.shutdownInput();
            //5. 关闭流对象和socket, 必须关闭
            bufferedReader.close();
        }
        System.out.println("客户端退出...");
    }

}

2. 

发送端

public class Homework02SenderB {
    public static void main(String[] args) throws IOException {

        //1.创建 DatagramSocket 对象,准备在9998端口 接收数据
        DatagramSocket socket = new DatagramSocket(9998);

        //2. 将需要发送的数据,封装到 DatagramPacket对象
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入你的问题: ");
        String question = scanner.next();
        byte[] data = question.getBytes(); //

        //说明: 封装的 DatagramPacket对象 data 内容字节数组 , data.length , 主机(IP) , 端口
        DatagramPacket packet =
                new DatagramPacket(data, data.length, InetAddress.getByName("192.168.12.1"), 8888);

        socket.send(packet);

        //3.=== 接收从A端回复的信息
        //(1)   构建一个 DatagramPacket 对象,准备接收数据
        //   在前面讲解UDP 协议时,老师说过一个数据包最大 64k
        byte[] buf = new byte[1024];
        packet = new DatagramPacket(buf, buf.length);
        //(2)    调用 接收方法, 将通过网络传输的 DatagramPacket 对象
        //   填充到 packet对象
        //老师提示: 当有数据包发送到 本机的9998端口时,就会接收到数据
        //   如果没有数据包发送到 本机的9998端口, 就会阻塞等待.
        socket.receive(packet);

        //(3)  可以把packet 进行拆包,取出数据,并显示.
        int length = packet.getLength();//实际接收到的数据字节长度
        data = packet.getData();//接收到数据
        String s = new String(data, 0, length);
        System.out.println(s);

        //关闭资源
        socket.close();
        System.out.println("B端退出");
    }
}

接受端

public class Homework02ReceiverA {
    public static void main(String[] args) throws IOException {
        //1. 创建一个 DatagramSocket 对象,准备在8888接收数据
        DatagramSocket socket = new DatagramSocket(8888);
        //2. 构建一个 DatagramPacket 对象,准备接收数据
        //   在前面讲解UDP 协议时,老师说过一个数据包最大 64k
        byte[] buf = new byte[1024];
        DatagramPacket packet = new DatagramPacket(buf, buf.length);
        //3. 调用 接收方法, 将通过网络传输的 DatagramPacket 对象
        //   填充到 packet对象
        System.out.println("接收端 等待接收问题 ");
        socket.receive(packet);

        //4. 可以把packet 进行拆包,取出数据,并显示.
        int length = packet.getLength();//实际接收到的数据字节长度
        byte[] data = packet.getData();//接收到数据
        String s = new String(data, 0, length);
        //判断接收到的信息是什么
        String answer = "";
        if("四大名著是哪些".equals(s)) {
            answer = "四大名著 <<红楼梦>> <<三国演示>> <<西游记>> <<水浒传>>";
        } else {
            answer = "what?";
        }


        //===回复信息给B端
        //将需要发送的数据,封装到 DatagramPacket对象
        data = answer.getBytes();
        //说明: 封装的 DatagramPacket对象 data 内容字节数组 , data.length , 主机(IP) , 端口
        packet =
                new DatagramPacket(data, data.length, InetAddress.getByName("192.168.12.1"), 9998);

        socket.send(packet);//发送

        //5. 关闭资源
        socket.close();
        System.out.println("A端退出...");

    }
}

3.

服务端

public class Homework03Server {
    public static void main(String[] args) throws Exception {
        ServerSocket serverSocket = new ServerSocket(9999);
        Socket socket = serverSocket.accept();
        InputStream inputStream = socket.getInputStream();
        byte[] b = new byte[1024];
        int len = 0;
        String downloadFileName = "";
        while ((len = inputStream.read(b))!= -1){
            downloadFileName += new String(b,0,len);
        }
        System.out.println(downloadFileName);
        //服务器上有两个文件 无名.mp3 高山流水.mp3
        //如果下载的是高山流水返回该文件,否则一律返回无名
        String resFileName = "";
        if("高山流水".equals(downloadFileName)){
            resFileName = "src\\高山流水.mp3";
        }else{
            resFileName = "src\\无名.mp3";
        }
        //创建输入流读取文件
        BufferedInputStream bufferedInputStream = new BufferedInputStream(new FileInputStream(resFileName));
        //使用工具类
        byte[] bytes = StreamUtils.streamToByteArray(bufferedInputStream);
        //得到socket相关的输出流
        BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(socket.getOutputStream());
        bufferedOutputStream.write(bytes);
        bufferedOutputStream.flush();
        socket.shutdownOutput();

        //关闭相关的资源
        inputStream.close();
        bufferedInputStream.close();
        bufferedOutputStream.close();
        socket.close();
        serverSocket.close();
        System.out.println("关闭服务端");
    }
}

客户端

public class Homework03Client {
    public static void main(String[] args) throws Exception {
        //接受用户输入 准备从服务端下载
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入要下载的名字");
        String downloadFileName = scanner.next();
        Socket socket = new Socket(InetAddress.getLocalHost(),9999);
        OutputStream outputStream = socket.getOutputStream();
        BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(outputStream);
        bufferedOutputStream.write(downloadFileName.getBytes());
        bufferedOutputStream.flush();
        socket.shutdownOutput();

        InputStream inputStream = socket.getInputStream();
        BufferedInputStream bufferedInputStream = new BufferedInputStream(inputStream);
        byte[] bytes1 = StreamUtils.streamToByteArray(bufferedInputStream);

        String fileName = "";
        if(downloadFileName.equals("高山流水")){
            fileName = "高山流水";
        }else {
            fileName ="无名";
        }
        String filePath = "src\\com\\yinhai\\homework\\downloadFile\\" + fileName + ".mp3" ;
        BufferedOutputStream bufferedOutputStream1 = new BufferedOutputStream(new FileOutputStream(filePath));
        bufferedOutputStream1.write(bytes1);
        bufferedOutputStream1.flush();

        bufferedInputStream.close();
        bufferedOutputStream.close();
        bufferedOutputStream1.close();
        socket.close();
    }
}

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

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

相关文章

Ansible命令使用

ansible ansible的命令 ansible命令模块Pingcommand 模块shell 模块copy 模块file 模块fetch 模块cron 模块yum 模块service 模块user 模块group 模块script 模块setup 模块get_url模块stat模块unarchive模块unarchive模块 ansible的命令 /usr/bin/ansible  Ansibe AD-Hoc 临…

Xilinx DDR3 MIG系列——Xiinx DDR3官方手册ds176_7series_MIS

本节目录 一、官方手册ds176_7series_MIS 1、DDR3功能支持 2、MIG官方手册资源 3、Vivado DDR3 MIG IP资源表的导出与查看本节内容 Xilinx官方提供了手册&#xff0c;以便硬件开发者设计DDR3的硬件电路&#xff0c;和FPGA开发者使用MIG官方ip核完成项目的逻辑开发。 针对Xilin…

类和对象(2):构造函数,析构函数

一、构造函数 1.1 概念 构造函数是一种特殊的成员函数&#xff0c;名字与类名相同&#xff0c;创建类类型对象时编译器自动调用——初始化对象&#xff0c;在对象整个生命周期内只调用一次。 PS: 1. 构造函数无返回值&#xff1b;2. 构造函数支持重载。 class Date { public:…

【沐风老师】3DMAX克隆修改器插件教程

3DMAX克隆修改器插件&#xff0c;它通过增量平移、旋转和缩放输入几何体来创建对象的副本。在某些方面&#xff0c;它类似于 3ds Max 的内置阵列工具&#xff0c;但有一个主要优点 -克隆是完全参数化的&#xff0c;因此您可以随时更改重复项的数量及其分布。其他功能包括随机变…

Yum配置、相关命令和常见问题

搭建光盘源 将系统盘读取出来&#xff0c;找到系统盘下存放软件包的目录 2.配置yun仓库 输入命令进入仓库编辑 #必须以.repo结尾 :wq 回车保存退出 3.命令行输入yum repolist 查看yum仓库 配置硬盘源 1.将硬盘源拷贝到目录&#xff0c;或者挂载到目录 2.指定repo文件baseu…

Vue3-组合式API生命周期函数

一进入页面的请求一律放在setup中执行 如果有些代码需要在mounted生命周期中执行&#xff0c;并且写成函数的调用方式可以调用多次&#xff0c;并不会冲突&#xff0c;而是按照顺序依次执行 <script setup>onMounted(()>{console.log("mounted生命周期函数-逻辑…

SQL必知会(二)-SQL查询篇(7)-使用函数处理数据

第8课、使用函数处理数据 表8-1 DBMS 函数的差异 函数语法提取字符串的组成DB2、Oracle、PostgreSQL 和 SQLite 使用 SUBSTR()&#xff1b;MariaDB、Mysql 和 SQL Server 使用 SUBSTRING()数据类型转换Oracle 使用多个函数&#xff0c;每种类型的转换有一个函数&#xff1b;D…

指针传 1

1. 内存 在计算机中内存划分为⼀个个的内存单元&#xff0c;每个内存单元的⼤⼩取1个字节。每个内存单元放了八个bite位&#xff0c;就像我们在高中时住的八人间&#xff0c;那么每个人就代表了一个bite位。 每个内存单元也都有⼀个编号&#xff08;这个编号就相当 于我们所住…

聊天机器人框架Rasa资源整理

Rasa是一个主流的构建对话机器人的开源框架&#xff0c;它的优点是几乎覆盖了对话系统的所有功能&#xff0c;并且每个模块都有很好的可扩展性。参考文献收集了一些Rasa相关的开源项目和优质文章。 一.Rasa介绍 1.Rasa本地安装 直接Rasa本地安装一个不好的地方就是容易把本地…

Django框架FAQ

文章目录 问题1:Django数据库恢复问题2:null和blank的区别问题3:Django创建超级用户报错问题4:Django同源策略 问题1:Django数据库恢复 问题: 从仓库拉下来的Django项目,没有sqlite数据库和migrations记录,如何通过model恢复数据库 解决方法: # 步骤1:导出数据 # 不指定 ap…

如何配置《动手学强化学习》的环境

如何配置《动手学强化学习》的环境 网站&#xff1a;https://hrl.boyuai.com/chapter/intro github仓库&#xff1a;https://github.com/boyu-ai/Hands-on-RL/tree/main 可以看到该教程要求使用gym0.18.3版本的gym库&#xff0c;本教程可以用于解决绝大多数需要使用Pendulum-…

阿里云从公网IP转为弹性公网IP,同时绑定多个IP教程

先将云服务器ECS 转为弹性IP 购买新的弹性辅助网卡 购买弹性公网iP 购买之后选择绑定资源选择第二步购买的网卡 进入ECS 终端 ,输入 ip address可以查看到eth1 的对应mac 地址 终端输入 vi /etc/sysconfig/network-scripts/ifcfg-eth1保存一下信息 DEVICEeth1 #表示新配置…

【MySQL基本功系列】第二篇 InnoDB事务提交过程深度解析

通过上一篇博文&#xff0c;我们简要了解了MySQL的运行逻辑&#xff0c;从用户请求到最终将数据写入磁盘的整个过程。 当数据写入磁盘时&#xff0c;存储引擎扮演着关键的角色&#xff0c;它负责实际的数据存储和检索。 在MySQL中&#xff0c;有多个存储引擎可供选择&#xf…

免费博客搭建笔记

title: 免费博客搭建笔记 tags: 博客搭建 本次是对自己在网上学习github搭建一个 &#x1f447;个人免费静态网站的总结当然不是很完美&#x1f447; Bow to the new king iYANG (yangsongl1n.github.io) 接着我会从我的写笔记的个人习惯来逐步介绍如何搭建这个网站 1.写笔…

【解决】conda-script.py: error: argument COMMAND: invalid choice: ‘activate‘

运行conda activate base报错&#xff1a; 试了网上找到的解决方法都不行&#xff1a; 最后切换了一下terminal&#xff1a; 从powershell改回cmd&#xff08;不知道为什么一开始手贱换成powershell&#xff09; 就可以了

XML解析文档解析

1.首先是我的项目结构以及我所引入的依赖&#xff1a; 2.引入的依赖&#xff1a;jdk用的是17 <properties><maven.compiler.source>17</maven.compiler.source><maven.compiler.target>17</maven.compiler.target> </properties> <dep…

黑客技术(网络安全)-自学

前言 一、什么是网络安全 网络安全可以基于攻击和防御视角来分类&#xff0c;我们经常听到的 “红队”、“渗透测试” 等就是研究攻击技术&#xff0c;而“蓝队”、“安全运营”、“安全运维”则研究防御技术。 无论网络、Web、移动、桌面、云等哪个领域&#xff0c;都有攻与防…

Peter算法小课堂—八皇后问题

独立集问题&#xff1a;安排互不冲突的个体 四个斜眼枪手 bool valid(int x,int y){for(int i1;i<min(x,y);i)if(f[x-i][y-i]) return 0;for(int i1;i<min(x,N-1-y);i)if(f[x-i][yi]) return 0;return 1; } void dfs(int x,int y,int c){if(cGUNS){ans;print();return;}i…

不可否认程序员的护城河已经越来越浅了

文章目录 那些在冲击程序员护城河低代码/无代码开发平台自动化测试和部署工具AI辅助开发工具在线学习和教育平台 面临冲击&#xff0c;程序员应该怎么做深入专业知识&#xff1a;不断学习全栈技能开发解决问题的能力建立人际网络管理和领导技能 推荐阅读 技术和应用的不断发展对…