More stuff in the user guide

This commit is contained in:
Trustin Lee 2009-04-10 03:16:37 +00:00
parent 87fd349c66
commit 2e88f7a108
3 changed files with 161 additions and 122 deletions

View File

@ -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>

View File

@ -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>

View File

@ -829,7 +829,7 @@ public class TimeDecoder extends &ReplayingDecoder;&lt;&VoidEnum;&gt; {
</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>