The day arch a pawn, unexpected!

Hello, I’m Tong Brother, this is the seventh episode of netty series.

Introduction to the

In the last chapter we looked at Buffer, the core component of Java NIO, which is usually used with channels, but how to use them in network IO. Today we’ll look at another core component of NIO, Selector, without which you can’t do network IO.

concept

Let’s look at two of the Selector, see class. Java nio. Channels. The Selector.

Note the I

A multiplexor of {@link SelectableChannel} objects.

It is a multiplexer for SelectableChannel objects, from which we can also see that Java NIO is actually multiplexing IO.

SelectableChannel has several subclasses that you’ll be familiar with:

  • DatagramChannel, UDP protocol connection
  • SocketChannel: TCP connection
  • ServerSocketChannel, which processes TCP Accept events

It is worth reviewing the process of multiplexing IO:

The first phase uses select polling to check if the connection is ready for data, and the second phase copies the data from kernel space to user space.

In Java, the first phase is implemented using the Selector multiplexer.

Note 2

A selector may be created by invoking the {@link #open open} method of this class, which will use the system’s default {@link java.nio.channels.spi.SelectorProvider selector provider} to create a new selector. A selector may also be created by invoking the {@link java.nio.channels.spi.SelectorProvider#openSelector openSelector} method of a custom selector provider. A selector remains open until it is closed via its {@link #close close} method.

The Selector can through its own open () method to create, it will be sent to you by the default Java. Nio. Channels. The spi. SelectorProvider class to create a new Selector. Can also by implementing Java nio. Channels. Spi. SelectorProvider class of abstract methods openSelector () to customize a Selector. The Selector, once created, will remain open until the close() method is called.

So, what Selector is going to be used by default?

By tracing the source code:

> Java nio. Channels. Selector# open (1) > Java nio. Channels. Spi. SelectorProvider# provider (1.1) > Sun. Nio. Ch. DefaultSelectorProvider# the create () / / return WindowsSelectorProvider 2 > Sun. Nio. Ch. WindowsSelectorProvider# openSelector () / / return WindowsSelectorImplCopy the code

As you can see, the default Provider implemented on Windows is WindowsSelectorProvider, whose openSelector() method returns WindowsSelectorImpl, It’s the default Selector implementation for the Windows platform.

Why mention Windows, isn’t it different under Linux?

Yes, because network IO is closely related to the operating system, different operating system implementation may be different, Linux JDK implementation is completely different, so why do we not realize it? I wrote my code under Windows, but wouldn’t it run the same way under Linux? That’s what the Java Virtual Machine (or Java runtime environment) does for us. It hides the details of the operating system, which is the essence of Java code being able to “Write Once, Run Anywhere.”

Selector versus Channel

We said that a selector is a multiplexer, and it’s used in the first phase of a network IO to poll to see if a connection is ready for data, so what’s the relationship between a selector and a Channel?

The Selector listens for events on multiple channels at the same time by constantly polling, and notice that it listens at the same time, and as soon as a Channel is ready, it returns those channels that are ready to be processed by the processing thread.

Therefore, in NIO programming, we can achieve the goal of processing multiple connection requests simultaneously by a thread through Selector, which can also reduce the consumption of server resources.

Basic usage

Create a Selector

By calling Selector. Open () is the way we usually do it:

Selector selector = Selector.open();Copy the code

Can also, of course, by implementing the Java. Nio. Channels. Spi. SelectorProvider. OpenSelector () abstract methods custom a Selector.

Registers a Channel with a Selector

To bind a Channel to a Selector, you need to register a Channel with a Selector by calling its register() method:

channel.configureBlocking(false);

SelectionKey key = channel.register(selector, SelectionKey.OP_READ);Copy the code

A Channel must be non-blocking to be registered with a Selector, so you can’t register a FileChannel with a Selector, because a FileChannel doesn’t have blocking or non-blocking modes.

The second parameter is passed to the listener event. There are four types of events:

  • Connect
  • Accept
  • Read
  • Write

When a Channel fires an event, it is usually said that the event is ready. For example, when data is ready to be read, it is read ready. Similarly, there are write ready, connect ready, and receive ready, which are not often heard.

In Java, these four listener events are defined in the SelectionKey:

  • Selectionkey. OP_READ, value 1 << 0 = 0000 0001
  • Selectionkey. OP_WRITE, value 1 << 2 = 0000 0100
  • Selectionkey. OP_CONNECT, value 1 << 3 = 0000 1000
  • OP_ACCEPT, the value is 1 << 4 = 0001 0000

So, it is also possible to listen for multiple events of interest through bits or commands:

int interestSet = SelectionKey.OP_READ | SelectionKey.OP_WRITE;Copy the code

SelectionKey

As you can see, a Channel registers with a Selector and returns a SelectionKey, so the SelectionKey can be thought of as a bridge between the Channel and the Selector, binding them together.

SelectionKey has the following important properties:

  • An interest set is a set of events of interest
  • Ready set, a set of events that are ready
  • Saved Channel
  • Selector that’s saved
  • B: That’s right

interest set

It holds the second parameter passed in when registering a Channel to Selector, which is the set of events of interest.

int interestSet = selectionKey.interestOps();

boolean isInterestedInAccept  = interestSet & SelectionKey.OP_ACCEPT;
boolean isInterestedInConnect = interestSet & SelectionKey.OP_CONNECT;
boolean isInterestedInRead    = interestSet & SelectionKey.OP_READ;
boolean isInterestedInWrite   = interestSet & SelectionKey.OP_WRITE;    Copy the code

You can use the bit and operation to see if the event is registered.

ready set

It holds the set of events that are ready.

int readySet = selectionKey.readyOps();
selectionKey.isAcceptable();
selectionKey.isConnectable();
selectionKey.isReadable();
selectionKey.isWritable();Copy the code

You can retrieve all ready events using the readyOps() method, and you can check whether an event is ready using the isXxxable() method.

Saved channels and selectors

Channel  channel  = selectionKey.channel();

Selector selector = selectionKey.selector();    Copy the code

The channel() and selector() methods get the bound channel and selector.

attachment

You can attach an object to the SelectionKey by calling the attach(obj) method, and pull the attached object out of the attachment() method if needed later. It can also be translated as an attachment. It can be considered as a data transfer medium, similar to ThreadLocal. Bind data in front and use it later.

selectionKey.attach(theObject);

Object attachedObj = selectionKey.attachment();Copy the code

Of course, you can also bind an attachment when registering a Channel with a Selector:

SelectionKey key = channel.register(selector, SelectionKey.OP_READ, theObject);Copy the code

Selector.select()

Once one or more channels have been registered with Selector, we can call its select() method, which returns ready events of interest at the time of registration.

There are three variants of the select() method:

  • Select (), with no arguments, blocks until a Channel has a ready event (of course, the event of interest we registered)
  • Select (timeout), with timeout, blocks until a Channel has a ready event, or times out
  • SelectNow (), which returns immediately without blocking, regardless of ready channels

The return value of select() is int, indicating the ready Channel between two select() calls. The next call to select() does not return the last ready Channel, even if the ready Channel returned from the last call to select() was not processed. For example, if the first call to select() returns a ready Channel but does not process it, and the second call to select() returns a ready Channel, it will return 1, not 2.

Selector.selectedKeys()

Once a call to select() returns a ready Channel, we can use the selectedKeys() method to retrieve the ready Channel.

Set<SelectionKey> selectedKeys = selector.selectedKeys();    Copy the code

We can then iterate over the selectionkeys to see if the events of interest are ready:

Set<SelectionKey> selectedKeys = selector.selectedKeys(); Iterator<SelectionKey> keyIterator = selectedKeys.iterator(); while(keyIterator.hasNext()) { SelectionKey key = keyIterator.next(); if(key.isAcceptable()) { // a connection was accepted by a ServerSocketChannel. } else if (key.isConnectable()) { // a connection was established with a remote server. } else if (key.isReadable()) { // a channel is ready for reading } else  if (key.isWritable()) { // a channel is ready for writing } keyIterator.remove(); }Copy the code

Finally, be sure to call keyiterator.remove (); Removes the SelectionKey that has already been processed.

Selector.wakeup()

Earlier we said that when the select() method is called, the caller thread blocks until a Channel is ready to return. Wakeup () can be called on another thread to forcibly wakeup the blocked thread, and the select() method will return immediately.

If wakeup() is called without a thread blocking on select(), the next call to select() will return immediately and will not be blocked. This is similar to the locksupport.unpark () method.

Selector.close()

Calling close() turns off the Selector and invalidates the associated SelectionKey, but does not close the Channel.

Take a chestnut

Public class EchoServer {public static void main(String[] args) throws IOException {// Create a Selector Selector = Selector.open(); ServerSocketChannel ServerSocketChannel = ServerSocketChannel.open(); Serversocketchannel. bind(new InetSocketAddress(8080)); / / set to non-blocking mode, this paper comes from the work from the red elder brother read source serverSocketChannel. ConfigureBlocking (false); / / will be registered on the selector Channel, and register the Accept events serverSocketChannel. Register (selector, SelectionKey. OP_ACCEPT); While (true) {// block selector on select. Select (); Select (timeout) or selectNow() if the value returned is greater than 0 selector.selectedKeys(); SelectKeys Iterator<SelectionKey> Iterator = selectionkeys.iterator (); while (iterator.hasNext()) { SelectionKey selectionKey = iterator.next(); / / if the accept event if (selectionKey isAcceptable ()) {/ / cast to ServerSocketChannel ServerSocketChannel SSC = (ServerSocketChannel) selectionKey.channel(); SocketChannel socketChannel = ssc.accept(); System.out.println("accept new conn: " + socketChannel.getRemoteAddress()); socketChannel.configureBlocking(false); Socketchannel.register (Selector, selectionkey.op_read); // Register socketChannel.register(Selector, selectionkey.op_read); } else if (selectionkey.isreadable ()) {// If it is a read event // cast to SocketChannel SocketChannel SocketChannel = (SocketChannel) selectionKey.channel(); ByteBuffer = ByteBuffer. Allocate (1024); Int length = socketchannel.read (buffer); int length = socketchannel.read (buffer); if (length > 0) { buffer.flip(); byte[] bytes = new byte[buffer.remaining()]; // Read the data into the byte array buffer.get(bytes); String Content = new String(bytes, "UTF-8").replace("\r\n", ""); if (content.equalsIgnoreCase("quit")) { selectionKey.cancel(); socketChannel.close(); } else { System.out.println("receive msg: " + content); } } } iterator.remove(); }}}}Copy the code

conclusion

Today we looked at the core Java NIO component Selector, so that’s the end of the three most important core components of NIO, and to be honest, the most important part of NIO is thinking, keep in mind that in NIO one thread can handle multiple connections.

Doesn’t it seem difficult to watch Java native NIO implement network programming? So why Netty? In the next chapter, we will enter the study of Netty and find out.

Finally, welcome to my work from the number tong elder brother read source code to learn the knowledge of source code & architecture.