Socket通信与WebSocket协议

news2025/1/15 20:36:48

文章目录

目录

文章目录

前言

一、Socket通信

1.1 BIO

1.2 NIO

1.3 AIO

二、WebSocket协议

总结



前言


 

一、Socket通信

Socket是一种用于网络通信的编程接口(API),它提供了一种机制,使不同主机之间可以通过网络进行数据传输和通信。Socket是支持TCP/IP协议栈的应用层与传输层之间的接口。

在Socket编程中,有两种常见的角色:客户端和服务器端。客户端负责发起连接请求,服务器端负责接收并处理连接请求。

Socket通信的基本流程如下:

  1. 服务器端创建一个ServerSocket对象,并指定一个端口号。该对象会监听该端口上的连接请求。
  2. 客户端创建一个Socket对象,并指定要连接的服务器的IP地址和端口号。
  3. 客户端使用Socket对象发起连接请求,向服务器发送连接请求。
  4. 服务器监听到连接请求后,通过accept()方法接受客户端的连接,创建一个Socket对象与客户端建立连接。
  5. 客户端和服务器端通过各自的Socket对象进行数据的读取和写入,实现双向的数据交换。
  6. 数据交换完成后,可以关闭连接。客户端和服务器端都可以使用close()方法关闭自己的Socket对象。

通过Socket编程,可以实现不同设备之间的网络通信。例如,可以使用Socket编程来开发基于TCP/IP的客户端-服务器应用、聊天程序、文件传输程序等。

需要注意的是,Socket编程只提供了底层的网络通信接口,对于数据的格式、协议、解析等需要自行定义和处理。在Java中,可以使用Java标准库中的java.net.Socketjava.net.ServerSocket来实现Socket编程。

socket编程步骤:

  1. 服务器监听:服务器启动后,它会有一个线程一直启动,等待着客户端端连接。它会定义好自己的端口号。
  2. 客户端请求:客户端端套接字提出连接请求,要连接的目标是服务端的套接字。客户端必须要指明服务端套接字的地址和端口号。
  3. 连接确认:当服务端收到客户端的连接请求就会响应客户端套接字的请求,建立一个新的线程处理客户端的请求。

1.1 BIO

BIO(Blocking I/O)是Java中的一种阻塞式I/O模型,也称为传统的I/O模型。在BIO中,每个I/O操作都会阻塞当前线程,直到数据准备好或者操作完成。

BIO的工作原理如下:

  1. 服务器端创建一个ServerSocket对象并监听指定的端口。
  2. 服务器通过accept()方法等待客户端发起连接请求。
  3. 当有客户端连接请求到达时,服务器通过accept()方法接受客户端的连接,并返回一个新的Socket对象。
  4. 服务器使用新的Socket对象与客户端进行数据的读取和写入。
  5. 客户端使用Socket对象与服务器进行通信,发送请求并接收响应。
  6. 服务器端和客户端通过读写操作进行数据的交互,但是这些操作都是阻塞的,直到数据完全发送或接收完毕。

BIO的特点:

  1. 阻塞:BIO的I/O操作是阻塞的,当没有数据可读或写时,线程会一直阻塞在相应的读写操作上,无法去处理其他任务。
  2. 线程池限制:由于每个连接都需要独占一个线程进行处理,当并发连接数很大时,线程资源会被耗尽,导致性能下降。
  3. 可靠性:由于阻塞的特性,BIO在网络不稳定或出现异常时可能会导致程序挂起或阻塞。

虽然BIO具有易于理解和使用的优点,但其在高并发应用场景下的性能较差。随着网络应用的发展,为了提高性能和扩展性,非阻塞I/O模型如NIO(New I/O)和异步I/O模型如AIO(Asynchronous I/O)逐渐成为主流。

需要注意的是,BIO仍然适用于某些特定的应用场景,特别是在连接数较少且对实时性要求不高的情况下。

服务器端

package com.rcg.testtwo;

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

/**
 * @author :1452863884@qq.com rcg
 * @date :Created in 2023/8/24 14:47
 * @description:服务端
 * @modified By:
 * @version:
 */
public class BioServer {
    public static void main(String[] args) {
        //定义端口号
        int port = 9999;
        //定义服务器套接字
        ServerSocket serverSocket = null;
        try {
            //创建服务器套接字
            serverSocket = new ServerSocket(port);
            //一直监听,是否有客户端请求过来
            while (true) {
                //每次都会新建一个线程,来处理接收到到请求
                Socket socket = serverSocket.accept();
                //每次都会新建一个线程,来处理接收到到请求
                new Thread(new SocketHandler(socket)).start();
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            //如果服务器套接字不为空,则关闭
            if (serverSocket!= null) {
                try {
                    serverSocket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
                serverSocket = null;
            }
        }
    }
    static class SocketHandler implements Runnable {
        //定义socket
        Socket socket = null;
     public SocketHandler(Socket socket) {
         this.socket = socket;
     }
     //处理读取的数据
     @Override
        public void run() {
            BufferedReader reader = null;
            PrintWriter writer = null;
            try {
                //读取数据,BIO 是面向流到,所以定义流 BufferedReader 来读取数据
                reader = new BufferedReader(new InputStreamReader(socket.getInputStream(), "UTF-8"));
                //将收到的数据返回给客户端
                writer = new PrintWriter(new OutputStreamWriter(socket.getOutputStream(), "UTF-8"));
                String readMessage = null;
                //循环读取数据
                while (true) {
                    if ((readMessage = reader.readLine()) == null) {
                        break;
                    }
                    System.out.println("server reading........" + readMessage);
                    //将数据返回给客户端
                    writer.println("server recive : " + readMessage);
                    writer.flush();
                }
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
                if (socket!= null) {
                    try {
                        socket.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    socket = null;
                }
                if (reader!= null) {
                    try {
                        reader.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    reader = null;
                }
                if (writer!= null) {
                    writer.close();
                    writer = null;
                }
            }
        }
    }
}

客户端

package com.rcg;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
import java.util.Scanner;

/**
 * @author :1452863884@qq.com rcg
 * @date :Created in 2023/8/24 14:54
 * @description:客户端
 * @modified By:
 * @version:
 */
public class BioClient {
    public static void main(String[] args) {
        //服务端到 ip 地址
        String host = "127.0.0.1";
        //和服务端到端口号一致
        int port = 9999;
        Socket socket = null;
        BufferedReader reader = null;
        PrintWriter writer = null;
        //接收键盘输入数据
        Scanner scanner = new Scanner(System.in);
        try {
            //创建 socket 对象
            socket = new Socket(host, port);
            //创建 BufferedReader 对象
            String message = null;
            reader = new BufferedReader(new InputStreamReader(socket.getInputStream(), "UTF-8"));
            writer = new PrintWriter(socket.getOutputStream(), true);
            //循环接收数据
            while (true) {
                message = scanner.nextLine();
                //如果接收到 exit 则退出
                if (message.equals("exit")) {
                    break;
                }
                //数据发送服务端
                writer.println("客户端输入:" + message);
                writer.flush();
                //接收服务端的响应
                System.out.println(reader.readLine());
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //关闭 socket 对象
            if (socket!= null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
                socket = null;
            }
            //关闭 BufferedReader 对象
            if (reader!= null) {
                try {
                    reader.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
                reader = null;
            }
            //关闭 PrintWriter 对象
            if (writer!= null) {
                writer.close();
                writer = null;
            }
        }
    }
}

效果 

 

 

 

 

 

1.2 NIO

NIO(New I/O)是Java中提供的一种非阻塞式I/O模型,与传统的阻塞式I/O(BIO)相比,NIO能更高效地处理I/O操作和并发连接。

NIO的关键组件包括通道(Channel),缓冲区(Buffer)、选择器(Selector)和非阻塞模式。下面对每个组件进行简要介绍:

  1. 通道(Channel):通道是数据源和目标之间的连接,可以用于读取和写入数据。在NIO中,所有I/O操作都是通过通道进行的。不同类型的通道,如文件通道、套接字通道等,适用于不同的I/O场景。

  2. 缓冲区(Buffer):缓冲区是一个连续的内存块,用于存储数据。它使得读取和写入数据更加高效。在NIO中,所有数据的读取和写入都是通过缓冲区进行的。

  3. 选择器(Selector):选择器是用于检测通道上的事件的对象。通过选择器,可以实现单个线程管理多个通道,从而高效地处理并发连接。选择器可以监控通道上的事件类型,如接受连接、读取数据、写入数据等,并根据事件的发生情况来执行相应的操作。

  4. 非阻塞模式:NIO使用非阻塞模式进行通信。在非阻塞模式下,当一个通道没有数据可读取时,线程不会被阻塞,而是可以继续处理其他任务。这样可以避免每个连接都需要独占一个线程的资源浪费问题,实现更高效的并发连接处理。

NIO的工作原理如下:

  1. 服务器创建一个选择器,并将其注册到一个或多个通道上。
  2. 当有事件发生(如连接、读取、写入等),选择器会通过轮询的方式检测到事件的发生。
  3. 当事件发生时,选择器会返回一个包含已就绪事件的键集合,程序可以通过这些键来获取感兴趣的事件和相应的通道。
  4. 程序根据事件类型执行相应的操作,如接受连接、读取数据、写入数据等。

NIO相比于BIO具有以下优点:

  1. 高并发性:NIO使用选择器和非阻塞模式,能够高效地处理并发连接,减少线程的使用,提高系统的并发性能。
  2. 可扩展性:NIO支持单线程管理多个通道,适用于需要管理大量连接的场景,提供了更好的可扩展性。
  3. 非阻塞式:NIO采用非阻塞模式,不会因为一个通道的读写操作导致阻塞,可以同时处理多个通道的操作,提高了系统的响应速度。

需要注意的是,NIO的实现相对复杂,需要合理地使用和管理缓冲区、处理事件等。在Java中,可以使用java.nio包下的类来进行NIO编程,如SelectableChannelByteBufferSelector等。

总的来说,NIO是一种更高效和灵活的I/O模型,适用于需要处理大量并发连接的场景,如网络服务器、聊天程序、游戏服务器等。

服务器端

package com.rcg.testtwo;

/**
 * @author :1452863884@qq.com rcg
 * @date :Created in 2023/8/24 15:49
 * @description:
 * @modified By:
 * @version:
 */

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Iterator;

public class NioServer {
    private static ByteBuffer readBuf = ByteBuffer.allocate(1024);
    private static ByteBuffer writeBuf = ByteBuffer.allocate(1024);

    public static void main(String[] args) {
        int port = 9999;
        Selector selector;
        try {
            //打开多路复用器
            selector = Selector.open();
            //定义一个 Channel
            ServerSocketChannel channel = ServerSocketChannel.open();
            //非阻塞模型
            channel.configureBlocking(false);
            //绑定端口号
            channel.bind(new InetSocketAddress(port));
            //channel 注册到 Selector 上面
            channel.register(selector, SelectionKey.OP_ACCEPT);
            System.out.println("服务已经启动了......");
            while (true) {
                //复用器开始监听
                selector.select();
                Iterator<SelectionKey> seletionKeys = selector.selectedKeys().iterator();
                while (seletionKeys.hasNext()) {
                    SelectionKey key = seletionKeys.next();
                    if (key.isValid()) {
                        //监听客户端第一次端连接信息
                        if (key.isAcceptable()) {
                            ServerSocketChannel ssc = (ServerSocketChannel) key.channel();
                            SocketChannel sc = ssc.accept();
                            sc.configureBlocking(false);
                            sc.register(selector, SelectionKey.OP_READ);
                            System.out.println("accept a client : " + sc.socket().getInetAddress().getHostName());
                        } else {
                            read(key);
                        }
                    }
                    seletionKeys.remove();
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    private static void read(SelectionKey key) {
        try {
            //清空缓冲区
            readBuf.clear();
            //获取注册在通道里面的 key 对象
            SocketChannel sc = (SocketChannel) key.channel();
            //读取数据
            int count = sc.read(readBuf);
            //如果没有数据,则关闭连接
            if (count == -1) {
                key.channel().close();
                key.cancel();
                return;
            }
            //有数据则进行读取 读取之前需要进行复位方法(把position 和limit进行复位)
            readBuf.flip();
            //6 根据缓冲区的数据长度创建相应大小的byte数组,接收缓冲区的数据
            byte[] bytes = new byte[readBuf.remaining()];
            //7 接收缓冲区数据
            readBuf.get(bytes);
            String body = new String(bytes).trim();
            System.out.println("服务端接受到客户端请求的数据: " + body);
            //9 告诉客户端已收到数据
            writeBuf.put(("你好,客户端,我已收到数据:" + body).getBytes());
            //对缓冲区进行复位
            writeBuf.flip();
            //写出数据到服务端
            sc.write(writeBuf);
            //清空缓冲区数据
            writeBuf.clear();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

客户端

package com.rcg;

/**
 * @author :1452863884@qq.com rcg
 * @date :Created in 2023/8/24 15:53
 * @description:
 * @modified By:
 * @version:
 */
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;
import java.util.Iterator;
public class NioClient {
    private static final String host = "127.0.0.1";
    private static final Integer port = 9999;
    public static void main(String[] args) {
        try {
            //创建一个SocketChannel对象
            SocketChannel channel = SocketChannel.open();
            //连接服务端
            channel.connect(new InetSocketAddress(host, port));
            //设置为非阻塞模式
            channel.configureBlocking(false);
            //创建一个Selector对象
            Selector selector = Selector.open();
            //注册channel,可读,可写
            channel.register(selector, SelectionKey.OP_READ, SelectionKey.OP_WRITE);
            //开启新的线程监听,否则的话,客户端在控制台输入信息不好操作。
            new Thread(() -> {
                while (true) {
                    try {
                        //多路复用器开始监听
                        selector.select();
                        Iterator<SelectionKey> selectionKeys = selector.selectedKeys().iterator();
                        while (selectionKeys.hasNext()) {
                            SelectionKey selectionKey = selectionKeys.next();
                            if (selectionKey.isValid() && selectionKey.isReadable()) {
                                //建立写缓冲区
                                ByteBuffer readBuf = ByteBuffer.allocate(1024);
                                //2 获取之前注册的socket通道对象
                                SocketChannel sc = (SocketChannel) selectionKey.channel();
                                //3 读取数据
                                int count = sc.read(readBuf);
                                if (count == -1) {
                                    selectionKey.channel().close();
                                    selectionKey.cancel();
                                    return;
                                }
                                //5 有数据则进行读取 读取之前需要进行复位方法(把position 和limit进行复位)
                                readBuf.flip();
                                //6 根据缓冲区的数据长度创建相应大小的byte数组,接收缓冲区的数据
                                byte[] bytes = new byte[readBuf.remaining()];
                                //7 接收缓冲区数据
                                readBuf.get(bytes);
                                String body = new String(bytes).trim();
                                System.out.println("收到服务端的数据:" + body);
                            }
                            //移除未处理的key
                            selectionKeys.remove();
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }).start();
            //设置缓冲区大小
            ByteBuffer writebuf = ByteBuffer.allocate(1024);
            while (true) {
                byte[] bytes = new byte[1024];
                System.in.read(bytes);
                //把数据放到缓冲区中
                writebuf.put(bytes);
                //对缓冲区进行复位
                writebuf.flip();
                //写出数据到服务端
                channel.write(writebuf);
                //清空缓冲区数据
                writebuf.clear();
            }
        } catch (
                IOException e) {
            e.printStackTrace();
        }
    }
}

 测试

 

1.3 AIO

AIO(Asynchronous I/O)是Java中提供的一种异步I/O模型,与传统的阻塞式I/O(BIO)和非阻塞式I/O(NIO)相比,AIO能更高效地处理I/O操作和并发连接。

AIO的关键组件包括通道(Channel)、缓冲区(Buffer)和完成通知机制。下面对每个组件进行简要介绍:

  1. 通道(Channel):通道是数据源和目标之间的连接,可以用于读取和写入数据。在AIO中,所有I/O操作都是通过通道进行的。不同类型的通道,如文件通道、套接字通道等,适用于不同的I/O场景。

  2. 缓冲区(Buffer):缓冲区是一个连续的内存块,用于存储数据。它使得读取和写入数据更加高效。在AIO中,所有数据的读取和写入都是通过缓冲区进行的。

  3. 完成通知机制:AIO使用回调和事件驱动的方式来处理I/O操作。当一个I/O操作完成时,操作系统会通知应用程序,并触发预先注册的回调函数,从而进行相应的处理。这种机制可以避免线程阻塞,提高系统的并发性能。

AIO的工作原理如下:

  1. 服务器创建一个通道,并注册一个或多个感兴趣的I/O事件和回调函数。
  2. 当有I/O事件发生时,操作系统会通知应用程序,并调用相应的回调函数。
  3. 在回调函数中,应用程序可以获取已完成的I/O操作的结果,并进行相应的处理,如读取数据、写入数据等。

AIO相比于BIO和NIO具有以下优点:

  1. 异步性:AIO通过使用回调和事件驱动的方式,实现真正的异步I/O操作。这意味着应用程序无需等待操作完成,而是可以继续执行其他任务,提高了系统的并发性能。

  2. 简化编程模型:AIO的异步特性可以简化编程模型,避免了繁琐的线程管理和同步操作。开发者只需要关注回调函数的处理即可,让操作系统来处理底层的I/O操作。

  3. 高性能:由于AIO的异步特性,可以充分利用系统资源,提供更高的并发性能和吞吐量。

需要注意的是,AIO在Java中是通过AsynchronousChannelCompletionHandler来实现的。可以使用java.nio.channels.AsynchronousChannelGroupjava.nio.channels.AsynchronousServerSocketChannel等类来创建和管理异步通道,使用java.nio.channels.CompletionHandler来定义回调函数。

总的来说,AIO适用于需要处理大量并发连接且对性能要求较高的场景。然而,AIO在某些平台上的性能可能不如NIO,具体取决于操作系统和硬件的支持程度。在选择使用AIO还是NIO时,需要考虑特定的应用需求和目标平台的特性。

服务端

package com.rcg.testtwo;

/**
 * @author :1452863884@qq.com rcg
 * @date :Created in 2023/8/24 16:02
 * @description:
 * @modified By:
 * @version:
 */
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousChannelGroup;
import java.nio.channels.AsynchronousServerSocketChannel;
import java.nio.channels.AsynchronousSocketChannel;
import java.nio.channels.CompletionHandler;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class AioServer {
    private final int port;
    public static void main(String[] args) {
        // 创建一个端口号为9999的AioServer对象
        int port = 9999;
        // 创建一个AioServer对象,并传入端口号
        new AioServer(port);
    }
    public AioServer(int port) {
        // 将端口号传入AioServer对象
        this.port = port;
        // 监听
        listen();
        // 循环
        while (true) {
            try {
                // 线程休眠1000000毫秒
                Thread.sleep(1000000);
            } catch (InterruptedException e) {
            }
        }
    }
    private void listen() {
        try {
            ExecutorService executorService = Executors.newCachedThreadPool();
            AsynchronousChannelGroup threadGroup = AsynchronousChannelGroup.withCachedThreadPool(executorService, 1);
            //创建一个异步的服务端套接字
            final AsynchronousServerSocketChannel serverSocketChannel = AsynchronousServerSocketChannel.open(threadGroup);
            //绑定端口
            serverSocketChannel.bind(new InetSocketAddress(port));
            System.out.println("服务已经启动,监听端口:" + port);
            //监听
            serverSocketChannel.accept(null, new CompletionHandler<AsynchronousSocketChannel, Object>() {
                final ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
                @Override
                public void completed(AsynchronousSocketChannel result, Object attachment) {
                    System.out.println("I/O 操作成功,开始获取数据");
                    try {
                        //清空缓冲区
                        byteBuffer.clear();
                        //从异步缓冲区中读取数据
                        result.read(byteBuffer).get();
                        //将缓冲区的数据转换为字符串
                        byteBuffer.flip();
                        System.out.println("服务端接收到数据:" + new String(byteBuffer.array()).trim());
                        //将数据写入异步缓冲区
                        result.write(byteBuffer);
                        //将缓冲区的数据转换为字符串
                        byteBuffer.flip();
                    } catch (Exception e) {
                        e.printStackTrace();
                    } finally {
                        try {
                            //关闭异步套接字
                            result.close();
                            //接收下一个操作
                            serverSocketChannel.accept(null, this);
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }
                    System.out.println("操作完成");
                }
                @Override
                public void failed(Throwable exc, Object attachment) {
                    System.out.println("I/O 操作失败:" + exc);
                }
            });
            Thread.sleep(1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

客户端

package com.rcg;

/**
 * @author :1452863884@qq.com rcg
 * @date :Created in 2023/8/24 16:02
 * @description:
 * @modified By:
 * @version:
 */
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousSocketChannel;
import java.nio.channels.CompletionHandler;
public class AioClient {
    private AsynchronousSocketChannel clientChannel;
    // 定义一个AsynchronousSocketChannel类型的变量clientChannel
    private static final String host = "127.0.0.1";
    // 定义一个字符串类型的变量host
    private static final Integer port = 9999;
    // 定义一个整型类型的变量port
    public AioClient() throws Exception {
        // 创建一个AsynchronousSocketChannel实例
        clientChannel = AsynchronousSocketChannel.open();
    }
    // 定义一个AioClient类的构造函数
    public void connect(String host, int port) {
        try {
            // 调用AsynchronousSocketChannel的open()方法,创建一个AsynchronousSocketChannel实例
            clientChannel.connect(new InetSocketAddress(host, port), null, new CompletionHandler<Void, Void>() {
                // 定义一个CompletionHandler类型的变量attachment,用于存放连接结果
                @Override
                public void completed(Void result, Void attachment) {
                    try {
                        // 调用AsynchronousSocketChannel的write()方法,发送数据
                        clientChannel.write(ByteBuffer.wrap("你好师姐,客户端链接成功了".getBytes())).get();
                        System.out.println("数据已经发送成功!");
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
                // 定义一个CompletionHandler类型的变量attachment,用于存放连接失败的异常
                @Override
                public void failed(Throwable exc, Void attachment) {
                }
            });
        } catch (Exception e) {
            e.printStackTrace();
        }
        try {
            // 等待1秒
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        // 定义一个ByteBuffer准备读取数据
        final ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
        // 调用AsynchronousSocketChannel的read()方法,读取数据
        clientChannel.read(byteBuffer, null, new CompletionHandler<Integer, Object>() {
            // 定义一个CompletionHandler类型的变量attachment,用于存放读取结果
            @Override
            public void completed(Integer result, Object attachment) {
                System.out.println("I/O操作完成" + result);
                System.out.println("获取返回结果:" + new String(byteBuffer.array()).trim());
            }
            // 定义一个CompletionHandler类型的变量attachment,用于存放读取失败的异常
            @Override
            public void failed(Throwable exc, Object attachment) {
                exc.printStackTrace();
            }
        });
    }
    // 定义一个main()方法,用于连接服务器
    public static void main(String[] args) throws Exception {
        // 创建一个AioClient实例
        AioClient aioClient = new AioClient();
        // 调用AioClient的connect()方法,连接服务器
        aioClient.connect(host, port);
    }
}

 

测试

 

 

 

二、WebSocket协议

WebSocket是一种在Web浏览器和服务器之间进行全双工通信的协议。它提供了一种持久连接的机制,允许客户端和服务器之间实时地交换数据,而不需要频繁地发起HTTP请求。

与传统的HTTP请求相比,WebSocket连接通过一个初始的HTTP握手阶段建立,并使用一种特殊的数据帧格式来传输数据。一旦建立了WebSocket连接,客户端和服务器之间可以随时互相发送消息,这样就实现了实时的双向数据交流。

WebSocket具有以下特点:

  1. 实时性:WebSocket连接保持持久性,客户端和服务器之间可以实时地发送和接收数据,避免了短轮询或长轮询的延迟。
  2. 双向通信:WebSocket连接支持全双工通信,即客户端和服务器可以同时发送和接收数据。
  3. 轻量级:WebSocket协议采用了更轻量级的数据帧格式,相对于HTTP请求来说,数据传输的开销更小。
  4. 更少的资源消耗:由于WebSocket连接的持久性,服务器端不需要为每个客户端连接创建一个新的线程或进程,从而减少了服务器资源的消耗。
  5. 跨域支持:WebSocket连接支持跨域通信,可以在不同域名下的客户端和服务器之间进行通信。

WebSocket广泛用于实时聊天应用、多人游戏、实时数据传输等需要高实时性和双向通信的Web应用场景。在前端开发中,可以使用JavaScript提供的WebSocket API来创建和管理WebSocket连接。在后端开发中,可以使用各种语言和框架提供的WebSocket库来处理WebSocket连接和消息的收发。

要在Web应用程序中使用WebSocket,你需要在客户端和服务器端分别进行相应的代码编写。下面是一个示例,展示了如何在JavaScript和Java中使用WebSocket。

客户端

// 创建WebSocket对象并指定服务器的URL
var socket = new WebSocket('ws://localhost:8080/mywebsocket');

// 连接建立时触发事件
socket.onopen = function(event) {
  console.log('WebSocket连接已建立');
  
  // 向服务器发送消息
  socket.send('Hello Server!');
};

// 收到服务器消息时触发事件
socket.onmessage = function(event) {
  var message = event.data;
  console.log('收到服务器消息:' + message);
  
  // 在此处对收到的消息进行处理
};

// 连接关闭时触发事件
socket.onclose = function(event) {
  console.log('WebSocket连接已关闭');
};

// 发生错误时触发事件
socket.onerror = function(event) {
  console.error('WebSocket出现错误');
};

服务器端(使用Java):

import javax.websocket.*;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;

@ServerEndpoint("/mywebsocket")
public class MyWebSocket {

    @OnOpen
    public void onOpen(Session session) {
        System.out.println("WebSocket连接已建立");
        
        // 向客户端发送消息
        try {
            session.getBasicRemote().sendText("Hello Client!");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @OnMessage
    public void onMessage(String message, Session session) {
        System.out.println("收到客户端消息:" + message);
        
        // 在此处对收到的消息进行处理
        
        // 向客户端发送消息
        try {
            session.getBasicRemote().sendText("Got your message: " + message);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @OnClose
    public void onClose() {
        System.out.println("WebSocket连接已关闭");
    }

    @OnError
    public void onError(Throwable error) {
        System.err.println("WebSocket发生错误");
        error.printStackTrace();
    }
}

 在示例中,客户端使用JavaScript的WebSocket对象创建WebSocket连接,并通过相应的事件处理函数来处理连接建立、消息收发、连接关闭和错误等事件。服务器端使用Java的javax.websocket库,使用@ServerEndpoint注解指定WebSocket的URL,并编写相应的方法来处理连接建立、消息收发和连接关闭等事件。


总结

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

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

相关文章

算法通关村十三关 | 进制转换问题处理模板

1. 七进制数 题目&#xff1a;LeetCode504&#xff1a;504. 七进制数 - 力扣&#xff08;LeetCode&#xff09; 思路 进制转换&#xff0c;对几转换就是对几求余&#xff0c;最后将所有的余数反过来即可、如果num< 0&#xff0c;先取绝对值&#xff0c;再进行操作。 100转7…

Ceres Solver 入门

1. Ceres Solver 是什么 Ceres 可以解决以下形式的边界约束鲁棒化非线性最小二乘问题&#xff1a; 给定初始值&#xff0c;通过优化算法&#xff0c;得到最优解。 其中&#xff0c; f i f_i fi​是CostFunction&#xff0c;也叫误差函数&#xff0c;或者代价函数。 ρ i \rho…

第十七课:利用 Setup Factory 制作 Qt 软件安装包

功能描述&#xff1a;详细介绍如何利用 Setup Factory 制作 Qt 软件安装包&#xff0c;从 Setup Factory 软件下载、安装&#xff0c;到如何利用 Setup Factory 制作软件安装包&#xff0c;手把手教你将 Qt 应用程序制作成具有安装向导的安装包。 一、Setup Factory 简介 Setu…

C语言程序结构、基本语法与数据类型

文章目录 1. 程序结构1.1 Hello World示例1.2 编译并执行C程序 2. 基本语法2.1 C 标记2.2 分号2.3 注释2.4 标识符2.5 关键字2.6 C中的空格 3. 数据类型3.1 整数类型3.2 浮点类型3.3 void类型 1. 程序结构 1.1 Hello World示例 #include <stdio.h>int main() {/* my fi…

DirectExchange直连交换机

目录 一、简介 二、使用步骤 三、demo 父pom文件 pom文件 配置文件 config 消费者 生产者 测试 一、简介 直连型交换机&#xff0c;根据消息携带的路由键将消息投递给对应队列。 大致流程&#xff0c;有一个队列绑定到一个直连交换机上&#xff0c;同时赋予一个路由…

AMEYA360代理品牌:纳芯微芯片解决方案为光伏市场赋能

近年来&#xff0c;光伏市场进入了一个新的增长维度。SolarPower Europe数据显示&#xff0c;2022年全球光伏新增装机量达239GW&#xff0c;占所有可再生能源新增容量的三分之二。国家能源局也宣称&#xff0c;2022年我国工商业光伏新增装机达25.87GW&#xff0c;同比增长236.7…

淘宝商品数据采集(如何快速获取淘宝商品信息),淘宝API接口申请指南

淘宝作为国内的电商平台&#xff0c;拥有海量的商品信息。对于想要进行淘宝商品数据采集的人来说&#xff0c;如何快速获取淘宝商品信息是一个重要的问题。本文将介绍一些快速获取淘宝商品信息的方法。 1. 使用淘宝开放平台PI 淘宝开放平台提供了多种PI接口&#xff0c;可以通…

如何选择合适的开源许可证?

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

java八股文面试[java基础]——异常

自定义异常&#xff1a; 异常Exception 是指程序运行时&#xff0c; 由于输入错误、网络、程序逻辑等原因导致运行时出现的问题。出现异常时&#xff0c;程序会暂时中断执行&#xff0c;并根据产生异常的原因&#xff0c;创建对应异常类型的异常对象&#xff0c;并抛出给JVM捕…

高速收费站的智慧之选,工控机助力顺畅通行!

2020年初取消高速公路省界收费站后&#xff0c;全国高速公路进入“一张网运行、一体化服务”的新阶段。随着ETC用户量快速增长、驾乘人员对收费站高效通行需求不断提升&#xff0c;收费数据在线化运营及精准化、智能化、人性化的收费服务将成为主流。如何提高收费系统集成度、降…

Day4:前端路由(进阶篇)

目标: 持续输出&#xff01;每日分享关于web前端常见知识、面试题、性能优化、新技术等方面的内容。 主要面向群体&#xff1a;前端开发工程师&#xff08;初、中、高级&#xff09;、应届、转行、培训等同学 Day4-今日话题 今天分享的是前端路由的进阶篇&#xff0c;将从路由的…

传感网应用开发1+X实训室建方案

一、概述 1.1建设背景 从院校实际教学情况与人才培养计划为出发点&#xff0c;贯彻传感网应用开发1X实训室职业技能等级标准&#xff0c;充分考虑传感网应用开发1X实训室从业人员的职业发展路径与成长路径&#xff0c;以职业素养、职业技能、知识水平为主要框架结构&#xff…

无涯教程-进程 - 镜像

现在&#xff0c;我们已经了解了如何获取进程及其父进程的基本信息&#xff0c;是时候来研究进程信息的细节了。 以下是进程镜像的图形表示。 进程镜像(Process Image)到底是什么? 进程镜像是执行程序时所需的可执行文件&#xff0c;该镜像通常包含以下部分- 代码段或文本片段…

三维模型数据加载速度不理想?这三种加载方式供你选择!

在四维轻云平台的使用过程中&#xff0c;有用户反映三维模型数据加载速度较慢。因此&#xff0c;平台推出了默认方式、质量优先、速度优先三种数据加载方式供用户选择。下面就来简单介绍一下这三种加载方式的特点&#xff0c;用户可根据需求选择合适的数据加载方式。 默认方式…

Django(2)-编写你的第一个 Django 应用

创建一个基本的投票应用程序。 它将由两部分组成&#xff1a; 一个让人们查看和投票的公共站点。 一个让你能添加、修改和删除投票的管理站点。 创建应用 $ python manage.py startapp polls每一个应用是一个python包&#xff0c;一个项目可以包含多个应用。 可以看到生成…

Telegraf 本地代码vscode调试

需要安装的软件&#xff1a; golang 1.20vscodevscode推荐的go插件 在RUN按钮中&#xff0c;创建Launch 自动生成launch.json文件&#xff0c;此处增加了&#xff1a;args参数。 {// Use IntelliSense to learn about possible attributes.// Hover to view descriptions…

JS逆向系列之商指针数据解密

文章目录 声明案例地址y解密算法分析ecryptByPrivateKey 解密算法分析写代码前的流程梳理参考代码往期逆向文章推荐声明 本文章中所有内容仅供学习交流,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关,若有侵权,请私信我立即删除! 案例地址 aHR0cDovL…

系统上线安全测评需要做哪些内容?

电力信息系统、航空航天、交通运输、银行金融、地图绘画、政府官网等系统再正式上线前需要做安全测试。避免造成数据泄露从而引起的各种严重问题。 那么系统上线前需要做哪些测试内容呢&#xff1f;下面由我给大家介绍 1、安全机制检测-应用安全 身份鉴别 登录控制模块 应提供…

字节一面:post为什么会发送两次请求?

前言 最近博主在字节面试中遇到这样一个面试题&#xff0c;这个问题也是前端面试的高频问题&#xff0c;因为在前端开发的日常开发中我们总是会与post请求打交道&#xff0c;一个小小的post请求也是牵扯到很多知识点的&#xff0c;博主在这给大家细细道来。 &#x1f680; 作者…

视频尺寸缩小,一键批量剪辑,轻松制作精简版

大家好&#xff01;在视频剪辑中&#xff0c;有时我们需要将大尺寸的视频缩小&#xff0c;以适应特定的需求和平台要求。为了帮助您轻松制作精简版视频&#xff0c;我们推出了一款全新的工具——视频尺寸缩小批量剪辑软件&#xff01;让您一键批量将视频尺寸缩小&#xff0c;轻…