Interface ChannelFuture
-
- All Known Subinterfaces:
ChannelProgressiveFuture,ChannelProgressivePromise,ChannelPromise
- All Known Implementing Classes:
DefaultChannelProgressivePromise,DefaultChannelPromise,DelegatingChannelPromiseNotifier,VoidChannelPromise
public interface ChannelFuture extends io.netty.util.concurrent.Future<Void>
The result of an asynchronousChannelI/O operation.All I/O operations in Netty are asynchronous. It means any I/O calls will return immediately with no guarantee that the requested I/O operation has been completed at the end of the call. Instead, you will be returned with a
ChannelFutureinstance which gives you the information about the result or status of the I/O operation.A
ChannelFutureis either uncompleted or completed. When an I/O operation begins, a new future object is created. The new future is uncompleted initially - it is neither succeeded, failed, nor cancelled because the I/O operation is not finished yet. If the I/O operation is finished either successfully, with failure, or by cancellation, the future is marked as completed with more specific information, such as the cause of the failure. Please note that even failure and cancellation belong to the completed state.+---------------------------+ | Completed successfully | +---------------------------+ +----> isDone() = true | +--------------------------+ | | isSuccess() = true | | Uncompleted | | +===========================+ +--------------------------+ | | Completed with failure | | isDone() = false | | +---------------------------+ | isSuccess() = false |----+----> isDone() = true | | isCancelled() = false | | | cause() = non-null | | cause() = null | | +===========================+ +--------------------------+ | | Completed by cancellation | | +---------------------------+ +----> isDone() = true | | isCancelled() = true | +---------------------------+Various methods are provided to let you check if the I/O operation has been completed, wait for the completion, and retrieve the result of the I/O operation. It also allows you to addChannelFutureListeners so you can get notified when the I/O operation is completed.Prefer
It is recommended to preferaddListener(GenericFutureListener)toawait()addListener(GenericFutureListener)toawait()wherever possible to get notified when an I/O operation is done and to do any follow-up tasks.addListener(GenericFutureListener)is non-blocking. It simply adds the specifiedChannelFutureListenerto theChannelFuture, and I/O thread will notify the listeners when the I/O operation associated with the future is done.ChannelFutureListeneryields the best performance and resource utilization because it does not block at all, but it could be tricky to implement a sequential logic if you are not used to event-driven programming.By contrast,
await()is a blocking operation. Once called, the caller thread blocks until the operation is done. It is easier to implement a sequential logic withawait(), but the caller thread blocks unnecessarily until the I/O operation is done and there's relatively expensive cost of inter-thread notification. Moreover, there's a chance of dead lock in a particular circumstance, which is described below.Do not call
await()insideChannelHandlerThe event handler methods in
ChannelHandlerare usually called by an I/O thread. Ifawait()is called by an event handler method, which is called by the I/O thread, the I/O operation it is waiting for might never complete becauseawait()can block the I/O operation it is waiting for, which is a dead lock.// BAD - NEVER DO THIS
@Overridepublic void channelRead(ChannelHandlerContextctx, Object msg) {ChannelFuturefuture = ctx.channel().close(); future.awaitUninterruptibly(); // Perform post-closure operation // ... } // GOOD@Overridepublic void channelRead(ChannelHandlerContextctx, Object msg) {ChannelFuturefuture = ctx.channel().close(); future.addListener(newChannelFutureListener() { public void operationComplete(ChannelFuturefuture) { // Perform post-closure operation // ... } }); }In spite of the disadvantages mentioned above, there are certainly the cases where it is more convenient to call
await(). In such a case, please make sure you do not callawait()in an I/O thread. Otherwise,BlockingOperationExceptionwill be raised to prevent a dead lock.Do not confuse I/O timeout and await timeout
The timeout value you specify withFuture.await(long),Future.await(long, TimeUnit),Future.awaitUninterruptibly(long), orFuture.awaitUninterruptibly(long, TimeUnit)are not related with I/O timeout at all. If an I/O operation times out, the future will be marked as 'completed with failure,' as depicted in the diagram above. For example, connect timeout should be configured via a transport-specific option:// BAD - NEVER DO THIS
Bootstrapb = ...;ChannelFuturef = b.connect(...); f.awaitUninterruptibly(10, TimeUnit.SECONDS); if (f.isCancelled()) { // Connection attempt cancelled by user } else if (!f.isSuccess()) { // You might get a NullPointerException here because the future // might not be completed yet. f.cause().printStackTrace(); } else { // Connection established successfully } // GOODBootstrapb = ...; // Configure the connect timeout option. b.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 10000);ChannelFuturef = b.connect(...); f.awaitUninterruptibly(); // Now we are sure the future is completed. assert f.isDone(); if (f.isCancelled()) { // Connection attempt cancelled by user } else if (!f.isSuccess()) { f.cause().printStackTrace(); } else { // Connection established successfully }
-
-
Method Summary
All Methods Instance Methods Abstract Methods Modifier and Type Method Description ChannelFutureaddListener(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>> listener)ChannelFutureaddListeners(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>>... listeners)ChannelFutureawait()ChannelFutureawaitUninterruptibly()Channelchannel()Returns a channel where the I/O operation associated with this future takes place.booleanisVoid()Returnstrueif thisChannelFutureis a void future and so not allow to call any of the following methods:addListener(GenericFutureListener)addListeners(GenericFutureListener[])await()Future.await(long, TimeUnit)()}Future.await(long)()}awaitUninterruptibly()sync()syncUninterruptibly()ChannelFutureremoveListener(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>> listener)ChannelFutureremoveListeners(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>>... listeners)ChannelFuturesync()ChannelFuturesyncUninterruptibly()-
Methods inherited from interface io.netty.util.concurrent.Future
await, await, awaitUninterruptibly, awaitUninterruptibly, cancel, cause, getNow, isCancellable, isSuccess
-
Methods inherited from interface java.util.concurrent.Future
get, get, isCancelled, isDone
-
-
-
-
Method Detail
-
channel
Channel channel()
Returns a channel where the I/O operation associated with this future takes place.
-
addListener
ChannelFuture addListener(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>> listener)
- Specified by:
addListenerin interfaceio.netty.util.concurrent.Future<Void>
-
addListeners
ChannelFuture addListeners(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>>... listeners)
- Specified by:
addListenersin interfaceio.netty.util.concurrent.Future<Void>
-
removeListener
ChannelFuture removeListener(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>> listener)
- Specified by:
removeListenerin interfaceio.netty.util.concurrent.Future<Void>
-
removeListeners
ChannelFuture removeListeners(io.netty.util.concurrent.GenericFutureListener<? extends io.netty.util.concurrent.Future<? super Void>>... listeners)
- Specified by:
removeListenersin interfaceio.netty.util.concurrent.Future<Void>
-
sync
ChannelFuture sync() throws InterruptedException
- Specified by:
syncin interfaceio.netty.util.concurrent.Future<Void>- Throws:
InterruptedException
-
syncUninterruptibly
ChannelFuture syncUninterruptibly()
- Specified by:
syncUninterruptiblyin interfaceio.netty.util.concurrent.Future<Void>
-
await
ChannelFuture await() throws InterruptedException
- Specified by:
awaitin interfaceio.netty.util.concurrent.Future<Void>- Throws:
InterruptedException
-
awaitUninterruptibly
ChannelFuture awaitUninterruptibly()
- Specified by:
awaitUninterruptiblyin interfaceio.netty.util.concurrent.Future<Void>
-
isVoid
boolean isVoid()
Returnstrueif thisChannelFutureis a void future and so not allow to call any of the following methods:addListener(GenericFutureListener)addListeners(GenericFutureListener[])await()Future.await(long, TimeUnit)()}Future.await(long)()}awaitUninterruptibly()sync()syncUninterruptibly()
-
-