Programmers love to tune their code, but I/O performance tuning is often an afterthought, or is ignored entirely. It's a shame, because even small investment in improving I/O performance can yield substantial dividends.
Buffers, and how buffers are handled, are the basis of all I/O. The very term "input/output" means nothing more than moving data in and out of buffers.
The image shows a simplified logical diagram of how block data moves from an external source, such as a disk, to a memory area inside a running process. The process requests that its buffer be filled by making the read() system call. This results in the kernel issuing a command to the disk controller hardware to fetch the data from disk. The disk controller writes the data directly into a kernel memory buffer by DMA without further assistance from the main CPU. Once the disk controller finishes filling the buffer, the kernel copies the data from the temporary buffer in kernel space to the buffer speicified by the process when it requested the read() operation. User space is where regular processes live. The JVM is a regular process and dwells in user space. User space is a nonprivileged area: code executing there cannot directly access hareware devices, for example. Kernel space is where the operating system lives. Kernel code has special privileges: it can communicate with device controllers, manipulate the state of processes in user space, etc. Most importantly, all I/O flows through kernel space, either directly or indirectly.
Operating system divide their memory address space into pages, which are fixed-size groups of bytes.
whats-the-difference-between-virtual-memory-and-swap-space
File I/O occurs within the context of a filesystem. A filesystem is a very different thing from a disk. Disks store data in sectors, which are usually 512 bytes each. They are hardware devices that know nothing about the semantics of files. They simply provide a number of slots where data can be stored. In this respect, the sectors of a disk are simliar to memory pages; all are of uniform size and are addressable as a large array. A filesystem is a higher level of abstraction. Filesystem are a particular method of arranging and interpreting data stored on a disk. The code you write almost always interacts with a filesystem, not with the disks directly. It is the filesystem that defines the abstractions of filenames, paths, files, file attributes, etc.
The Linux Filesystem Explained
Virtual memory and disk I/O are intimately linked and, in many respects, are simply two aspects of the same thing. Keep this in mind when handling large amounts of data. Most operating system are far more efficient when handling data buffers that are page-aligned and are mutiples of the native page size.
Buffers are not safe for use by mutiple concurrent threads. If a buffer is to be used by more than one thread then access to the buffer should be controlled by appropriate synchronization.
The meaning of position and limit depends on whether the Buffer is in read or write mode. Capacity always means the same, no matter the buffer mode. Capacity-Position-Limit
Remember in Java, characters are represented internally in Unicode,and each Unicode character occupies 16 bits.
We need to set the limit to the current position, then reset the position to 0.
public Buffer flip() {
limit = position;
position = 0;
mark = -1;
return this;
}
Flip is used to flip the ByteBuffer from "reading from I/O"(put) to "writing to I/O"(get). what-is-the-purpose-of-bytebuffers-flip-method-and-why-is-it-called-flip
The rewind() method is similar to flip() but does not affect the limit. It only sets the position back to 0. You can use rewind() to go back and reread the data in a buffer that has already been flipped.
What if you flip a buffer twice? It effectively becomes zero-sized. Apply the same steps to the buffer; set the limit to the position and the position to 0. Both the limit and position become 0.
Once a buffer has been filled and drained, it can be reused. The clear() method resets a buffer to an empty state. It doesn't change any of the data elements of the buffer but simply sets the limit to the capacity and the position back to 0.
public IntBuffer compact() {
System.arraycopy(hb, ix(position()), hb, ix(0), remaining());
position(remaining());
limit(capacity());
discardMark();
return this;
}
public Buffer clear() {
position = 0;
limit = capacity;
mark = -1;
return this;
}
Some buffer methods will discard the mark if one is set(rewind(),clear(),flip() always discard the mark). Calling the versions of limit() or position() that take index arguments will discard the mark if the new value being set is less than the current mark.
- Both objects are the same type. Buffers containing different data types are never equal, and no Buffer is ever equal to a non-Buffer object.
- Both buffers have the same number of remaining elements. The buffer capacities need not be the same, and the indexes of the data remaining in the buffers need not to be same. But the count of elements remaining(from position to limit) in each buffer must be the same.
- The sequence of remaining data elements, which would be returned from get(), must be identical in each buffer.
buffer.get(myArray); is equivalent to buffer.get(myArray,0,myArray.length); If the number of elements you ask for cannot be transferred, no data is transferred, the buffer state is left unchanged, and a BufferUnderflowException is thrown. So when you pass in an array and don't specify the length, you're asking for the entire array to be filled. If the buffer doesn't contain at least enough elements to completely fill the array, you'll get an exception. This means that if you want to transfer a small buffer into a large array, you need to explicitly specify the length of the data remaining in the Buffer.
char[] bigArray = new char[1000]; int length = buffer.remaining(); buffer.get(bigArray,0,length); char[] smallArray = new char[10]; while(buffer.hasRemaining()){ int length = Math.min(buffer.remaining(),smallArray.length); buffer.get(smallArray,0,length); }
buffer.put(myArray); is equivalent to buffer.put(myArray,0,myArray.length); If the buffer has room to accept the data in the array(buffer.remaining() >= myArray.length), the data will be copied into the buffer starting at the current position, and the buffer position will be advanced by the number of data elements added. If there is no sufficient room in the buffer, no data will be transferred into, and a BufferOverflowException will be thrown.
If you want to provide your own array to be used as the buffer's backing store, call the wrap() method:
char[] myArray = new char[100]; CharBuffer charBuffer = CharBuffer.wrap(myArray);
This constructs a new buffer object, but the data elements will live in the array. This implies that changes made to the buffer by invoking put() will be reflected in the array, and any changes made directly to the array will be visible to the buffer object.
Doing this:
CharBuffer charBuffer = CharBuffer.wrap(myArray,12,42);
create a CharBuffer with a position of 12, a limit of 54, and a capacity of myArray.length. This method does not, as you might expect, create a buffer that occupies only a subrange of the array. The buffer will have access to the full extent of the array; the offset and length arguments only set the initial state.
Buffers created by either allocate() or wrap() are always nondirect. Nondirect buffers have backing arrays.
Slicing a buffer is similar to duplicating, but slice() creates a new buffer that starts at the original buffer's current position and whose capacity is the number of elements remaining in the original buffer.
slice() and duplicate() have the same backing array with original buffer.
When moving data between the JVM and the operating system, it's necessary to break down the other data types into their constituent bytes.
The way multibyte numeric values are stored in memory is commonly referred to as endian-ness. If the numerically most-significant byte of the number, the big end, is at the lower address, then the system is big-endian. If the least-significant byte come first, it's little-endian.
The IPs define a notion of network byte order, which is big-endian. All multibyte numeric values used within the protocol portions of IP packets must be converted between the local host byte order and the common network byte order.
//java.nio.ByteBuffer boolean nativeByteOrder = (ByteOrder.nativeOrder() == ByteOrder.BIG_ENDIAN);
Bytebuffer objects posses a host of convenience mehods for getting and putting the buffer content as other primitive data types. The way these methods encode or decode the bytes is dependent on the ByteBuffer's current byte-order setting.
The most significant way in which byte buffers are distinguished from other buffer types is that they can be the source and/or targets of I/O performed by Channel.
Direct buffers are optimal for I/O, but they may be more expensive to create than nondirect byte buffers. The memory used by direct buffers is allocated by calling through to native, operating system-specific code, bypassing the standard JVM heap. Setting up and tearing down direct buffers could be significantly more expensive than heap-resident buffers, depending on the host operating system and JVM implementation. The memory-storage areas of direct buffers are not subject to garbage collection because they are outside the standard JVM heap.
While ByteBuffer is the only type that can be allocated as direct, isDirect() could be true for nonbyte view buffers if the underlying buffer is a direct ByteBuffer.
View buffers are created by a factory method on an existing buffer object instance. The view object maintains its own attribute, capacity, position, limit, and mark, but shares data elements with the original buffer.
Endianess has no meaning for a byte[]. Endianess only matter for multi-byte data types like short, int, long, float, or double. how-write-big-endian-bytebuffer-to-little-endian-in-java
Whenever a view buffer accesses the underlying bytes of a Bytebuffer, the bytes are packed to compose a data element according to the view buffer's byte-order setting. When a view buffer is created, it inherits the byte-order setting of the underlying ByteBuffer at the time the view is created. The byte-order setting of the view cannot be changed later. A CharBuffer view of a ByteBuffer
int value = buffer.getInt();
would return an int value composed of the byte values in locations 1-4 of the buffer. The actual value returned would depend on the current ByteOrder setting of the buffer. To be more specific:
int value = buffer.order(ByteOrder.BIG_ENDIAN).getInt();
returns the numeric value 0x3BC5315E, while:
int value = buffer.order(ByteOrder.LITTLE_ENDIAN).getInt();
returns the value 0x5E31C53B.
If the primitive data type you're trying to get requires more bytes than what remains in the buffer, a BufferUnderflowException will be thrown. The put methods perform the inverse operation of the gets. Primitive data values will be broken into bytes according to the byte order of the buffer and stored. If insufficient space is available to store all the bytes, a BufferOverflowException will be thrown.
Channels can be created in serveral ways. The socket channels have factory methods to create new socket channels directly. But a FileChannel object can be obtained only by calling the getChannel() method on an open RandomAccessFile, FileInputStream, or FileOutputStream object. You cannot create a FileChannel object directly.
SocketChannel sc = SocketChannel.open(); sc.connect(new InetSocketAddress("somehost"),someport); ServerSocketChannel ssc = ServerSocketChannel.open(); ssc.socket().bind(new InetSocketAddress(somelocalport)); DatagramChannel dc = DatagramChannel.open(); RandomAccessFile raf = new RandomAccessFile("somefile","r"); FileChannel fc = raf.getChannel();
Channels can be unidirectional or bidirectional. A given channel class might implement ReadableByteChannel, which defines the read() method. Another might implement WritableByteChannel to provide write(). A class implementing one or the other of these interface is unidirectional: it can transfer data in only one direction. If a class implements both interfaces, it is bidirectional and can transfer data in both directions.
Channels can operate in blocking or nonblocking modes. A channel in nonblocking mode never puts the invoking thread to sleep. The requested operation either completes immediately or returns a result indicating that nothing was done. Such as sockets and pipes, can be placed in nonblocking mode.
Java IO is a stream-oriented package which means that it can be read one or more bytes at a time from a stream. It uses a stream for transferring the data between data source/sink and java program. It is a unidirectional data transfer. Java NIO is a buffer-oriented package. This means that the data is read into a buffer from which it is further processed using a channel. NIO is a bidirectional data transfer. difference-between-java-io-and-java-nio
Unlike buffers, channels cannot be reused. An open channel represents a specific connection to a specific I/O service and encapsulates the state of that connection. When a channel is closed, that connection is lost, and the channel is no longer connected to anything.
Calling a channel's close() method might cause the thread to block briefly while the channel finalizes the closing of the underlying I/O service, even if the channel is in nonblocking mode. Blocking behavior when a channel is closed, if any, is highly operating system- and filesystem-dependent. It's harmless to call close() on a channel multiple times, but if the first thread has blocked in close(), any additional threads calling close() block until the first thread has completed closing the channel. Subsequent calls to close() on the closed channel do nothing and return immediately.
Data is gathered from each of the buffers referenced by the array of buffers and assembled into a stream of bytes that are sent down the channel.
Data arriving on the channel is scattered to the list of buffers, filling each in turn from its position to its limit. The position and limit values shown here are before the read operation commenced.
The version of read() and write() that take offset and length arguments provide a way to use subsets of the buffers in an array of buffers. The offset value in this case refers to which buffer to begin using, not an offset into the data. The length argument indicates the number of buffers to use. For example, if we have a five-element array named fiveBuffers that has already been initialized with references to five buffers, the following code would write the content of the second, third, and fourth buffers:
int bytesRead = channel.write(fiveBuffers,1,3);
File channel are alway blocking and cannot be placed into nonblocking mode.
Regular files are always readable and they are also always writable. This is clearly stated in the relevant POSIX specification. I cann't stress this enough. Putting a regular file in non-blocking has ABSOLUTELY no effects other than changing one bit in the file flags. Non-blocking I/O with regular files
Calling the getChannel() method returns a FileChannel object connected to the same file, with the same access permissions as the file object.
FileChannel objects are thread-safe. Multiple threads can concurrently call methods on the same instance without causing any problems, but not all operations are multithreaded. Threads attempting one of these operations will wait if another thread is already executing an operation that affects the channel position or file size. Concurrency behavior can also be affected by the underlying operating system or filesystem.
The FileChannel class guarantees that all instances within the same JVM will see a consistent view of a given file. But the JVM cannot make guarantee about factors beyond its control. The view of a file seen through a FileChannel instance may or may not be consistent with the view of that file seen by an external, non-java processes. The semantics of concurrent file access by multiple processes is hightly dependent on the underlying operating system and/or filesystem.
For comparison, image below lists the correspondences of FileChannel, RandomAccessFile, and POSIX I/O system calls.
Doing a write() with the position set beyond the file size will cause the file to grow to accommodate the new bytes written. The behavior is identical to that for an absolute write() and may result in a file hole.
The FileChannel position is reflected from the underlying file descriptor, which is shared by the file object from which the channel reference was obtained. This means that updates made to the position by one object will be seen by the other:
RandomAccessFile randomAccessFile = new RandomAccessFile ("filename", "r"); // Set the file position randomAccessFile.seek (1000); // Create a channel from the file FileChannel fileChannel = randomAccessFile.getChannel(); // This will print "1000" System.out.println ("file pos: " + fileChannel.position()); // Change the position using the RandomAccessFile object randomAccessFile.seek (500); // This will print "500" System.out.println ("file pos: " + fileChannel.position()); // Change the position using the FileChannel object fileChannel.position (200); // This will print "200" System.out.println ("file pos: " + randomAccessFile.getFilePointer());
Attempting an absolute read beyond the end of the file, as returned by size(), will return end-of-file. Doing an absolute write() at a position beyond the file size will cause the file to grow to accommodate the new bytes being written.
When it's necessary to reduce the size of a file, truncate() chops off any data beyond the new size you specify. If the current size is greater than the new size,all bytes beyond the new size are discarded. If the new size provided is greater than or equal to the current file size, the file is not modified. A side effect of truncate() is either case is that it sets the file position to the new size provided.
force() tells the channel to force any pending modifications made to the file out to disk. All modern filesystems cache data and defer disk updates to boost performance. Calling the force() method requests that all pending modifications to the file be synchronized to disk immediately.
Locks are associated with files, not channels. Use locks to coordinate with external processes, not between threads in the same JVM.
FileLock objects are thread-safe; multiple threads may access a lock object concurrently.
Although a FileLock object is associated with a speicify FileChannel instance, the lock it represents is associated with an underlying file, not the channel. This can cause conflicts, or possibly deadlock, if you don't release a lock when you're finished with it.
FileLock lock = fileChannel.lock() try { <perform read/write/whatever on channel> } catch (IOException) [ <handle unexpected exception> } finally { lock.release() }
The MappedByteBuffer object returned from map() behaves like a memory-based buffer in most respects, but its data elements are stored in a file on disk. Calling get() will fetch data from the disk file, and this data reflects the current content of the file, even if the file has been modified by an external process since the mapping was established. The data visible through a file mapping is exactly the same as you would see by reading the file conventionally.
Accessing a file through the memory-mapping mechanism can be far more efficient than reading or writing data by conventional means, even when using channel. No explicit system calls need to be made, which can be time-consuming. More importantly, the virtual memory system of the operating system automatically caches memory pages. These pages will be cached using system memory and will not consume space from the JVM's memory heap.
Large, structured files that contain indexes or other sections that are referenced or updated frequently can benefit tremendously from memory mapping.
Unlike ranges for file locks, mapped file ranges should not extend beyond the actual size of the file.
The requested mapping mode will be constrained by the access permissions of the FileChannel object on which map() is called. If the channel was opened as read-only, map() will throw a NonWritableChannelException if you ask for MapMode.READ_WRITE mode. NonReadableChannelException will be thrown if you request MapMode.READ_ONLY on a channel without read permission.
MapMode.PRIVATE indicates that you want a copy-on-write mapping. This means that any modifications you make via put() will result in a private copy of the data that only the MappedByteBuffer instance can see. No changes will be made to the underlying file, and any changes made will be lost when the buffer is garbage collected.
Once established, a mapping remains in effect until the MappedByteBuffer object is garbage collected. Unlike locks, mapped buffers are not tied to the channel that created them. Closing the associated FileChannel does not destroy the mapping; only disposal of the buffer object itself breaks the mappings.
When updating a file through a MappedByteBuffer object, you should always use MappedByteBuffer.force() rather than FileChannel.force(). The channel object may not be aware of all file updates made through the mapped buffer.
If the mapping was established with MapMode.READ_ONLY or MapMode.PRIVATE, then calling force() has no effect, since there will never be any changes to flush to disk.
The transferTo() and transferFrom() methods allow you to cross-connect one channel to another, eliminating the need to pass data through an intermediate buffer. These methods exist only on the FileChannel class, so one of the channels involved in a channel-to-channel transfer must be a FileChannel. You can't do direct transfer between socket channels, but socket channels implement WritableByteChannel and ReadableByteChannel, so the content of a file can be transferred to a socket with transferTo(), or data can be read from a socket directly into a file with transferFrom().
The new socket channels can operate in nonblocking mode and are selectable. These two capabilities enable tremendous scalablity and flexibility in large application, such as web servers and middleware components.
Notice that DatagramChannel and SocketChannel implement the interfaces that define read and write capabilities, but ServerSocketChannel does not. ServerSocketChannel listens for incoming connects and creates new SocketChannel objects. It never transfers any data itself.
java.nio.channels.SocketChannel = java.net.Socket
socketChannel.socket() = Socket
socketChannel.socket().getChannel() = SocketChannel
socket.getChannel() = null
java.nio.channels.ServerSocketChannel = java.net.ServerSocket
serverSocketChannel.socket() = ServerSocket
serverSocketChannel.socket().getChannel() = ServerSocketChannel
serverSocket.getChannel() = null
java.nio.channels.DatagramChannel = java.net.DatagramSocket
datagramChannel.socket() = DatagramSocket
datagramChannel.socket().getChannel() = DatagramChannel
datagramSocket.getChannel() = null
ServerSocketChannel doesn't have a bind() method, it's necessary to fetch the peer socket and use it to bind to a port to begin listening for connections. Also use the peer ServerSocket API to set other socket options as needed.
ServerSocketChannel ssc = ServerSocketChannel.open(); ServerSocket serverSocket = ssc.socket(); // Listen on port 1234 serverSocket.bind (new InetSocketAddress (1234));
The ServerSocketChannel has an accept() method, as does its peer java.net.ServerSocket object. Once you're created a ServerSocketChannel and used the peer socket to bind it, you can then invoke accept() on either. If you choose to invoke accept() on the ServerSocket, it will behave the same as any other ServerSocket: always blocking and returning a java.net.Socket object. On the other hand, the accept() method of ServerSocketChannel returns objects of type SocketChannel and is capable of operating in nonblocking mode. If a security manager is in place, both methods perform the same security checks.
SocketChannel socketChannel = SocketChannel.open (new InetSocketAddress ("somehost", somePort));
is equivalent to this:
SocketChannel socketChannel = SocketChannel.open(); socketChannel.connect (new InetSocketAddress ("somehost", somePort));
Stream-oriented sockets take time to set up because a packet dialog must take place between the two connecting systems to establish the state information needed to maintain the stream socket. Connecting to remote systems across the open Internet can be especially time-consuming. If a concurrent connection is underway on a SocketChannel, the isConnectPending() method returns true.
Call finishConnect() to complete the connection process. This method can be called safely at any time. One of the following will happen when invoking finishConnect() on SocketChannel object in nonblocking mode:
- The connect() method has not yet been called. A NoConnectionPendingException is thrown.
- Connection establishment is underway but not yet complete. Nothing happends, and finishConnect() immediately returns false.
- The SocketChannel has been switched back to blocking mode since calling connect() in nonblocking mode. If necessary, the invoking thread blocks util connection establishment is complete. finishConnect() then returns true.
- Connection establishment has completed since the initial invocation of connect() or the last call to finishConnect(). Internal state is updated in the SocketChannel object to complete the transition to connected state, and finishConnect() returns true. The SocketChannel object can then be used to transfer data.
- The connection is already established. Nothing happends, and finishConnect() returns true.
While in this intermediate connection-pending state, you should invoke only finishConnect(), isConnectPending(), or isConnected() on the channel. Once connection establishment has been successfully completed, isConnected() returns true.
InetSocketAddress addr = new InetSocketAddress (host, port); SocketChannel sc = SocketChannel.open(); sc.configureBlocking (false); sc.connect (addr); while ( ! sc.finishConnect()) { doSomethingElse(); } doSomethingWithChannel (sc); sc.close();
Socket channels are thread-safe. Mutilple threads do not need to take special steps to protect against concurrent access, but only one read and one write operation will be in progress at any given time. Keep in mind that sockets are stream-oriented, not packet-oriented. They guarantee that the bytes sent will arrive in the same order but make no promises about maintaining groupings. A sender may write 20 bytes to a socket, and the receiver gets only 3 of those bytes when invoking read(). The remaining 17 bytes may still be in transit. For this reason, it's rarely a good design choice to have multiple, noncooperating threads share the same side of a stream socket.
The connect() and finishConnect() methods are mutually synchronized, and any read or write calls will block while one of these operations is in progress, even in nonblocking mode. Test the connection state with isConnected() if there's any doubt or if you can't afford to let a read or write block on a channel in this circumstance.
Just as SocketChannel models connection-oriented stream protocols such as TCP/IP, DatagramChannel models connectionless packet-oriented protocols such as UDP.
DatagramChannel object can act both as server(listener) and client(sender).
If the ByteBuffer you provide does not have sufficient remaining space to hold the packet you're receiving, any bytes that don't fit will be sliently discard.
Here are some reasons to choose datagram sockets over stream sockets:
- Your application can tolerate lost or out-of-order data.
- You want to fire and forget and don't need to know if the packets you sent were received.
- Throughput is more important than reliability.
- You need to send to multiple receivers(multicast or broadcast) simultaneously.
- The packet metaphor fits the task at hand better than the stream metaphor.
A pipe is a conduit through which data can be passed in a single direction between two entities. The Pipe class implements a pipe paradigm, but the pipes it creates are intraprocess(within the JVM process) rather than interprocess(between processes). A pipe is a pair of looped channels
Summary of java.nio.channels.Channels utility methods The wrapper Channel objects returned by these methods may or may not implement the InterruptibleChannel interface. Also, they might not extend from SelectableChannel. Therefore, it may not be possible to use these wrapper channels interchangeable with the other channel types defined in the java.nio.channels package. The specifics are implementation-dependent. If your application relies on these semantics, test the returned channel object with the instanceof operator.
Selectors provide the ability to do readiness selection, which enables multiplexed I/O.
True readiness selection must be done by the operating system. One of the most important functions performed by an operating system is to handle I/O requests and notify processes when their data is ready.
A channel must first be placed in nonblocking mode(by calling configureBlocking(false)) before it can be registered with a selector.
Selectors contain sets of channels currently registered with them. Only one registeration of a given channel with a given selector can be in effect at any given time.
An exceptional situation is when you attempt to reregister a channel with a selector for which the associated key has been cancelled, but the channel is still registered. Channels are not immediately deregistered when the associated key is cancelled. They remain registered until the next selection operation occurs.
SelectionKey Objects represent a specific registeration relationship. When it's time to terminate that relationship, call the cancel() method on the SelectionKey object.
When a channel is closed, all keys associated with it are automatically cancelled. When a selector is closed, all channels registered with that selector are deregister, and the associated keys are invalidated(cancelled). Once a key has been invalidated, calling any of its methods related to selection will throw a CancelledKeyException.
Changes made to the interest set of a key while a select() is in progress on the associated Selector will not affect that selection operation. Any changes will be seen on the next invocation of select().
The ready set is a subset of the interest set and represents those operations from the interest set which were determineed to be ready on the channel by the last invocation of select().
if(key.isWritable()) is equivalent to : if((key.readyOps() & SelectionKey.OP_WRITE) != 0)
If the selection key is long-lived, but the object you attach should not be, remember to clear the attachment when you're done. Otherwise, your attached object will not be garbage collected, and you may have a memory leak.
One last thing to note about the SelectionKey class relates to concurrency. Generally, SelectionKey objects are thread-safe, but it's important to know that operations that modify the interest set are synchronized by Selector objects. This could cause calls to the interestOps() method to block for an indeterminate amount of time. The specific locking policy used by a selector, such as whether the locks are held throughout the selection process, is implementation-dependent. Luckily, this multiplexing capability is specifically designed to enable a single thread to manage many channels. Using selectors by multiple threads should be an issue in only the most complex of applications. Frankly, if you're sharing selectors among many threads and encountering synchronization issues, your design probably needs a rethink.
Essentially, selectors are a wrapper for a native call to select(), poll(), or a similar operating system-specific system call(epoll, kqueue).
Selector objects are thread-safe, but the key sets they contain are not.