一、I/O模型之BIO

I/O模型之BIO

基本介绍

  • Java BIO 就是传统的 Java IO 编程,其相关的类和接口再 java.io 包下
  • BIO(blocking I/O):同步阻塞,服务器实现模式为一个连接一个线程,即客户端有连接请求时服务器端就需要启动一个线程进行处理,但是如果新启动的这一个线程不做任何事情就会造成不必要的开销,可以通过线程池机制改善
  • BIO 模型适用于连接数目比较小且固定的架构,这种方式对服务器资源要求比较高,并发性能差,JDK1.4以前使用

BIO编程的流程

  1. 服务器端启动一个SeverSocket
  2. 客户端启动Socket对服务器端进行通信,默认情况下服务器端需要对每个客户建立一个线程与之通讯
  3. 客户端发出请求后,先咨询服务器是否有线程响应,如果没有则会等待或者被拒绝
  4. 如果服务器有响应,客户端当前线程会等待请求结束后才继续执行

应用实例

实例说明

  1. 使用BIO模型编写一个服务器端,监听6666端口,当有客户连接时,就启动一个线程与之通讯
  2. 要求使用线程池机制改善,可以连接多个客户端
  3. 服务器端可以接收客户端发送的数据(使用telnet)

实例代码:

public class BIOServer {
    public static void main(String[] args) throws Exception {
        //1、创建一个线程池
        ExecutorService threadPool = Executors.newCachedThreadPool();

        ServerSocket serverSocket = new ServerSocket(6666);

        System.out.println("服务器启动了!!!");
        while (true){
            //监听,等待客户端连接
            final Socket socket = serverSocket.accept();
            System.out.println("连接到一个客户端");
            //2、如果有客户端连接了,就创建一个线程,与之通讯(单独写一个方法)
            threadPool.execute(new Runnable() {
                public void run() {
                    //与客户端进行通讯
                    handler(socket);
                }
            });
        }

    }

    //编写一个与客户端通讯的handler方法
    public static void handler(Socket socket){
        //用于接收数据
        byte[] bytes = new byte[1024];
        //通过socket获取输入流
        InputStream inputStream = null;
        try {
            System.out.println("线程id="+Thread.currentThread().getId()+"名字="+Thread.currentThread().getName());
            inputStream = socket.getInputStream();

            //循环读取客户端发送的数据
            while(true){
                int read = inputStream.read(bytes);
                if (read!=-1){
                    //输出客户端发送的数据
                    System.out.println(“收到信息:”+new String(bytes,0,read));
                }else{
                    break;
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            System.out.println("关闭与客户端的连接......");
            try {
                inputStream.close();
                socket.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

测试结果:

服务器启动了!!!
连接到一个客户端
线程id=12名字=pool-1-thread-1
收到消息:nihao
posted @ 2019-11-28 22:34  ねぇ  阅读(313)  评论(0编辑  收藏  举报