ServerBootstrap简介

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了ServerBootstrap简介相关的知识,希望对你有一定的参考价值。

参考技术A ServerBootstrap是一个用来创建服务端Channel的工具类,创建出来的Channel用来接收进来的请求;只用来做面向连接的传输,像TCP/IP。说他是创建Channel的,那么是如何创建的呢?其实也是利用配置的ChannelFactory创建的,这个赋值是在Bootstrap里面做的。

setFactory:

@1:最终还是调用Bootstrap的setFactory方法。

bind:

返回值:是一个用来接受客户端请求、绑定过地址的Channel。
@1:存放异步IO操作结果的队列。
@2:创建一个空的DefaultChannelPipeline。
@3:增加一个ChannelHandler(Binder)到pipeline上面。
@4:创建一个server-socket channel。
@5:当异步IO操作成功的完成的时候,会返回true。

channelOpen:

@1:参数是代表着渠道状态改变的事件(基于事件)。
@2:发送一个绑定的请求到指定Channel的ChannelPipeline中的ChannelDownstreamHandler。

上面是针对ServerBootstrap简单的一点分析,偏向于表面,大的结构,没有深入到具体的细节里面。无论要掌握何种知识,都要由浅入深,不要一上来就抓住某个点不放。循序渐进,有了坚持和热情,就相当于已经成功了。

学习 java netty -- ServerBootstrap

前言:我们自己使用java nio开发网络程序是非常繁琐的,netty为我们做好了一切,其中ServerBootstrap是一个启动辅助类,了解它我们就能开发出简单的nio 服务端程序。


不理解Nio中channel和handler等可参考上一篇文章
学习 java netty (一) – java nio


ServerBootstrap():

//创建一个ServerBootstrap对象
ServerBootstrap server = new ServerBootstrap;

ServerBootstrap只有一个无参构造函数,这里使用了构建器模式,原因是ServerBootstrap的参数过多,构建器模式就是用来处理构造函数参数过多而导致需要多个构造函数的问题。
简单来看:
当参数过多我们写出来的构造函数可能是
ServerBootstrap(int i);
ServerBootstrap(int i, double b);
ServerBootstrap(int i, double b, char c);

使用构建器模式后
ServerBootstrap(); 只有一个无参构造函数,其余需要构建的部分我们写到成员函数里,那么使用起来就像这样。
ServerBootstrap server = ServerBootstrap();
server
.construct1()
.construct2()
.consturct3()

当然也可以
server.construct1().construct3()…
我们使用哪个构建哪个即可,这就是构建器模式
构建器模式缺点:
写起来比较复杂,用起来开销很大,只适用于构造函数参数过多的情况。
注意构建器模式要返回this


ServerBootstrap继承自AbstractBootstrap
看一下ServerBootstrap class内部的变量和方法

public class ServerBootstrap extends AbstractBootstrap<ServerBootstrap, ServerChannel>
    //记录异常日志等错误,用了java的反射机制和工厂模式
    private static final InternalLogger logger = InternalLoggerFactory.getInstance(ServerBootstrap.class);
    //netty4.0引入了名为ChannelOption的新的类型,它提供了类型安全地访问socket选项。ChannelOption class内部就是socket套接字配置参数,例如SO_KEEPALIVE,SO_RECVBUF等
    private final Map<ChannelOption<?>, Object> childOptions = new LinkedHashMap<ChannelOption<?>, Object>();
    //实体的属性
    private final Map<AttributeKey<?>, Object> childAttrs = new LinkedHashMap<AttributeKey<?>, Object>();
    //事件循环,并绑定线程池
    private volatile EventLoopGroup childGroup;
    //处理channel的handler
    private volatile ChannelHandler childHandler;

    //方法
    //只有一个事件循环体,来接受accept和处理IO
    public ServerBootstrap group(EventLoopGroup group);
    //有两个事件循环体,一个用来accept请求,另外一个用来处理clinet的io
    public ServerBootstrap group(EventLoopGroup parentGroup, EventLoopGroup childGroup);
    //设置channel属性
    public <T> ServerBootstrap childOption(ChannelOption<T> childOption, T value);
    //设置那些符合给定属性的channel
    public <T> ServerBootstrap childAttr(AttributeKey<T> childKey, T value) ;
    //child处理channel的handler
    public ServerBootstrap childHandler(ChannelHandler childHandler);
    //return childGroup
    public EventLoopGroup childGroup();
    //初始化并配置一些参数
    void init(Channel channel) throws Exception
    public ServerBootstrap validate();

还有一些不太重要方法上面没有列举出来,还有些方法来自于父类AbstractBootstrap,代码太长就不贴了,感兴趣可以去git上看源码
AbstractBootstrap.java
但是从ServerBootstrap启动辅助类和ServerBootstrap class内部的参数和接口可以看出ServerBootstrap主要是用来配置服务端的。仅仅使用了一个对象ServerBootstrap就完成了复杂的服务端配置,让我们编程简化了许多。


看个例子:

public class EchoServer 
    private final int port;

    public EchoServer(int port) 
        this.port = port;
    

    public void run() throws Exception
        EventLoopGroup bossGroup = new NioEventLoopGroup();
        EventLoopGroup workerGroup = new NioEventLoopGroup();
        try 
            //创建一个ServerBootstrap,下面开始配置
            ServerBootstrap b = new ServerBootstrap();
            //创建了两个EventLoopGroup(线程池和selector,reactor模型)一个main loop一个child loop通过ServerBoootstrap的group方法组合起来
            //接着通过option方法传递服务端NioServerSocketChannel(服务端套接字)设置它的backlog参数。
            //再通过handler设置服务端socket(ServerSocketChannel)的处理事件的handler是记录日志logger
            //最后childhandler设置每一个连接到服务端的socket(socketchannel)的handler(childhandler),是创建一个EchoServerHandler类去处理。(利用channelhandler我们可以完成功能定制)
            //channelpipeline负责管理和执行channelhandler,可以向channelpipeline中add添加channelhandler
            //(注意ServerBootstrap部分方法来自父类)
            b.group(bossGroup, workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .option(ChannelOption.SO_BACKLOG, 100)
                    .handler(new LoggingHandler(LogLevel.INFO))
                    .childHandler(new ChannelInitializer<SocketChannel>() 
                        public void initChannel(SocketChannel ch) throws Exception 
                            //abstractBoostStrap中的Handler是个工厂类,它为每个新接入的客户端都创建一个新的Handler
                            //下面代码每新连接一个socket,都会创建一个EchoServerHandler
                            System.out.println("hello");
                            ch.pipeline().addLast(new EchoServerHandler());
                            //ch.pipeline().addLast(new LoggingHandler(LogLevel.INFO));
                        
                    );
            //绑定端口,netty中所有IO操作都是异步的,它会立即返回,但不能保证完成操作
            ChannelFuture f = b.bind(port).sync();
            System.out.println("bind....");
            f.channel().closeFuture().sync();
        catch(Exception e)
            e.printStackTrace();
        finally 
            bossGroup.shutdownGracefully();
            workerGroup.shutdownGracefully();
        
    


    public static void main(String[] args) throws Exception 
        int port;
        if(args.length > 0)
            port = Integer.parseInt(args[0]);
        else
            port = 10000;
        
        new EchoServer(port).run();
    

上面仅仅一行代码就完成了很多参数的设置,为我们编程简化了很多。

ServerBootstrap创建时序图:

图片非原创

补充工厂模式:
工厂模式利用java的反射,参数为Class<>,我们传递参数为.class(构建对象的类型)而不是类对象实例,然后根据需要在方法里构建相应的对象,这也延迟了对象的创建,简单来说就是我们需要什么类型的物品,告诉工厂,工厂产生给我们相应的实体。

backlog等参数的含义

以上是关于ServerBootstrap简介的主要内容,如果未能解决你的问题,请参考以下文章

Netty 核心源码解读 —— ServerBootstrap 篇

netty服务端启动--ServerBootstrap源码解析

09 - ServerBootstrap-绑定过程doBind0

学习 java netty -- ServerBootstrap

学习 java netty -- ServerBootstrap

95-20-050-启动器-ServerBootstrap