https://github.com/singgel/NettyDemo
Server端:
总的来说,服务端还是比较简单的,自己一共写了三个核心类。分别是
- NettyServerListener:服务启动监听器
- ServerChannelHandlerAdapter:通道适配器,主要用于多线程共享
- RequestDispatcher:请求分排器
https://github.com/singgel/NettyDemo
Client端:
Client我感觉要比Server端要麻烦一点。这里还是先给出核心类吧。
- NettyClient : netty客户端
- ClientChannelHandlerAdapter : 客户端通道适配器
- CustomChannelInitalizer:自定义通道初始化工具
- RPCProxyFactoryBean:RPC通信代理工厂
https://github.com/singgel/NettyDemo
---
netty的hello Word:
https://github.com/singgel/NettyDemo/tree/master/src/test/java/com/singgel
Netty的启动服务程序
package com.singgel.netty;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioServerSocketChannel;
/**
* @Author: hekuangsheng
* @Date: 2018/11/8
*
* Netty的启动服务程序
*/
public class ServerTest {
public static void main(String[] args) throws InterruptedException {
/**
* bossGroup, 父类的事件循环组只是负责连接,获取到连接后交给 workergroup子的事件循环组,
* 参数的获取,业务的处理等工作均是由workergroup这个子事件循环组来完成,一个事件循环组一样
* 可以完成所有的工作,但是Netty推荐的方式是使用两个事件循环组。
*/
EventLoopGroup bossGroup = new NioEventLoopGroup(); //创建父事件循环组
EventLoopGroup workerGroup = new NioEventLoopGroup(); //创建子类的事件循环组
try{
//创建启动服务器的对象
ServerBootstrap serverBootstrap = new ServerBootstrap();
/**
* group方法接收两个参数, 第一个为父时间循环组,第二个参数为子事件循环组
*/
serverBootstrap.group(bossGroup, workerGroup)
.channel(NioServerSocketChannel.class) //bossGroup的通道,只是负责连接
.childHandler(new TestChannelnitializer()); //workerGroup的处理器,
ChannelFuture channelFuture = serverBootstrap.bind(8899).sync(); //绑定端口
channelFuture.channel().closeFuture().sync();
}finally{
bossGroup.shutdownGracefully();
workerGroup.shutdownGracefully();
}
}
}
通道初始化程序
通道的初始化程序主要是为workerGroup添加各种Handler.
package com.singgel.netty;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel;
import io.netty.handler.codec.http.HttpServerCodec;
/**
* @Author: hekuangsheng
* @Date: 2018/11/8
*
* 通道的初始化程序主要是为workerGroup添加各种Handler.
*
* 初始化一个通道,主要用于设置各种Handler
*/
public class TestChannelnitializer extends ChannelInitializer<SocketChannel> {
@Override
protected void initChannel(SocketChannel ch) throws Exception {
ChannelPipeline pipeline = ch.pipeline();
/**
* Handler就相当于Servlet中的过滤器, 请求和响应都会走Handler
* HttpServerCodec: http的编解码器,用于Http请求和相应
*/
pipeline.addLast("httpServerCodec", new HttpServerCodec());
pipeline.addLast("testHttpServerHandler", new TestHttpServerHandler());
}
}
自定义的Handler
package com.singgel.netty;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.codec.http.*;
import io.netty.util.CharsetUtil;
/**
* @Author: hekuangsheng
* @Date: 2018/11/8
*
* 自定义处理器
*/
public class TestHttpServerHandler extends SimpleChannelInboundHandler<HttpObject> {
@Override
protected void channelRead0(ChannelHandlerContext ctx, HttpObject msg) throws Exception {
if(msg instanceof HttpRequest){
//要返回的内容, Channel可以理解为连接,而连接中传输的信息要为ByteBuf
ByteBuf content = Unpooled.copiedBuffer("Hello World", CharsetUtil.UTF_8);
//构造响应
FullHttpResponse response =
new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.OK, content);
//设置头信息的的MIME类型
response.headers().set(HttpHeaderNames.CONTENT_TYPE, "text/plain"); //内容类型
//设置要返回的内容长度
response.headers().set(HttpHeaderNames.CONTENT_LENGTH, content.readableBytes()); //内容长度
//将响应对象返回
ctx.writeAndFlush(response);
}
}
//通道注册成功
@Override
public void channelRegistered(ChannelHandlerContext ctx) throws Exception {
System.out.println("channel register...");
super.channelRegistered(ctx);
}
/**
* 自定义的Handler被添加,也就是在TestChannelnitializer的initChannel方法中,
* pipeline.addLast("testHttpServerHandler", new TestHttpServerHandler());
* 这行代码执行的时候,该方法被触发
*/
@Override
public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
System.out.println("handler added...");
super.handlerAdded(ctx);
}
//通道处于活动状态,即可用状态
@Override
public void channelActive(ChannelHandlerContext ctx) throws Exception {
System.out.println("channel active...");
super.channelActive(ctx);
}
//通道处于不活动状态
@Override
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
System.out.println("channel inactive...");
super.channelInactive(ctx);
}
//通道取消注册
@Override
public void channelUnregistered(ChannelHandlerContext ctx) throws Exception {
System.out.println("channel unregister...");
super.channelUnregistered(ctx);
}
}
首先启动ServerTest的主函数,然后再网页中输入: http://localhost:88899,就会出现你希望看到的东西(前提是代码要写对)