More stuff in the user guide
This commit is contained in:
parent
87fd349c66
commit
2e88f7a108
@ -181,16 +181,56 @@
|
|||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<title>Miscellaneous Constructs</title>
|
<title>Advanced Components for More Rapid Development</title>
|
||||||
<para>
|
<para>
|
||||||
|
On top of the core components mentioned above, that already enable the
|
||||||
|
implementation of all types of network applications, Netty provides a set
|
||||||
|
of advanced features to accelerate the development pace even more.
|
||||||
</para>
|
</para>
|
||||||
</section>
|
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<title>A View from the Feature Set Standpoint</title>
|
<title>Codec framework</title>
|
||||||
<para>
|
<para>
|
||||||
|
As demonstrated in <xref linkend="start.pojo"/>, it is always a good
|
||||||
|
idea to separate a protocol codec from a business logic. However, there
|
||||||
|
are some complications when implementing this idea from scratch. You
|
||||||
|
have to deal with the fragmentation of messages. Some protocols are a
|
||||||
|
multi-layered protocol built on top of other lower level protocol. Some
|
||||||
|
are too complicated to be implemented as a single state machine.
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
Consequently, a good network application framework should provide an
|
||||||
|
extensible, reusable, unit-testable, and multi-layered codec framework
|
||||||
|
that generates maintainable user codec.
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
Netty provides a number of fundamental and advanced codecs built on top
|
||||||
|
of the core to address most issues you will encounter when you write a
|
||||||
|
protocol codec regardless if it is simple or not, binary or text -
|
||||||
|
simply whatever.
|
||||||
|
</para>
|
||||||
|
</section>
|
||||||
|
|
||||||
</para>
|
<section>
|
||||||
|
<title>SSL / TLS Support</title>
|
||||||
|
<para>
|
||||||
|
|
||||||
|
</para>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<title>HTTP Implementation</title>
|
||||||
|
<para>
|
||||||
|
|
||||||
|
</para>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<title>Google Protocol Buffer Integration</title>
|
||||||
|
<para>
|
||||||
|
|
||||||
|
</para>
|
||||||
|
</section>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
%CustomDTD;
|
%CustomDTD;
|
||||||
]>
|
]>
|
||||||
<chapter id="buffer">
|
<chapter id="buffer">
|
||||||
<title>ChannelBuffer</title>
|
<title>ChannelBuffer - Why and How</title>
|
||||||
<para>
|
<para>
|
||||||
As mentioned in <xref linkend="architecture"/>, Netty uses its own buffer
|
As mentioned in <xref linkend="architecture"/>, Netty uses its own buffer
|
||||||
API instead of NIO <classname>ByteBuffer</classname> to represent a sequence
|
API instead of NIO <classname>ByteBuffer</classname> to represent a sequence
|
||||||
@ -13,108 +13,104 @@
|
|||||||
augment its behavior at all. Netty's new buffer type, &ChannelBuffer; has
|
augment its behavior at all. Netty's new buffer type, &ChannelBuffer; has
|
||||||
been designed from ground up to address the problems of
|
been designed from ground up to address the problems of
|
||||||
<classname>ByteBuffer</classname> and to meet the daily needs of network
|
<classname>ByteBuffer</classname> and to meet the daily needs of network
|
||||||
application developers.
|
application developers. In this chapter, we will overview the features of
|
||||||
|
the new buffer API to explain what exactly it is good for.
|
||||||
</para>
|
</para>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<title>The Advantages of Using New Buffer API</title>
|
<title>Extensibility</title>
|
||||||
<para>
|
<para>
|
||||||
|
&ChannelBuffer; has rich set of operations optimized for rapid protocol
|
||||||
|
implementation. For example, &ChannelBuffer; provides various operations
|
||||||
|
for accessing unsigned values and strings and searching for certain byte
|
||||||
|
sequence in a buffer. You can also extend or wrap existing buffer type
|
||||||
|
to add convenient accessors. The custom buffer type still implements
|
||||||
|
&ChannelBuffer; interface rather than introducing an incompatible type.
|
||||||
</para>
|
</para>
|
||||||
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<title>Extensibility</title>
|
<title>Transparent Zero Copy</title>
|
||||||
<para>
|
<para>
|
||||||
&ChannelBuffer; has rich set of operations optimized for rapid protocol
|
To lift up the performance of a network application to the extreme,
|
||||||
implementation. For example, &ChannelBuffer; provides various operations
|
you need to reduce the number of memory copy operation. You might
|
||||||
for accessing unsigned values and strings and searching for certain byte
|
have a set of buffers that could be sliced and combined to compose
|
||||||
sequence in a buffer. You can also extend or wrap existing buffer type
|
a whole message. Netty provides a composite buffer which allows you
|
||||||
to add convenient accessors. The custom buffer type still implements
|
to create a new buffer from the arbitrary number of existing buffers
|
||||||
&ChannelBuffer; interface rather than introducing an incompatible type.
|
with no memory copy. For example, a message could be composed of two
|
||||||
</para>
|
parts; header and body. In a modularized application, the two parts
|
||||||
</section>
|
could be produced by different modules and assembled later when the
|
||||||
|
message is sent out.
|
||||||
<section>
|
</para>
|
||||||
<title>Transparent Zero Copy</title>
|
<programlisting>+--------+----------+
|
||||||
<para>
|
|
||||||
To lift up the performance of a network application to the extreme,
|
|
||||||
you need to reduce the number of memory copy operation. You might
|
|
||||||
have a set of buffers that could be sliced and combined to compose
|
|
||||||
a whole message. Netty provides a composite buffer which allows you
|
|
||||||
to create a new buffer from the arbitrary number of existing buffers
|
|
||||||
with no memory copy. For example, a message could be composed of two
|
|
||||||
parts; header and body. In a modularized application, the two parts
|
|
||||||
could be produced by different modules and assembled later when the
|
|
||||||
message is sent out.
|
|
||||||
</para>
|
|
||||||
<programlisting>+--------+----------+
|
|
||||||
| header | body |
|
| header | body |
|
||||||
+--------+----------+</programlisting>
|
+--------+----------+</programlisting>
|
||||||
<para>
|
<para>
|
||||||
If <classname>ByteBuffer</classname> were used, you would have to
|
If <classname>ByteBuffer</classname> were used, you would have to
|
||||||
create a new big buffer and copy the two parts into the new
|
create a new big buffer and copy the two parts into the new
|
||||||
buffer. Alternatively, you can perform a gathering write operation
|
buffer. Alternatively, you can perform a gathering write operation
|
||||||
in NIO, but it restricts you to represent the composite of buffers
|
in NIO, but it restricts you to represent the composite of buffers
|
||||||
as an array of <classname>ByteBuffer</classname>s rather than a
|
as an array of <classname>ByteBuffer</classname>s rather than a
|
||||||
single <classname>ByteBuffer</classname>, breaking the abstraction and
|
single <classname>ByteBuffer</classname>, breaking the abstraction and
|
||||||
introducing complicated state management. Moreover, it's of no use if
|
introducing complicated state management. Moreover, it's of no use if
|
||||||
you are not going to read or write from an NIO channel.
|
you are not going to read or write from an NIO channel.
|
||||||
</para>
|
</para>
|
||||||
<programlisting>ByteBuffer[]<co id="example.buffer1.co1"/> message = new ByteBuffer[] { header, body };</programlisting>
|
<programlisting>ByteBuffer[]<co id="example.buffer1.co1"/> message = new ByteBuffer[] { header, body };</programlisting>
|
||||||
<calloutlist>
|
<calloutlist>
|
||||||
<callout arearefs="example.buffer1.co1">
|
<callout arearefs="example.buffer1.co1">
|
||||||
<para>
|
<para>
|
||||||
The composite is not a <classname>ByteBuffer</classname> anymore.
|
The composite is not a <classname>ByteBuffer</classname> anymore.
|
||||||
</para>
|
</para>
|
||||||
</callout>
|
</callout>
|
||||||
</calloutlist>
|
</calloutlist>
|
||||||
<para>
|
<para>
|
||||||
By contrast, &ChannelBuffer; does not have such caveats because it is
|
By contrast, &ChannelBuffer; does not have such caveats because it is
|
||||||
fully extensible and has a built-in composite buffer type.
|
fully extensible and has a built-in composite buffer type.
|
||||||
</para>
|
</para>
|
||||||
<programlisting>&ChannelBuffer;<co id="example.buffer2.co1"/> message = &Channels;.wrappedBuffer(header, body);
|
<programlisting>&ChannelBuffer;<co id="example.buffer2.co1"/> message = &Channels;.wrappedBuffer(header, body);
|
||||||
&ChannelBuffer;<co id="example.buffer2.co2"/> messageWithFooter = &Channels;.wrappedBuffer(message, footer);
|
&ChannelBuffer;<co id="example.buffer2.co2"/> messageWithFooter = &Channels;.wrappedBuffer(message, footer);
|
||||||
messageWithFooter.getUnsignedInt(
|
messageWithFooter.getUnsignedInt(
|
||||||
messageWithFooter.readableBytes() - footer.readableBytes() - 1<co id="example.buffer2.co3"/>);</programlisting>
|
messageWithFooter.readableBytes() - footer.readableBytes() - 1<co id="example.buffer2.co3"/>);</programlisting>
|
||||||
<calloutlist>
|
<calloutlist>
|
||||||
<callout arearefs="example.buffer2.co1">
|
<callout arearefs="example.buffer2.co1">
|
||||||
<para>
|
<para>
|
||||||
The composite is always a &ChannelBuffer;. It is completely
|
The composite is always a &ChannelBuffer;. It is completely
|
||||||
transparent.
|
transparent.
|
||||||
</para>
|
</para>
|
||||||
</callout>
|
</callout>
|
||||||
<callout arearefs="example.buffer2.co2">
|
<callout arearefs="example.buffer2.co2">
|
||||||
<para>
|
<para>
|
||||||
You can even create a composite by mixing an existing composite and
|
You can even create a composite by mixing an existing composite and
|
||||||
an ordinary buffer.
|
an ordinary buffer.
|
||||||
</para>
|
</para>
|
||||||
</callout>
|
</callout>
|
||||||
<callout arearefs="example.buffer2.co3">
|
<callout arearefs="example.buffer2.co3">
|
||||||
<para>
|
<para>
|
||||||
Because the composite is still a &ChannelBuffer;, you can access
|
Because the composite is still a &ChannelBuffer;, you can access
|
||||||
its content easily, and the accessor method will behave just like
|
its content easily, and the accessor method will behave just like
|
||||||
it's a single buffer even if the region you want to access spans
|
it's a single buffer even if the region you want to access spans
|
||||||
over multiple components. The unsigned integer being read here is
|
over multiple components. The unsigned integer being read here is
|
||||||
located across body and footer.
|
located across body and footer.
|
||||||
</para>
|
</para>
|
||||||
</callout>
|
</callout>
|
||||||
</calloutlist>
|
</calloutlist>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<title>Automatic Capacity Expansion</title>
|
<title>Automatic Capacity Expansion</title>
|
||||||
<para>
|
<para>
|
||||||
Many protocols define variable length messages, which means there's no
|
Many protocols define variable length messages, which means there's no
|
||||||
way to determine the length of a message until you construct the
|
way to determine the length of a message until you construct the
|
||||||
message or it is difficult and inconvenient to calculate the length
|
message or it is difficult and inconvenient to calculate the length
|
||||||
precisely. It is just like when you build a <classname>String</classname>.
|
precisely. It is just like when you build a <classname>String</classname>.
|
||||||
You often estimate the length of the resulting string and let
|
You often estimate the length of the resulting string and let
|
||||||
<classname>StringBuffer</classname> to expand the capacity of its
|
<classname>StringBuffer</classname> to expand the capacity of its
|
||||||
internal buffer on demand. Netty allows you to do the same via
|
internal buffer on demand. Netty allows you to do the same via
|
||||||
a <firstterm>dynamic</firstterm> buffer which is created by the
|
a <firstterm>dynamic</firstterm> buffer which is created by the
|
||||||
&Channels;<literal>.</literal><methodname>dynamicBuffer()</methodname>
|
&Channels;<literal>.</literal><methodname>dynamicBuffer()</methodname>
|
||||||
method.
|
method.
|
||||||
</para>
|
</para>
|
||||||
<programlisting>&ChannelBuffer;<co id="example.buffer3.co1"/> dynamicBuffer = &Channels;.dynamicBuffer(4);
|
<programlisting>&ChannelBuffer;<co id="example.buffer3.co1"/> dynamicBuffer = &Channels;.dynamicBuffer(4);
|
||||||
dynamicBuffer.writeByte('1');<co id="example.buffer3.co2"/>
|
dynamicBuffer.writeByte('1');<co id="example.buffer3.co2"/>
|
||||||
dynamicBuffer.writeByte('2');
|
dynamicBuffer.writeByte('2');
|
||||||
dynamicBuffer.writeByte('3');
|
dynamicBuffer.writeByte('3');
|
||||||
@ -122,33 +118,36 @@ dynamicBuffer.writeByte('4');
|
|||||||
dynamicBuffer.writeByte('5');<co id="example.buffer3.co3"/>
|
dynamicBuffer.writeByte('5');<co id="example.buffer3.co3"/>
|
||||||
dynamicBuffer.writeByte('6');
|
dynamicBuffer.writeByte('6');
|
||||||
dynamicBuffer.writeByte('7');</programlisting>
|
dynamicBuffer.writeByte('7');</programlisting>
|
||||||
<calloutlist>
|
<calloutlist>
|
||||||
<callout arearefs="example.buffer3.co1">
|
<callout arearefs="example.buffer3.co1">
|
||||||
<para>
|
<para>
|
||||||
A new dynamic buffer is created. Internally, the actual buffer
|
A new dynamic buffer is created. Internally, the actual buffer
|
||||||
is created lazily to avoid potentially wasted memory space.
|
is created lazily to avoid potentially wasted memory space.
|
||||||
</para>
|
</para>
|
||||||
</callout>
|
</callout>
|
||||||
<callout arearefs="example.buffer3.co3">
|
<callout arearefs="example.buffer3.co3">
|
||||||
<para>
|
<para>
|
||||||
When the first write attempt is made, the internal buffer is created
|
When the first write attempt is made, the internal buffer is created
|
||||||
with the specified initial capacity (4).
|
with the specified initial capacity (4).
|
||||||
</para>
|
</para>
|
||||||
</callout>
|
</callout>
|
||||||
<callout arearefs="example.buffer2.co3">
|
<callout arearefs="example.buffer3.co3">
|
||||||
<para>
|
<para>
|
||||||
When the number of written bytes exceeds the initial capacity (4),
|
When the number of written bytes exceeds the initial capacity (4),
|
||||||
the internal buffer is reallocated automatically with a larger
|
the internal buffer is reallocated automatically with a larger
|
||||||
capacity.
|
capacity.
|
||||||
</para>
|
</para>
|
||||||
</callout>
|
</callout>
|
||||||
</calloutlist>
|
</calloutlist>
|
||||||
</section>
|
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<title>Summary</title>
|
<title>Documentation in progress</title>
|
||||||
<para>
|
<para>
|
||||||
|
This user guide is still under construction and waiting for your feed
|
||||||
|
back. Any idea to improve the documentation is more than appreciated.
|
||||||
|
Please join us in the <ulink url="&Community;">community</ulink> now to
|
||||||
|
share your idea!
|
||||||
</para>
|
</para>
|
||||||
</section>
|
</section>
|
||||||
</chapter>
|
</chapter>
|
||||||
|
@ -829,7 +829,7 @@ public class TimeDecoder extends &ReplayingDecoder;<&VoidEnum;> {
|
|||||||
</section>
|
</section>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section id="start.pojo">
|
||||||
<title>
|
<title>
|
||||||
Speaking in POJO instead of ChannelBuffer
|
Speaking in POJO instead of ChannelBuffer
|
||||||
</title>
|
</title>
|
||||||
|
Loading…
Reference in New Issue
Block a user