More documentation on state management
This commit is contained in:
parent
15b3317df5
commit
3d2579b8cb
@ -15,6 +15,8 @@
|
|||||||
*/
|
*/
|
||||||
package org.jboss.netty.channel;
|
package org.jboss.netty.channel;
|
||||||
|
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provides the properties and operations which are specific to a
|
* Provides the properties and operations which are specific to a
|
||||||
* {@link ChannelHandler} and the {@link ChannelPipeline} it belongs to.
|
* {@link ChannelHandler} and the {@link ChannelPipeline} it belongs to.
|
||||||
@ -149,7 +151,9 @@ public interface ChannelHandlerContext {
|
|||||||
* Retrieves an object which is {@link #setAttachment(Object) attached} to
|
* Retrieves an object which is {@link #setAttachment(Object) attached} to
|
||||||
* this context.
|
* this context.
|
||||||
* <p>
|
* <p>
|
||||||
* As an alternative, you might want to use a {@link ChannelLocal} variable.
|
* As an alternative, you might want to use a {@link ChannelLocal} variable
|
||||||
|
* or a {@link ConcurrentMap} whose key is {@link ChannelHandlerContext}.
|
||||||
|
* Please refer to {@link ChannelPipelineCoverage} for the detailed examples.
|
||||||
*
|
*
|
||||||
* @return {@code null} if no object was attached or
|
* @return {@code null} if no object was attached or
|
||||||
* {@code null} was attached
|
* {@code null} was attached
|
||||||
@ -161,7 +165,9 @@ public interface ChannelHandlerContext {
|
|||||||
* specific to the {@link ChannelHandler} which is associated with this
|
* specific to the {@link ChannelHandler} which is associated with this
|
||||||
* context.
|
* context.
|
||||||
* <p>
|
* <p>
|
||||||
* As an alternative, you might want to use a {@link ChannelLocal} variable.
|
* As an alternative, you might want to use a {@link ChannelLocal} variable
|
||||||
|
* or a {@link ConcurrentMap} whose key is {@link ChannelHandlerContext}.
|
||||||
|
* Please refer to {@link ChannelPipelineCoverage} for the detailed examples.
|
||||||
*/
|
*/
|
||||||
void setAttachment(Object attachment);
|
void setAttachment(Object attachment);
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,7 @@ import java.lang.annotation.Inherited;
|
|||||||
import java.lang.annotation.Retention;
|
import java.lang.annotation.Retention;
|
||||||
import java.lang.annotation.RetentionPolicy;
|
import java.lang.annotation.RetentionPolicy;
|
||||||
import java.lang.annotation.Target;
|
import java.lang.annotation.Target;
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Specifies if the same instance of the annotated {@link ChannelHandler} type
|
* Specifies if the same instance of the annotated {@link ChannelHandler} type
|
||||||
@ -62,6 +63,16 @@ import java.lang.annotation.Target;
|
|||||||
* }
|
* }
|
||||||
* </pre>
|
* </pre>
|
||||||
*
|
*
|
||||||
|
* Please note that the handler annotated with {@code "all"} can even be added
|
||||||
|
* to the same pipeline more than once:
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* ChannelPipeline p = ...;
|
||||||
|
* StatelessHandler h = ...;
|
||||||
|
* p.addLast("handler1", h);
|
||||||
|
* p.addLast("handler2", h);
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
* <h3>{@code ChannelPipelineCoverage("one")}</h3>
|
* <h3>{@code ChannelPipelineCoverage("one")}</h3>
|
||||||
*
|
*
|
||||||
* {@code "one"} means you must create a new instance of the annotated handler
|
* {@code "one"} means you must create a new instance of the annotated handler
|
||||||
@ -105,6 +116,91 @@ import java.lang.annotation.Target;
|
|||||||
* }
|
* }
|
||||||
* </pre>
|
* </pre>
|
||||||
*
|
*
|
||||||
|
* <h3>Writing a stateful handler with the {@code "all"} coverage</h3>
|
||||||
|
*
|
||||||
|
* Although it's recommended to write a stateful handler with the {@code "one"}
|
||||||
|
* coverage, you might sometimes want to write it with the {@code "all"}
|
||||||
|
* coverage for some reason. In such a case, you need to use either
|
||||||
|
* {@link ChannelHandlerContext#setAttachment(Object) ChannelHandlerContext.attachment}
|
||||||
|
* property or a {@link ConcurrentMap} whose key is {@link ChannelHandlerContext}.
|
||||||
|
* <p>
|
||||||
|
* The following is an example that uses the context attachment:
|
||||||
|
* <pre>
|
||||||
|
* public class StatefulHandler extends SimpleChannelHandler {
|
||||||
|
*
|
||||||
|
* public void channelOpen(ChannelHandlerContext ctx, ChannelStateEvent e) {
|
||||||
|
* // Initialize the message ID counter.
|
||||||
|
* // Please note that the attachment (the counter in this case) will be
|
||||||
|
* // dereferenced and marked for garbage collection automatically on
|
||||||
|
* // disconnection.
|
||||||
|
* <strong>ctx.setAttachment(Integer.valueOf(0));</strong>
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
|
||||||
|
* // Fetch the current message ID.
|
||||||
|
* <strong>int messageId = ((Integer) ctx.getAttachment()).intValue();</strong>
|
||||||
|
*
|
||||||
|
* // Prepend a message ID and length field to the message.
|
||||||
|
* ChannelBuffer body = (ChannelBuffer) e.getMessage();
|
||||||
|
* ChannelBuffer header = ChannelBuffers.buffer(8);
|
||||||
|
* header.writeInt(messageId);
|
||||||
|
* header.writeInt(body.readableBytes());
|
||||||
|
*
|
||||||
|
* // Update the stateful property.
|
||||||
|
* <strong>ctx.setAttachment(Integer.valueOf(messageId + 1));</strong>
|
||||||
|
*
|
||||||
|
* // Create a message prepended with the header and send a new event.
|
||||||
|
* ChannelBuffer message = ChannelBuffers.wrappedBuffer(header, body);
|
||||||
|
* Channels.fireMessageReceived(ctx, message, e.getRemoteAddress());
|
||||||
|
* }
|
||||||
|
* ...
|
||||||
|
* }
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
|
* and here's another example that uses a map:
|
||||||
|
* <pre>
|
||||||
|
* public class StatefulHandler extends SimpleChannelHandler {
|
||||||
|
*
|
||||||
|
* <strong>private final ConcurrentMap<ChannelHandlerContext, Integer> messageIds =
|
||||||
|
* new ConcurrentHashMap<ChannelHandlerContext, Integer>();</strong>
|
||||||
|
*
|
||||||
|
* public void channelOpen(ChannelHandlerContext ctx, ChannelStateEvent e) {
|
||||||
|
* // Initialize the message ID counter.
|
||||||
|
* <strong>messageIds.put(ctx, Integer.valueOf(0));</strong>
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* public void channelClosed(ChannelHandlerContext ctx, ChannelStateEvent e) {
|
||||||
|
* // Remove the message ID counter from the map.
|
||||||
|
* // Please note that the context attachment does not need this step.
|
||||||
|
* <strong>messageIds.remove(ctx);</strong>
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
|
||||||
|
* // Fetch the current message ID.
|
||||||
|
* <strong>int messageId = messageIds.get(ctx).intValue();</strong>
|
||||||
|
*
|
||||||
|
* // Prepend a message ID and length field to the message.
|
||||||
|
* ChannelBuffer body = (ChannelBuffer) e.getMessage();
|
||||||
|
* ChannelBuffer header = ChannelBuffers.buffer(8);
|
||||||
|
* header.writeInt(messageId);
|
||||||
|
* header.writeInt(body.readableBytes());
|
||||||
|
*
|
||||||
|
* // Update the stateful property.
|
||||||
|
* <strong>messageIds.put(ctx, Integer.valueOf(messageId + 1));</strong>
|
||||||
|
*
|
||||||
|
* // Create a message prepended with the header and send a new event.
|
||||||
|
* ChannelBuffer message = ChannelBuffers.wrappedBuffer(header, body);
|
||||||
|
* Channels.fireMessageReceived(ctx, message, e.getRemoteAddress());
|
||||||
|
* }
|
||||||
|
* ...
|
||||||
|
* }
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
|
* Please note that the examples above in this section assume that the handlers
|
||||||
|
* are added before the {@code channelOpen} event and removed after the
|
||||||
|
* {@code channelClosed} event. The initialization and removal of the message
|
||||||
|
* ID property could have been more complicated otherwise.
|
||||||
|
*
|
||||||
* @author The Netty Project (netty-dev@lists.jboss.org)
|
* @author The Netty Project (netty-dev@lists.jboss.org)
|
||||||
* @author Trustin Lee (tlee@redhat.com)
|
* @author Trustin Lee (tlee@redhat.com)
|
||||||
*
|
*
|
||||||
|
Loading…
Reference in New Issue
Block a user