Java IO, NIO, AIO和Netty

背景

最近在回顾一下Java IO相关的知识,顺带写一下入门级别的文章。感觉工作以后很少写文章,一直想写点高质量的文章导致最后一篇文章都很难写。所以不写原理,只写实践,随大流,有问题请留言。(后续有时间再补充原理性的东西,从硬件到操作系统到JVM到JDK)

实现案例

创建一个server,可以接受多个client端的连接,接收到信息后返回一个接收到的信息。

传统IO实现

传统的IO就是我们所说的BIO(block io),

server端源码如下

package tech.sohocoder.postman.io;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.net.InetSocketAddress;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadFactory;

public class Server {

    private ServerSocket serverSocket;


    private void start() throws IOException, ClassNotFoundException {
        InetSocketAddress inetSocketAddress = new InetSocketAddress("localhost", 9000);
        serverSocket = new ServerSocket();
        serverSocket.bind(inetSocketAddress);
        ExecutorService executorService = Executors.newCachedThreadPool(new CaughtExceptionsThreadFactory());
        while (true) {
            Socket socket = serverSocket.accept();
            System.out.println("accept socket: " + socket.getRemoteSocketAddress());
            executorService.submit(new SocketHandler(socket));
        }
    }

    private static class MyUncaughtExceptionHandler implements Thread.UncaughtExceptionHandler {
        @Override public void uncaughtException(Thread t, Throwable e) {
            e.printStackTrace();
        }
    }

    private class SocketHandler implements Runnable {

        private Socket socket;

        public SocketHandler(Socket socket) {
            this.socket = socket;
        }

        @Override
        public void run() {
            try {
                while (true) {
                    ObjectInputStream ois = new ObjectInputStream(socket.getInputStream());
                    String message = ois.readObject().toString();
                    System.out.println("Message Received: " + message);
                    ObjectOutputStream oos = new ObjectOutputStream(socket.getOutputStream());
                    //write object to Socket
                    oos.writeObject("Hi Client " + message);
                    if (message.equals("quit")) {
                        ois.close();
                        oos.close();
                        socket.close();
                        break;
                    }
                }
            }catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Server server = new Server();
        server.start();
    }
}

client端源码如下

package tech.sohocoder.postman.io;

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

public class Client {

    private Socket socket;

    public void start() throws IOException, ClassNotFoundException {
        socket = new Socket("localhost", 9000);
        if(socket.isConnected()) {
            System.out.println("socket is connected");
            BufferedReader in = new BufferedReader(new InputStreamReader(System.in));
            for (;;) {
                final String input = in.readLine();
                final String line = input != null ? input.trim() : null;
                if (null == line || line.isEmpty()) { // skip `enter` or `enter` with spaces.
                    continue;
                }
                ObjectOutputStream oos = new ObjectOutputStream(socket.getOutputStream());
                oos.writeObject(line);
                ObjectInputStream ois = new ObjectInputStream(socket.getInputStream());
                System.out.println("Message: " + ois.readObject());
                if(line.equals("quit")) {
                    oos.close();
                    ois.close();
                    socket.close();
                    break;
                }
            }
        }
        System.out.println("Bye");
    }

    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Client client = new Client();
        client.start();
    }
}

NIO的阻塞实现

NIO实际上就是面向缓存及通道的新型IO(由JSR 51定义,后面JSR 203进行了扩展,有兴趣阅读一下这两个JSR)可以支持阻塞和非阻塞方式。先实现一下阻塞方式

client

package tech.sohocoder.nio.block;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;

import static java.lang.System.out;

public class Client {

    private SocketChannel socketChannel;

    public void start() throws IOException {
        socketChannel = SocketChannel.open();
        SocketAddress socketAddress = new InetSocketAddress("localhost", 9000);
        socketChannel.connect(socketAddress);

        BufferedReader in = new BufferedReader(new InputStreamReader(System.in));
        for (;;) {
            final String input = in.readLine();
            final String line = input != null ? input.trim() : null;
            if (null == line || line.isEmpty()) { // skip `enter` or `enter` with spaces.
                continue;
            }

            ByteBuffer byteBuffer = ByteBuffer.wrap(line.getBytes());
            socketChannel.write(byteBuffer);

            if(line.equals("quit")) {
                out.println("quit!");
                socketChannel.close();
                break;
            }

            ByteBuffer returnByteBuffer = ByteBuffer.allocate(1024);
            socketChannel.read(returnByteBuffer);
            String message = new String(returnByteBuffer.array()).trim();
            out.println("Receive message: " + message);
        }
    }

    public static void main(String[] args) throws IOException {
        tech.sohocoder.nio.noblock.Client client = new tech.sohocoder.nio.noblock.Client();
        client.start();
    }
}

server

package tech.sohocoder.nio.block;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;

public class Server {

    private ServerSocketChannel serverSocketChannel;

    private void start() throws IOException {
        serverSocketChannel = ServerSocketChannel.open();
        SocketAddress socketAddress = new InetSocketAddress(9000);
        serverSocketChannel.bind(socketAddress);

        while (true) {
            System.out.println("listening...");
            SocketChannel socketChannel = serverSocketChannel.accept();
            ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
            int readLength = socketChannel.read(byteBuffer);
            if(readLength != -1) {
                String receiveStr = new String(byteBuffer.array()).trim();
                System.out.println(receiveStr);
                socketChannel.write(byteBuffer);
            }
            socketChannel.close();
        }
    }

    public static void main(String[] args) throws IOException {
        Server server = new Server();
        server.start();
    }
}

NIO的非阻塞方式

NIO如果需要非阻塞,需要使用到selector。selector是在JDK1.4加入,主要是用于支持IO多路复用,Linux下jdk实现就是基于epoll。

client端代码保存一致。

server端实际上就是使用一个线程来支持多个连接

package tech.sohocoder.nio.noblock;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
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;
import java.util.Set;

import static java.lang.System.out;

public class Server {

    private ServerSocketChannel serverSocketChannel;

    private Selector selector;

    private void start() throws IOException, InterruptedException {
        serverSocketChannel = ServerSocketChannel.open();
        SocketAddress socketAddress = new InetSocketAddress(9000);
        serverSocketChannel.bind(socketAddress);

        serverSocketChannel.configureBlocking(false);

        int opSelectionKey = serverSocketChannel.validOps();

        selector = Selector.open();

        SelectionKey selectionKey = serverSocketChannel.register(selector, opSelectionKey);

        out.println(selector);
        out.println(selectionKey);
        while(true) {
            out.println("waiting for connected...");
            selector.select();
            Set set  = selector.selectedKeys();
            Iterator iterator = set.iterator();
            while (iterator.hasNext()) {
                SelectionKey mySelectionKey = iterator.next();
                if(mySelectionKey.isAcceptable()) {
                    SocketChannel socketChannel = serverSocketChannel.accept();
                    socketChannel.configureBlocking(false);
                    SelectionKey selectionKey1 = socketChannel.register(selector, SelectionKey.OP_READ);
                    out.println("socket channel selectionkey: " + selectionKey1);
                    out.println("connect from : " + socketChannel.getRemoteAddress());
                }else if(mySelectionKey.isReadable()){
                    SocketChannel socketChannel = (SocketChannel) mySelectionKey.channel();
                    ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
                    socketChannel.read(byteBuffer);
                    String message = new String(byteBuffer.array()).trim();
                    out.println("Receive message: " + message);
                    if(message.equals("quit")) {
                        out.println("close connection: " + socketChannel.getRemoteAddress());
                        socketChannel.close();
                        mySelectionKey.cancel();
                    }else {
                        ByteBuffer returnByteBuffer = ByteBuffer.wrap(" receive your message".getBytes());
                        socketChannel.write(returnByteBuffer);
                    }
                }
                iterator.remove();
            }
        }

    }

    public static void main(String[] args) throws IOException, InterruptedException {
        Server server = new Server();
        server.start();
    }
}

AIO实现

上面的IO,NIO的阻塞实际上是同步阻塞的方式,NIO的非阻塞是同步非阻塞方式。AIO(asynchronous I/O))是异步IO,实现是异步非阻塞方式,在jdk1.7中引入。

server端源码如下:

package tech.sohocoder.aio;

import java.io.IOException;
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.ExecutionException;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;

import static java.lang.System.out;

public class Server {

    private AsynchronousServerSocketChannel asynchronousServerSocketChannel;

    private void start() throws IOException, InterruptedException {
        // worker thread pool
        AsynchronousChannelGroup asynchronousChannelGroup = AsynchronousChannelGroup.withCachedThreadPool(Executors.newCachedThreadPool(), 4);
        asynchronousServerSocketChannel = AsynchronousServerSocketChannel.open(asynchronousChannelGroup);
        int port = 9000;
        InetSocketAddress socketAddress = new InetSocketAddress("localhost", port);
        asynchronousServerSocketChannel.bind(socketAddress);

        out.println("Starting listening on port " + port);
        // add handler
        asynchronousServerSocketChannel.accept(asynchronousServerSocketChannel, new CompletionHandler() {
            @Override
            public void completed(AsynchronousSocketChannel asynchronousSocketChannel, Object o) {

                try {
                    out.println("connect from : " + asynchronousSocketChannel.getRemoteAddress());
                } catch (IOException e) {
                    e.printStackTrace();
                }
                // accept next connection
                asynchronousServerSocketChannel.accept(asynchronousServerSocketChannel, this);
                while (true) {
                    ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
                    Future future = asynchronousSocketChannel.read(byteBuffer);
                    try {
                        future.get();
                        String message = new String(byteBuffer.array()).trim();
                        out.println("Receive message: " + message);
                        if (message.equals("quit")) {
                            out.println("close client: " + asynchronousSocketChannel.getRemoteAddress());
                            asynchronousSocketChannel.close();
                            break;
                        }

                        ByteBuffer returnByteBuffer = ByteBuffer.wrap("receive your message".getBytes());
                        Future returnFuture = asynchronousSocketChannel.write(returnByteBuffer);
                        returnFuture.get();

                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } catch (ExecutionException e) {
                        e.printStackTrace();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }

            @Override
            public void failed(Throwable throwable, Object o) {
                out.println("error to accept: " + throwable.getMessage());
            }
        });
        asynchronousChannelGroup.awaitTermination(Integer.MAX_VALUE, TimeUnit.MINUTES);
    }

    public static void main(String[] args) throws IOException, InterruptedException {
        Server server = new Server();
        server.start();
    }
}

Netty实现

Netty是java中使用很广泛的库,既可以实现NIO也可以实现AIO,还是针对上面的例子来实现一下

server端

package tech.sohocoder.netty;

import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.codec.string.StringEncoder;

import static java.lang.System.out;

public class Server {

    private void start() throws InterruptedException {
        EventLoopGroup bossEventLoopGroup = new NioEventLoopGroup(1);
        EventLoopGroup workerEventLoopGroup = new NioEventLoopGroup();
        try {
            ServerBootstrap serverBootstrap = new ServerBootstrap();
            serverBootstrap.group(bossEventLoopGroup, workerEventLoopGroup)
                    .channel(NioServerSocketChannel.class)
                    .childHandler(new ChannelInitializer() {
                        @Override
                        protected void initChannel(SocketChannel socketChannel) throws Exception {
                            // add handler into pipeline
                            socketChannel.pipeline()
                                    .addLast(new StringDecoder())
                                    .addLast(new StringEncoder())
                                    .addLast(new ServerHandler());
                        }
                    });

            ChannelFuture channelFuture = serverBootstrap.bind(9000).sync();
            out.println("listening...");
            channelFuture.channel().closeFuture().sync();
        }finally {
           bossEventLoopGroup.shutdownGracefully();
        }

    }

    public static void main(String[] args) throws InterruptedException {
        Server server = new Server();
        server.start();
    }

}

这里面需要使用到ServerHandler,具体代码如下

package tech.sohocoder.netty;

import io.netty.channel.ChannelDuplexHandler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelPromise;

import java.net.SocketAddress;

import static java.lang.System.out;

public class ServerHandler extends ChannelDuplexHandler {

    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
        out.println("Receive message: " + msg);
        String message = "receive your message";
        ctx.writeAndFlush(message);
    }

    @Override
    public void connect(ChannelHandlerContext ctx, SocketAddress remoteAddress, SocketAddress localAddress, ChannelPromise promise) throws Exception {
        super.connect(ctx, remoteAddress, localAddress, promise);
    }

    @Override
    public void channelActive(ChannelHandlerContext ctx) throws Exception {
        out.println("connect from: " + ctx.channel().remoteAddress().toString());
        super.channelActive(ctx);
    }

    @Override
    public void channelInactive(ChannelHandlerContext ctx) throws Exception {
        out.println("close connection: " + ctx.channel().remoteAddress().toString());
        super.channelInactive(ctx);
    }

}

client端也用netty写一下

package tech.sohocoder.aio;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;

import static java.lang.System.out;

public class Client {

    private SocketChannel socketChannel;

    public void start() throws IOException {
        socketChannel = SocketChannel.open();
        SocketAddress socketAddress = new InetSocketAddress("localhost", 9000);
        socketChannel.connect(socketAddress);
        if(socketChannel.isConnected()) {
            out.println("connect to " + socketChannel.getRemoteAddress());
        }

        BufferedReader in = new BufferedReader(new InputStreamReader(System.in));
        for (;;) {
            final String input = in.readLine();
            final String line = input != null ? input.trim() : null;
            if (null == line || line.isEmpty()) { // skip `enter` or `enter` with spaces.
                continue;
            }

            ByteBuffer byteBuffer = ByteBuffer.wrap(line.getBytes());
            socketChannel.write(byteBuffer);

            if(line.equals("quit")) {
                out.println("quit!");
                socketChannel.close();
                break;
            }

            ByteBuffer returnByteBuffer = ByteBuffer.allocate(1024);
            socketChannel.read(returnByteBuffer);
            String message = new String(returnByteBuffer.array()).trim();
            out.println("Receive message: " + message);
        }
    }

    public static void main(String[] args) throws IOException {
        Client client = new Client();
        client.start();
    }
}

同样要实现一个ClientHandler

package tech.sohocoder.netty;

import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;

import static java.lang.System.out;

public class ClientHandler extends SimpleChannelInboundHandler {
    @Override
    protected void channelRead0(ChannelHandlerContext channelHandlerContext, String s) throws Exception {
        out.println("Receive message: " + s);
    }
}

你可能感兴趣的:(Java IO, NIO, AIO和Netty)