Jetty 中的慢速传输,在特定缓冲区大小下使用分块传输编码
我正在调查 Jetty 6.1.26 的性能问题。Jetty似乎使用 ,并且根据所使用的缓冲区大小,在本地传输时这可能会非常慢。Transfer-Encoding: chunked
我创建了一个小型的 Jetty 测试应用程序,其中包含一个演示该问题的 servlet。
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.mortbay.jetty.Server;
import org.mortbay.jetty.nio.SelectChannelConnector;
import org.mortbay.jetty.servlet.Context;
public class TestServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
final int bufferSize = 65536;
resp.setBufferSize(bufferSize);
OutputStream outStream = resp.getOutputStream();
FileInputStream stream = null;
try {
stream = new FileInputStream(new File("test.data"));
int bytesRead;
byte[] buffer = new byte[bufferSize];
while( (bytesRead = stream.read(buffer, 0, bufferSize)) > 0 ) {
outStream.write(buffer, 0, bytesRead);
outStream.flush();
}
} finally {
if( stream != null )
stream.close();
outStream.close();
}
}
public static void main(String[] args) throws Exception {
Server server = new Server();
SelectChannelConnector ret = new SelectChannelConnector();
ret.setLowResourceMaxIdleTime(10000);
ret.setAcceptQueueSize(128);
ret.setResolveNames(false);
ret.setUseDirectBuffers(false);
ret.setHost("0.0.0.0");
ret.setPort(8080);
server.addConnector(ret);
Context context = new Context();
context.setDisplayName("WebAppsContext");
context.setContextPath("/");
server.addHandler(context);
context.addServlet(TestServlet.class, "/test");
server.start();
}
}
在我的实验中,我使用的是一个 128MB 的测试文件,servlet 将该文件返回到客户端,客户端使用 localhost 进行连接。使用用Java编写的简单测试客户端下载此数据需要3.8秒,这是非常慢的(是的,它是33MB / s,这听起来并不慢,除了这是纯粹的本地并且输入文件被缓存;它应该快得多)。URLConnection
现在,它变得奇怪了。如果我使用wget下载数据,这是一个HTTP / 1.0客户端,因此不支持分块传输编码,它只需要0.1秒。这是一个更好的数字。
现在,当我更改为 4096 时,Java 客户端需要 0.3 秒。bufferSize
如果我完全删除对的调用(它似乎使用24KB的块大小),Java客户端现在需要7.1秒,wget突然变得同样慢!resp.setBufferSize
请注意,我绝不是Jetty的专家。我在诊断Hadoop 0.20.203.0中的性能问题时偶然发现了这个问题,该问题具有减少任务洗牌功能,该问题使用Jetty以类似于简化的示例代码的方式传输文件,缓冲区大小为64KB。
这个问题在我们的Linux(Debian)服务器和我的Windows机器上都重现,并且在Java 1.6和1.7上都重现,因此它似乎完全依赖于Jetty。
有没有人知道是什么原因导致这种情况,如果有什么我可以做的?