| | Contents: | | |
| Related content: | | |
| Subscriptions: | | |
| Build your own Servlet-based Web server, with nonblocking I/O
Taylor Cowan (taylor_cowan@yahoo.com)
Senior Software Systems Engineer, Travelocity
03 Feb 2004 Think it's impossible to combine NIO and the Servlet API? Think again. In this article, Java developer Taylor Cowan shows you how to apply the producer/consumer model to consumer nonblocking I/O, thus easing the Servlet API into a whole new compatibility with NIO. In the process, you'll see what it takes to build an actual Servlet-based Web server that implements NIO; and you'll find out how that server stacks up against a standard Java I/O server (Tomcat 5.0) in an enterprise environment. NIO was among the most celebrated (if not the most glamorous) additions to the Java platform with JDK 1.4. Many articles followed, explaining the basics of NIO and how to leverage the benefits of nonblocking channels. One thing missing through all this, however, was an adequate demonstration of just how NIO might improve the scalability of a J2EE Web tier. For the enterprise developer this information is particularly relevant, because implementing NIO isn't as simple as changing a few import statements to a new I/O package. First, the Servlet API assumes blocking I/O semantics, so it can't take advantage of nonblocking I/O by default. Second, threads aren't the resource hogs they were in JDK 1.0, so using fewer threads does not necessarily indicate a server's ability to handle more clients. In this article, you'll learn how to work around the Servlet API's aversion to nonblocking I/O to create a Servlet-based Web server that implements NIO. We'll then see how this server scales against a standard I/O server (Tomcat 5.0) in a multiplexed Web server environment. In keeping with the realities of life in the enterprise, we'll focus on how NIO compares to standard I/O when an exponentially increasing number of clients retain their socket connections. Note that this article is for Java developers familiar with the basics of I/O programming on the Java platform. See the Resources section for an introduction to nonblocking I/O. Threads on a budget
Threads have a well-earned reputation for being expensive. In the early days of the Java platform (JDK 1.0), thread overhead was such a burden that developers were forced to custom build solutions. One common workaround was to use a pool of threads created at VM startup, rather than creating each new thread on demand. Despite recent improvements to thread performance at the VM layer, standard I/O still requires that a unique thread be allocated to handle each new open socket. This works well enough in the short term, but standard I/O falls short when the number of threads increases beyond 1K. The CPU simply becomes overburdened by context switching between threads. With the introduction of NIO in JDK 1.4, enterprise developers finally have a built-in solution to the thread-per-use model: Multiplexed I/O allows a growing number of users to be served by a fixed number of threads. Multiplexing refers to the sending of multiple signals, or streams, simultaneously over a single carrier. A day-to-day example of multiplexing occurs when we use a cell phone. Wireless frequencies are a scarce resource, so wireless providers use multiplexing to send multiple calls over a single frequency. In one example, calls are divided into segments that are given a very short time duration and reassembled at the receiving end. This is called time-division multiplexing, or TDM. Within NIO the receiving end is comparable to a "selector" (see java.nio.channels.Selector ). Instead of calls, the selector handles multiple open sockets. Just as in TDM, the selector reassembles segments of data being written from multiple clients. This allows the server to manage multiple clients with a single thread. The Servlet API and NIO
Nonblocking reads and writes are essential to NIO, but they don't come trouble free. A nonblocking read makes no guarantee to the caller besides the fact that it won't block. The client or server application may read the complete message, a partial message, or nothing at all. On the other hand, a nonblocking read might read more than enough, forcing an overhead buffer for the next call. And, finally, unlike streams a zero byte read does not indicate that the message has been fully received. These factors make it impossible to implement even a simple readline method without polling. All servlet containers must provide a readline method on their input streams. As a result, many developers have given up on building a Servlet-based Web application server that implements NIO. Fortunately, there is a solution; one that combines the power of the Servlet API and the multiplexed I/O of NIO. In the sections that follow, you'll learn how to apply the producer/consumer model to consumer nonblocking I/O, using the java.io.PipedInput and PipedOutputStream classes. As the nonblocking channel is read, it is written into a pipe that is being consumed by a second thread. Note that this decomposition maps threads differently from most Java-based client-server apps. Here, we have a thread solely responsible for processing a nonblocking channel (the producer) and another thread solely responsible for consuming the data as a stream (the consumer). Pipes also alleviate the nonblocking I/O problem for application servers, because servlets will assume blocking semantics as they consume the I/O. The example server
Our example server demonstrates the producer/consumer solution to the incompatibility of the Servlet API and NIO. The server is similar enough to the Servlet API to provide proof of concept for a full-fledged NIO-based application server, and it has been written specifically to measure the performance of NIO against standard Java I/O. It handles simple HTTP get requests and supports keep-alive connections from clients. This is important because multiplexing I/O only proves beneficial when the server is required to handle a large number of open socket connections. The server is divided into two packages, org.sse.server and org.sse.http . The server package holds classes that provide primary server functionality such as receiving new client connections, reading messages, and spawning worker threads to handle requests. The http package supports a subset of the HTTP protocol. A detailed explanation of HTTP is beyond the scope of this article. Download the code examples from the Resources section for implementation details. Now, let's take a look at the most important classes in the org.sse.server package. The Server class
The Server class holds the multiplexer loop, the heart of any NIO-based server. In Listing 1, the call to select() blocks until the server either receives a new client or detects available bytes being written to an open socket. The major difference between this and standard Java I/O is that all data is read within this loop. Normally, a new thread would be given the task of reading bytes from a particular socket. It is actually possible to handle many thousands of clients with a single thread using the NIO selector event-driven approach, although we'll see later that threads still have a role to play. Each call to select() returns a collection of events indicating that a new client is available, new data is ready to read, or a client is ready to receive a response. The server's handleKey() method is only interested in new clients (key.isAcceptable() ) or incoming data (key.isReadable() ). At that point the work is passed off to the ServerEventHandler class. Listing 1. Server.java selector loop
public void listen() {
SelectionKey key = null;
try {
while (true) {
selector.select();
Iterator it = selector.selectedKeys().iterator();
while (it.hasNext()) {
key = (SelectionKey) it.next();
handleKey(key);
it.remove();
}
}
} catch (IOException e) {
key.cancel();
} catch (NullPointerException e) {
// NullPointer at sun.nio.ch.WindowsSelectorImpl, Bug: 4729342
e.printStackTrace();
}
}
| The ServerEventHandler class
The ServerEventHandler class responds to server events. When a new client becomes available it instantiates a new Client object representing the state of that client. Data is read from the channel in a nonblocking fashion and written to the Client object. The ServerEventHandler also maintains a queue of requests. A variable number of worker threads are spawned to process (consume) requests off the queue. In traditional producer/consumer fashion, Queue is written so that threads block when it becomes empty, and are notified when new requests are available. In Listing 2, the remove() method has been overridden to support waiting threads. If the list is empty, the number of waiting threads is incremented and the current thread is blocked. This essentially provides a very simple thread pool. Listing 2. Queue.java
public class Queue extends LinkedList
{
private int waitingThreads = 0;
public synchronized void insert(Object obj)
{
addLast(obj);
notify();
}
public synchronized Object remove()
{
if ( isEmpty() ) {
try { waitingThreads++; wait();}
catch (InterruptedException e) {Thread.interrupted();}
waitingThreads--;
}
return removeFirst();
}
public boolean isEmpty() {
return (size() - waitingThreads <= 0);
}
}
| The number of worker threads is independent of the number of Web clients. Instead of allocating one thread per open socket, we place all requests into a generic queue serviced by a set of RequestHandlerThread instances. Ideally, the number of threads should be tuned based on the number of processors and the length or duration of each request. If requests take a long time by way of resource or processing needs, the perceived quality of service can be improved by adding more threads. Note that this doesn't necessarily improve overall throughput, but it does improve the user's experience. Even under heavy load each thread will be given a slice of processing time. This principle applies equally to servers based on standard Java I/O; however, those servers are limited in that they are required to allocate one thread per open socket connection. NIO servers are relieved of this and therefore can scale to larger numbers of users. The bottom line is that NIO servers still need threads, just not quite as many. Request processing
The Client class serves two purposes. First, it solves the blocking/nonblocking problem by converting the incoming nonblocking I/O into a blocking InputStream consumable by the Servlet API. Second, it manages the request state of a particular client. Because nonblocking channels give no indication when a message has been fully read, we are forced to handle this at the protocol layer. The Client class indicates at any given point in time if it is currently involved in an ongoing request. If it is ready to handle a new request, the write() method enqueues the client for request processing. If it is already engaged in a request it simply transforms the incoming bytes into an InputStream using the PipedInputStream and PipedOutputStream classes. Figure 1 shows the interactions of two threads around a pipe. The main thread writes bytes read from the channel into the pipe. The pipe provides the same data to consumers as an InputStream . Another important feature of the pipe is that it is buffered. If it were not, the main thread could become blocked trying to write to the pipe. Because the main thread is solely responsible for multiplexing between all clients, we cannot afford to allow it to block. Figure 1. PipedInput/OutputStream
After the Client has enqueued itself, it is ready be consumed by a worker thread. The RequestHandlerThread class takes on this role. So far we've seen how the main thread loops continuously, either accepting new clients or reading new I/O. The worker threads loop awaiting new requests. When a client becomes available on the request queue, it is immediately consumed by the first waiting thread blocked on the remove() method. Listing 3. RequestHandlerThread.java
public void run() {
while (true) {
Client client = (Client) myQueue.remove();
try {
for (; ; ) {
HttpRequest req = new HttpRequest(client.clientInputStream,
myServletContext);
HttpResponse res = new HttpResponse(client.key);
defaultServlet.service(req, res);
if (client.notifyRequestDone())
break;
}
} catch (Exception e) {
client.key.cancel();
client.key.selector().wakeup();
}
}
}
| The thread then creates new HttpRequest and HttpResponse instances and invokes the service method of the default servlet. Notice that the HttpRequest is constructed with the clientInputStream property of the Client object. This is the PipedInputStream responsible for converting nonblocking I/O to a blocking stream. From this point on, request processing is similar to what you would expect in the J2EE Servlet API. When the call to the servlet returns, the worker thread checks to see if another request is available from the same client before returning to the pool. Note that the word pool is used lightly here. The thread will in fact attempt another remove() call on the queue and will become blocked until the next available request. Running the example
The example server implements a subset of the HTTP 1.1 protocol. It processes normal HTTP get requests. It takes two command-line arguments. The first one specifies the port number and the second designates the directory where your HTML files reside. After unzipping the files, cd into the project directory and issue the following command, replacing the webroot directory with your own:
java -cp bin org.sse.server.Start 8080
"C:\mywebroot"
| Also note that the server doesn't implement directory listings, so you must specify a valid URL pointing to a file under your webroot. Performance results
The example NIO server was compared to Tomcat 5.0 under heavy load. Tomcat was chosen because it is a 100 percent Java solution based on standard Java I/O. Some advanced app servers are optimized with JNI native code to improve scalability and therefore don't offer a good comparison between standard I/O and NIO. The objective was to determine if NIO gives any considerable performance benefits and under what conditions. Here are the specs: - Tomcat was configured with a maximum thread count of 2000 while the example server was only allowed to run with four worker threads.
- Each server was tested against the same set of simple HTTP
get s consisting of mostly textual content.
- The load tool (Microsoft Web Application Stress Tool) was set to use "keep-alive" sessions, resulting in roughly one socket per user. This in turn results in one thread per user on Tomcat, while the NIO server handles the same load with a constant number of threads.
Figure 2 shows the request-per-second rate under increasing load. At 200 users performance was similar. As the number of users exceeded 600, however, Tomcat's performance began to deteriorate drastically. This is most likely due to the cost of context switching between so many threads. In contrast, the NIO-based server's performance degraded in a linear fashion. Keep in mind that Tomcat must allocate one thread per user, while the NIO server was configured with only four worker threads. Figure 2. Requests per second
Figure 3 provides further indication of NIO's performance. It shows the number of socket-connect errors per minute of operation. Again, Tomcat's performance deteriorated drastically at about 600 users, while the NIO-based server's error rate remained relatively low. Figure 3. Socket-connect errors per second
Conclusion
In this article you've learned that it is indeed possible to write a Servlet-based Web server using NIO, even with its nonblocking features enabled. This is good news for enterprise developers because NIO scales better than standard Java I/O in enterprise environments. Unlike standard Java I/O, NIO can handle many clients with a fixed number of threads. The Servlet-based NIO Web server yields better performance when it comes to handling clients that keep and hold socket connections. Resources - Download the source code used in this article.
- See "Merlin brings nonblocking I/O to the Java platform" (developerWorks, March 2002) for additional insight into NIO semantics.
- The comprehensive developerWorks tutorial, "Getting started with NIO" (developerWorks, July 2003) covers the NIO library in great detail, from the high-level concepts to under-the-hood programming.
- Merlin Hughes's two-part "Turning streams inside out" (developerWorks, July 2002) offers well-crafted engineering solutions to some of the pervasive challenges of Java I/O -- in both the standard and NIO versions.
- Get some background on the trouble with standard Java I/O, with Allen Holub's "Proposal for fixing the Java programming language's threading problems" (developerWorks, October 2000).
- Visit the NIO home page to learn about nonblocking I/O from the source.
- JavaNIO.info is the ideal place to locate resources having to do with NIO.
- For a book-length education on NIO, see the classic work in the field: Ron Hitchens's Java NIO (O'Reilly & Associates, 2002).
- You'll find articles about every aspect of Java programming in the developerWorks Java technology zone.
- Visit the Developer Bookstore for a comprehensive listing of technical books, including hundreds of Java-related titles.
- Also see the Java technology zone tutorials page for a complete listing of free Java-focused tutorials from developerWorks.
Download
| Name |
| |
| Size |
| |
| Download method |
| |
| j-nioserver-source.zip |
| |
|
|
| |
| FTP |
| | | | | | | | | | | | | |
About the author
Taylor Cowan is a software engineer and occasional freelance author specializing in J2EE. He received his Masters Degree in Computer Science from the University of North Texas, as well as a Bachelor of Music in Jazz Arranging. |
|
댓글
댓글 쓰기