实例:Netty 基于Http协议下的数据传输Demo
2015-04-07 13:27
381 查看
Http/Https协议是最重要最常用到的协议之一,Netty提供了一些了的Handler来处理Http协议下的编码工作。下面就介绍一个Netty实例:
1.通过HttpClient发送Protobuf类型数据到服务端
2.服务端Netty负责把接收到的Http请求中的数据再发送到客户端。
3.其中Netty对发送的数据量没有限制,因为Http发送的message往往是由一系列infragment构成,Netty可以把接收到的http请求片段信息整合(Aggregator)到一起,最终得到一个FullHttpRequest。
Client端:
[java]
view plaincopy
package NettyDemo.echo.client;
import java.io.IOException;
import org.apache.http.HttpEntity;
import org.apache.http.HttpHost;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.conn.params.ConnRoutePNames;
import org.apache.http.entity.ByteArrayEntity;
import org.apache.http.impl.client.DefaultHttpClient;
import NettyDemo.echo.protocal.AddressBookProtos;
import NettyDemo.echo.server.HttpProtobufServer;
public class HttpClientDemo {
public static void main(String[] args) throws ClientProtocolException,
IOException {
DefaultHttpClient httpclient = new DefaultHttpClient();
HttpHost proxy = new HttpHost(HttpProtobufServer.IP,
HttpProtobufServer.PORT);
httpclient.getParams().setParameter(ConnRoutePNames.DEFAULT_PROXY,
proxy);
HttpPost httppost = new HttpPost("");
AddressBookProtos.AddressBook.Builder address = AddressBookProtos.AddressBook
.newBuilder();
for (int i = 0; i < 20000; i++) {
AddressBookProtos.Person.Builder person = AddressBookProtos.Person
.newBuilder();
person.setName("HeavenWang");
person.setId(i);
person.setEmail("wanghouda@126.com");
address.addPerson(person);
}
ByteArrayEntity entity = new ByteArrayEntity(address.build().toByteArray());
httppost.setEntity(entity);
HttpResponse response = httpclient.execute(httppost);
HttpEntity receiveEntity = response.getEntity();
System.out.println("----------------------------------------");
System.out.println(response.getStatusLine());
if (receiveEntity != null) {
System.out.println("Response content length: " + receiveEntity.getContentLength());
}
System.out.println("success");
}
}
服务器端NettyService:
[java]
view plaincopy
<span style="font-size:12px;">package NettyDemo.echo.server;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpRequestDecoder;
import io.netty.handler.codec.http.HttpResponseEncoder;
import io.netty.handler.codec.http.HttpServerCodec;
import NettyDemo.echo.handler.HttpProtobufServerHandler;
public class HttpProtobufServer {
public static final String IP = "127.0.0.1";
public static final int PORT = 8080;
//private static MessageLite lite=AddressBookProtos.AddressBook.getDefaultInstance();
/**用于分配处理业务线程的线程组个数 */
protected static final int BIZGROUPSIZE = Runtime.getRuntime().availableProcessors()*2; //默认
/** 业务出现线程大小*/
protected static final int BIZTHREADSIZE = 4;
private static final EventLoopGroup bossGroup = new NioEventLoopGroup(BIZGROUPSIZE);
private static final EventLoopGroup workerGroup = new NioEventLoopGroup(BIZTHREADSIZE);
protected static void run() throws Exception {
ServerBootstrap b = new ServerBootstrap();
b.group(bossGroup, workerGroup);
b.channel(NioServerSocketChannel.class);
b.childHandler(new ChannelInitializer<SocketChannel>() {
@Override
public void initChannel(SocketChannel ch) throws Exception {
ChannelPipeline pipeline = ch.pipeline();
pipeline.addLast("decoder", new HttpRequestDecoder());
/**usually we receive http message infragment,if we want full http message,
* we should bundle HttpObjectAggregator and we can get FullHttpRequest。
* 我们通常接收到的是一个http片段,如果要想完整接受一次请求的所有数据,我们需要绑定HttpObjectAggregator,然后我们
* 就可以收到一个FullHttpRequest-是一个完整的请求信息。
**/
pipeline.addLast("servercodec",new HttpServerCodec());
pipeline.addLast("aggegator",new HttpObjectAggregator(1024*1024*64));//定义缓冲数据量
pipeline.addLast(new HttpProtobufServerHandler());
pipeline.addLast("responseencoder",new HttpResponseEncoder());
}
});
b.bind(IP, PORT).sync();
System.out.println("TCP服务器已启动");
}
protected static void shutdown() {
workerGroup.shutdownGracefully();
bossGroup.shutdownGracefully();
}
public static void main(String[] args) throws Exception {
System.out.println("开始启动http服务器...");
HttpProtobufServer.run();
// TcpServer.shutdown();
}
}
</span>
Handler:
[java]
view plaincopy
package NettyDemo.echo.handler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.codec.http.DefaultFullHttpRequest;
import io.netty.handler.codec.http.DefaultFullHttpResponse;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.HttpVersion;
public class HttpProtobufServerHandler extends SimpleChannelInboundHandler<Object>{
@Override
protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {
DefaultFullHttpRequest request=(DefaultFullHttpRequest)msg;
DefaultFullHttpResponse response=new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.ACCEPTED, request.content());
ctx.writeAndFlush(response);
}
}
附:FullHttpRequest构成,因此一个FullHttpRequest会包含请求message的所有片段。
1.通过HttpClient发送Protobuf类型数据到服务端
2.服务端Netty负责把接收到的Http请求中的数据再发送到客户端。
3.其中Netty对发送的数据量没有限制,因为Http发送的message往往是由一系列infragment构成,Netty可以把接收到的http请求片段信息整合(Aggregator)到一起,最终得到一个FullHttpRequest。
Client端:
[java]
view plaincopy
package NettyDemo.echo.client;
import java.io.IOException;
import org.apache.http.HttpEntity;
import org.apache.http.HttpHost;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.conn.params.ConnRoutePNames;
import org.apache.http.entity.ByteArrayEntity;
import org.apache.http.impl.client.DefaultHttpClient;
import NettyDemo.echo.protocal.AddressBookProtos;
import NettyDemo.echo.server.HttpProtobufServer;
public class HttpClientDemo {
public static void main(String[] args) throws ClientProtocolException,
IOException {
DefaultHttpClient httpclient = new DefaultHttpClient();
HttpHost proxy = new HttpHost(HttpProtobufServer.IP,
HttpProtobufServer.PORT);
httpclient.getParams().setParameter(ConnRoutePNames.DEFAULT_PROXY,
proxy);
HttpPost httppost = new HttpPost("");
AddressBookProtos.AddressBook.Builder address = AddressBookProtos.AddressBook
.newBuilder();
for (int i = 0; i < 20000; i++) {
AddressBookProtos.Person.Builder person = AddressBookProtos.Person
.newBuilder();
person.setName("HeavenWang");
person.setId(i);
person.setEmail("wanghouda@126.com");
address.addPerson(person);
}
ByteArrayEntity entity = new ByteArrayEntity(address.build().toByteArray());
httppost.setEntity(entity);
HttpResponse response = httpclient.execute(httppost);
HttpEntity receiveEntity = response.getEntity();
System.out.println("----------------------------------------");
System.out.println(response.getStatusLine());
if (receiveEntity != null) {
System.out.println("Response content length: " + receiveEntity.getContentLength());
}
System.out.println("success");
}
}
服务器端NettyService:
[java]
view plaincopy
<span style="font-size:12px;">package NettyDemo.echo.server;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpRequestDecoder;
import io.netty.handler.codec.http.HttpResponseEncoder;
import io.netty.handler.codec.http.HttpServerCodec;
import NettyDemo.echo.handler.HttpProtobufServerHandler;
public class HttpProtobufServer {
public static final String IP = "127.0.0.1";
public static final int PORT = 8080;
//private static MessageLite lite=AddressBookProtos.AddressBook.getDefaultInstance();
/**用于分配处理业务线程的线程组个数 */
protected static final int BIZGROUPSIZE = Runtime.getRuntime().availableProcessors()*2; //默认
/** 业务出现线程大小*/
protected static final int BIZTHREADSIZE = 4;
private static final EventLoopGroup bossGroup = new NioEventLoopGroup(BIZGROUPSIZE);
private static final EventLoopGroup workerGroup = new NioEventLoopGroup(BIZTHREADSIZE);
protected static void run() throws Exception {
ServerBootstrap b = new ServerBootstrap();
b.group(bossGroup, workerGroup);
b.channel(NioServerSocketChannel.class);
b.childHandler(new ChannelInitializer<SocketChannel>() {
@Override
public void initChannel(SocketChannel ch) throws Exception {
ChannelPipeline pipeline = ch.pipeline();
pipeline.addLast("decoder", new HttpRequestDecoder());
/**usually we receive http message infragment,if we want full http message,
* we should bundle HttpObjectAggregator and we can get FullHttpRequest。
* 我们通常接收到的是一个http片段,如果要想完整接受一次请求的所有数据,我们需要绑定HttpObjectAggregator,然后我们
* 就可以收到一个FullHttpRequest-是一个完整的请求信息。
**/
pipeline.addLast("servercodec",new HttpServerCodec());
pipeline.addLast("aggegator",new HttpObjectAggregator(1024*1024*64));//定义缓冲数据量
pipeline.addLast(new HttpProtobufServerHandler());
pipeline.addLast("responseencoder",new HttpResponseEncoder());
}
});
b.bind(IP, PORT).sync();
System.out.println("TCP服务器已启动");
}
protected static void shutdown() {
workerGroup.shutdownGracefully();
bossGroup.shutdownGracefully();
}
public static void main(String[] args) throws Exception {
System.out.println("开始启动http服务器...");
HttpProtobufServer.run();
// TcpServer.shutdown();
}
}
</span>
Handler:
[java]
view plaincopy
package NettyDemo.echo.handler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.codec.http.DefaultFullHttpRequest;
import io.netty.handler.codec.http.DefaultFullHttpResponse;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.HttpVersion;
public class HttpProtobufServerHandler extends SimpleChannelInboundHandler<Object>{
@Override
protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {
DefaultFullHttpRequest request=(DefaultFullHttpRequest)msg;
DefaultFullHttpResponse response=new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.ACCEPTED, request.content());
ctx.writeAndFlush(response);
}
}
附:FullHttpRequest构成,因此一个FullHttpRequest会包含请求message的所有片段。
相关文章推荐
- 实例:Netty 基于Http协议下的数据传输Demo
- 基于jquery异步传输json数据格式实例代码
- 基于Netty-Socket-io的无直接调用式的数据传输
- android网络操作---数据传输(基于HTTP协议)
- UDT协议-基于UDP的可靠数据传输协议
- 基于嵌入式TCP/IP的实时数据传输
- 基于ARM高精度数据采集及网络传输方案
- 基于UDP协议实现可靠的数据传输
- 基于ARM的图像采集和数据传输系统
- tcp 数据传输实例测试
- 分享:基于UDP协议实现可靠的数据传输
- 基于JRTPLIB库的RTP数据传输设计文档(2)
- UDP数据传输实例
- 基于JRTPLIB库的RTP数据传输设计文档
- 基于蓝牙的数据传输系统的设计
- 基于HFC系统的高效数据传输协议
- UDT协议-基于UDP的可靠数据传输协议
- UDT协议-基于UDP的可靠数据传输协议
- 基于TCP、UDP网络传输的实例分析