This is the 13th day of my participation in the Gwen Challenge in November. Check out the details: The Last Gwen Challenge in 2021 “.

A, the channel

1.1 Main methods of channel

1) Close () can be used to close a channel. 2) closeFuture() can be used to close a channel in either of the following ways

The sync method waits synchronously for a channel to close and the addListener method waits asynchronously for a channel to close

3) Pipeline () method adds processor 4) Write () method writes data 5) writeAndFlush() method writes data and brushes out

1.2 What is channelFuture?

Let’s take a look at the client code that was used in the previous article

public static void main(String[] args) throws InterruptedException { Channel channel = new Bootstrap() .group(new NioEventLoopGroup(1)) .handler(new ChannelInitializer<NioSocketChannel>() { @Override protected void initChannel(NioSocketChannel ch) throws Exception { System.out.println("init..." ); ch.pipeline().addLast(new StringEncoder()); } }) .channel(NioSocketChannel.class) .connect("localhost", 8080) .sync() .channel(); channel.writeAndFlush("ccc"); Thread.sleep(1000); channel.writeAndFlush("ccc"); }Copy the code

Call connect(), which returns a ChannelFuture object that can be retrieved through the channel() method.

    public ChannelFuture connect(String inetHost, int inetPort) {
        return this.connect(InetSocketAddress.createUnresolved(inetHost, inetPort));
    }
Copy the code
public interface ChannelFuture extends Future<Void> { Channel channel(); . . }Copy the code

Note that the connect method is an asynchronous method and does not actually establish a connection after the call, so the correct channel is not immediately available in the ChannelFuture object.

To see what happens, start a server on port 8080. The server code is not provided here, just use the previous one. To start the test client code we wrote:

public class ChannelFutureTest { public static void main(String[] args) throws Exception { ChannelFuture channelFuture =  new Bootstrap() .group(new NioEventLoopGroup()) .channel(NioSocketChannel.class) .handler(new ChannelInitializer<NioSocketChannel>() { @Override protected void initChannel(NioSocketChannel nioSocketChannel) { } }) .connect("localhost", 8080); System.out.println(channelFuture.channel()); // synchronize wait to connect channelfuture.sync (); System.out.println(channelFuture.channel()); }}Copy the code

Results:

[id: 0 x7aa12c28] [id: 0 x7aa12c28, L: / 127.0.0.1:52286 - R: localhost / 127.0.0.1:8080]Copy the code

As shown in the above result, the first print is a single ID address. When sync() is executed, the address will block synchronously waiting for the connection, and a timeout exception will be thrown if the connection fails. When the connection is successfully established, execution continues and the last line of the previous result is printed.

In addition to using the sync() method, there is an asynchronous way:

/ / asynchronous channelFuture. AddListener ((ChannelFutureListener) future - > {System. Out. Println (future) channel ()); });Copy the code

Results:

[id: 0 xd9d474f1] [id: 0 xd9d474f1, L: / 127.0.0.1:59564 - R: localhost / 127.0.0.1:8080]Copy the code

ChannelFutureListener, which is called when the connection is established (with the operationComplete method), is a functional interface call.

1.3 What is CloseFuture?

Let’s demonstrate this with a code that involves the close method of channel and the close method of CloseFuture. Closing is to release occupied resources.

Look at the following code:

public class ChannelFutureTest { public static void main(String[] args) throws Exception { ChannelFuture channelFuture =  new Bootstrap() .group(new NioEventLoopGroup()) .channel(NioSocketChannel.class) .handler(new ChannelInitializer<NioSocketChannel>() { @Override protected void initChannel(NioSocketChannel ch) { ch.pipeline().addLast(new StringEncoder()); } }) .connect("localhost", 8080); Channel = channelfuture.sync ().channel(); new Thread(()->{ Scanner scanner = new Scanner(System.in); while (true) { String line = scanner.nextLine(); If ("q".equals(line)) {system.out.println (" close channel"); // close channel.close(); break; } channel.writeAndFlush(line); } }, "input").start(); System.out.println(" Handle the operation after a channel is closed "); }Copy the code

Our client, as shown in the code above, allows the user to manually type Q to close the program or send content to the server.

But run the client directly as shown above and find System.out.println(” handle the operation after a channel is closed “); This command executes directly because our main shutdown business logic is implemented by enabling child threads.

This means that we execute code on the child thread, the channel, before it closes, which may cause problems with our business logic.

Therefore, we need to print after the channel is closed, which is the remaining business operations after the channel is closed in the real scenario.

Println (” Handle the operation after channel closure “); Add the following code before:

// Get closeFuture ChannelFuture closeFuture = channel.closeFuture(); // sync blocks closeFuture.sync();Copy the code

The code gets a closeFuture object, where the sync method blocks synchronically until the channel in the child thread is actually closed.

Input 1, 2, 3, q to see the result directly:

1 2 3 Q Closing a channel This operation is performed after a channel is closedCopy the code

As with channelFuture, closeFuture can also listen asynchronously to see if a channel is closed, except that the sync method blocks synchronously.

Println (” Handle the operation after a channel is closed “); Put it in the following code:

Closefuture.addlistener ((ChannelFutureListener) Future -> System.out.println(" Handle the operation after channel closure ");) ;Copy the code

Input 1, 2, 3, q to see the result:

1 2 3 Q Closing a channel This operation is performed after a channel is closedCopy the code

Provide a NioEventLoopGroup dedicated to closing.

NioEventLoopGroup group = new NioEventLoopGroup();
Copy the code

We have successfully detected the closure of the channel through the above code, but we believe that our friends will find that although the channel is closed, the whole program is still running, and the whole resources are not released. This is because threads in the EventLoopGroup are not stopped. Here’s a method to introduce:

shutdownGracefully()

This method is the method in the EventLoopGroup. We need to do the following. For your sake, I have put all the client content in the following code:

Public class ChannelFutureTest {public static void main(String[] args) throws Exception { Gracefully, the NioEventLoopGroup group = new NioEventLoopGroup(); ChannelFuture channelFuture = new Bootstrap() .group(group) .channel(NioSocketChannel.class) .handler(new ChannelInitializer<NioSocketChannel>() { @Override protected void initChannel(NioSocketChannel ch) { ch.pipeline().addLast(new StringEncoder()); } }) .connect("localhost", 8080); Channel = channelfuture.sync ().channel(); new Thread(() -> { Scanner scanner = new Scanner(System.in); while (true) { String line = scanner.nextLine(); If ("q".equals(line)) {system.out.println (" close channel"); // close channel.close(); break; } channel.writeAndFlush(line); } }, "input").start(); ChannelFuture CloseFuture = channel.closeFuture(); / / sync / / closeFuture. Sync (); / / asynchronous - EventLoopGroup thread not closed / / closeFuture addListener ((ChannelFutureListener) future - > System.out.println(" Handle the operation after channel closure ")); AddListener ((ChannelFutureListener) Future -> group.shutdownListener ()); }}Copy the code

Results:

1 2 3 Q Close the channel Process finished with exit code 0Copy the code

So much for channel, please give a thumbs up if it’s helpful.