• Netty简略了解与 源码分析


    最近看了一下netty,还是把学习的总结一下,不然过几天全忘记了。

    1.netty是啥

    一个提供异步事件驱动的网络应用框架,在java io的基础上进行了包装开发,特性:高性能、可维护、快速开发。

    使用例子见官网:Netty.docs: User guide for 5.x

    2. netty整体框架

    2.1 netty核心 Selector

     2.2 启动核心步骤

    主线程

    2.2.1创建Selector

    2.2.2创建ServerSocketChannel 与初始化ServerSocketChannel

    1. io.netty.bootstrap.AbstractBootstrap#doBind:
    2. final ChannelFuture regFuture = this.initAndRegister();

    2.2.4给ServerSocketChannel选择运行NioEventLoop

    1. io.netty.channel.MultithreadEventLoopGroup#register(io.netty.channel.Channel):
    2. //选择一个处理注册事件
    3. public EventExecutor next() {
    4. return this.chooser.next();
    5. }

    NioEventLoop 线程:

    2.2.5 ServerSocketChannel 注册到Selector

    1. io.netty.channel.AbstractChannel.AbstractUnsafe#register:
    2. //eventLoop 处理
    3. if (eventLoop.inEventLoop()) {
    4. this.register0(promise);
    5. } else {
    6. try {
    7. eventLoop.execute(new OneTimeTask() {
    8. public void run() {
    9. AbstractUnsafe.this.register0(promise);
    10. }
    11. });
    12. }

     2.2.6 绑定地址启动

    1. io.netty.channel.socket.nio.NioServerSocketChannel#doBind
    2. this.javaChannel().bind(localAddress)

    2.2.7 注册接受连接事件(OP_ACCEPT)到Selector 上

     2.3 客服端与服务端交互

    下图主要展示的是服务端流程,客户端与服务端有一些不同:启动类是Bootstrap,直接在第2步时就创建NioSocketChannel,并将NioSocketChannel注册给Selector,注册完成后initChannel,之后解析连接,并向服务端发起连接事件。待连接完成后激活,注册读事件到Selector,调用ChannelActive-》beginRead-》selectionKey.interestOps(interestOps | this.readInterestOp)。

     3.1 Netty之责任链模式

    在netty中核心事件的处理都是用的责任链模式,核心方法定义在接口ChannelHandlerInvoker中,并在相对应的方法还在接口ChannelHandlerContext、ChannelPipeline 有定义。调用过程为:ChannelPipeline-》ChannelHandlerContext--》ChannelHandlerInvoker--》具体实现Handler。

    责任链上下文AbstractChannelHandlerContext ,责任处理器的接口在ChannelPipeline中,Pipeline会有两指针指向头尾责任链头尾节点,Inbound事件从头结点开始向尾遍历,Outbound事件从尾节点开始向头遍历。

     在调用时,首先获取到pipeline,再通过pipeline调用方法,例如注册方法

    AbstractChannel.this.pipeline.fireChannelRegistered();
    DefaultChannelPipeline:
    1. public ChannelPipeline fireChannelRegistered() {
    2. this.head.fireChannelRegistered();
    3. return this;
    4. }
    AbstractChannelHandlerContext 中包含的重要成员变量:
    1. volatile AbstractChannelHandlerContext next;
    2. volatile AbstractChannelHandlerContext prev;
    3. private final AbstractChannel channel;
    4. private final DefaultChannelPipeline pipeline;
    5. final int skipFlags;
    6. private final ChannelHandler handler;
    1. public ChannelHandlerContext fireChannelRegistered() {
    2. AbstractChannelHandlerContext next = this.findContextInbound();
    3. next.invoker().invokeChannelRegistered(next);
    4. return this;
    5. }
    6. private AbstractChannelHandlerContext findContextInbound() {
    7. AbstractChannelHandlerContext ctx = this;
    8. do {
    9. ctx = ctx.next;
    10. } while((ctx.skipFlags & 2044) == 2044);
    11. return ctx;
    12. }
    13. public final ChannelHandlerInvoker invoker() {
    14. return (ChannelHandlerInvoker)(this.invoker == null ? this.channel().eventLoop().asInvoker() : this.wrappedEventLoop());
    15. }

    其中AbstractChannel 包含PausableChannelEventLoop,PausableChannelEventLoop implements ChannelHandlerInvoker 接口,因此next.invoker().invokeChannelRegistered(next);可以调用ChannelHandlerInvoker的方法

         AbstractChannel.this.eventLoop = AbstractChannel.this.new PausableChannelEventLoop(eventLoop);
    DefaultChannelHandlerInvoker:具体任务执行交给NioEventLoop,如果当前运行的线程时NioEventLoop,直接调用,否则提交给executor(例如单线运行的线程是main)
    1. public void invokeChannelRegistered(final ChannelHandlerContext ctx) {
    2. if (this.executor.inEventLoop()) {
    3. ChannelHandlerInvokerUtil.invokeChannelRegisteredNow(ctx);
    4. } else {
    5. this.executor.execute(new OneTimeTask() {
    6. public void run() {
    7. ChannelHandlerInvokerUtil.invokeChannelRegisteredNow(ctx);
    8. }
    9. });
    10. }
    11. }

    具体调用在ChannelHandlerInvokerUtil中,调用通过findContextInbound找到的上下文中的hander了

    1. public static void invokeChannelRegisteredNow(ChannelHandlerContext ctx) {
    2. try {
    3. ctx.handler().channelRegistered(ctx);
    4. } catch (Throwable var2) {
    5. notifyHandlerException(ctx, var2);
    6. }
    7. }
    class ChannelInitializer extends ChannelHandlerAdapter:

    方法执行后想责任链继续往下走,会继续调用 ctx.fireChannelRegistered();

    1. public final void channelRegistered(ChannelHandlerContext ctx) throws Exception {
    2. ChannelPipeline pipeline = ctx.pipeline();
    3. boolean success = false;
    4. try {
    5. this.initChannel(ctx.channel());
    6. pipeline.remove(this);
    7. ctx.fireChannelRegistered();
    8. success = true;
    9. } catch (Throwable var8) {
    10. logger.warn("Failed to initialize a channel. Closing: " + ctx.channel(), var8);
    11. } finally {
    12. if (pipeline.context(this) != null) {
    13. pipeline.remove(this);
    14. }
    15. if (!success) {
    16. ctx.close();
    17. }
    18. }
    19. }

     4.1 Netty之读数据

              读数据技:自适应分配器

    1.分配一个初始 1024 字节的 byte buffer 来接受数
    2. 记录实际接受数据大小,调整下次分配 byte buffer
    3. 尝试继续读取直到没有数据或满 16

     5.1 Netty之写数据

    1Netty 待写数据太多,超过一定的水位线( writeBufferWaterMark .high) () ),会将可写的标志位改false ,让应用端自己做决定要不要发送数据

    2、只要有数据要写,且能写,则一直尝试,直到 16 次( writeSpinCount ),写 16 次还没有写完,就直接 schedule 一个 task 来继续写,而不是用注册写事件来触发,更简洁有力

    3、批量写数据时,如果尝试写的都去了,接下来会尝试写更多(maxBytesPerGatheringWrite

     6.1 Netty之多线程模式

    在netty中,NioEventLoop会执行任务,其中线程池才用的是ForkJoinPool,

    在netty中的ForkJoinPo是netty自己定义的。nEventExecutors默认16。

    1. eventLoop.execute(new OneTimeTask() {
    2. public void run() {
    3. AbstractUnsafe.this.register0(promise);
    4. }
    5. });

    任务添加过程下图所示:

    run:

     

     唤醒selector,如果selector阻塞等待有事件注册,执行后selector直接返回

    1. protected void wakeup(boolean inEventLoop) {
    2. if (!inEventLoop && this.wakenUp.compareAndSet(false, true)) {
    3. this.selector.wakeup();
    4. }
    5. }
    在里面设置任务队列数,与java11中设置任务队列数一样。
    1. WorkQueue[] nws;
    2. label214: {
    3. nps = this.parallelism;
    4. s = nps > 1 ? nps - 1 : 1;
    5. s |= s >>> 1;
    6. s |= s >>> 2;
    7. s |= s >>> 4;
    8. s |= s >>> 8;
    9. s |= s >>> 16;
    10. s = s + 1 << 1;
    11. nws = (ws = this.workQueues) != null && ws.length != 0 ? null : new WorkQueue[s];
    1. protected ForkJoinWorkerThread(ForkJoinPool pool) {
    2. super("aForkJoinWorkerThread");
    3. this.pool = pool;
    4. this.workQueue = pool.registerWorker(this);
    5. }

    WorkQueue

    1. WorkQueue(ForkJoinPool pool, ForkJoinWorkerThread owner, int mode, int seed) {
    2. this.pool = pool;
    3. this.owner = owner;
    4. this.mode = (short)mode;
    5. this.hint = seed;
    6. this.base = this.top = 4096;
    7. }

    这很像WorkStealingPool,Executors.newWorkStealingPool();其底层也是ForkJoinPool。

    jdk11中的ForkJoinPool:与ThreadPoolExecutor创建的线程池不一样,ThreadPoolExecutor是所有线程共用一个任务队列,ForkJoinPool有多个任务队列。

    参数:

    int parallelism,并行度,

    ForkJoinWorkerThreadFactory factory,创建线程的工程

    UncaughtExceptionHandler handler:在内部线程执行时发生不可恢复的错误终止时的处理程序。

    asyncMode:如果为ture,则为从未加入的 Fork任务建立本地先进先出调度模式。

    corePoolSize:线程池中保留存活的线程数。

    maximumPoolSize:线程池允许的最大线程数。

    minimumRunnable:最小没有阻塞的线程数,如果小于,则新创建线程,直到达到最大线程数。

    Predicate saturate:如果该参数不为空,当尝试创建超过最大线程数的线程时,则会调用Predicate。Predicate返回ture,不会抛出异常,线程池会继续以小于目标可运行的线程数继续运行,但不能保证进度。否则,当一个线程被阻塞,由于达到最大线程数,无法替换时,抛出异常RejectedExecutionException。

    keepAliveTime:自从最后一次被使用后的存活时间。

    unit:存活时间单位。

    ForkJoinPool的任务队列数:parallelism<1,n=4,否则parallelism-1后的二进制最高位1(其余位为0)往左移2位,这就是任务队列数。

    1. public ForkJoinPool(int parallelism,
    2. ForkJoinWorkerThreadFactory factory,
    3. UncaughtExceptionHandler handler,
    4. boolean asyncMode,
    5. int corePoolSize,
    6. int maximumPoolSize,
    7. int minimumRunnable,
    8. Predicate<? super ForkJoinPool> saturate,
    9. long keepAliveTime,
    10. TimeUnit unit) {
    11. // check, encode, pack parameters
    12. if (parallelism <= 0 || parallelism > MAX_CAP ||
    13. maximumPoolSize < parallelism || keepAliveTime <= 0L)
    14. throw new IllegalArgumentException();
    15. if (factory == null)
    16. throw new NullPointerException();
    17. long ms = Math.max(unit.toMillis(keepAliveTime), TIMEOUT_SLOP);
    18. int corep = Math.min(Math.max(corePoolSize, parallelism), MAX_CAP);
    19. long c = ((((long)(-corep) << TC_SHIFT) & TC_MASK) |
    20. (((long)(-parallelism) << RC_SHIFT) & RC_MASK));
    21. int m = parallelism | (asyncMode ? FIFO : 0);
    22. int maxSpares = Math.min(maximumPoolSize, MAX_CAP) - parallelism;
    23. int minAvail = Math.min(Math.max(minimumRunnable, 0), MAX_CAP);
    24. int b = ((minAvail - parallelism) & SMASK) | (maxSpares << SWIDTH);
    25. int n = (parallelism > 1) ? parallelism - 1 : 1; // at least 2 slots
    26. n |= n >>> 1; n |= n >>> 2; n |= n >>> 4; n |= n >>> 8; n |= n >>> 16;
    27. n = (n + 1) << 1; // power of two, including space for submission queues
    28. this.workerNamePrefix = "ForkJoinPool-" + nextPoolId() + "-worker-";
    29. this.workQueues = new WorkQueue[n];
    30. this.factory = factory;
    31. this.ueh = handler;
    32. this.saturate = saturate;
    33. this.keepAlive = ms;
    34. this.bounds = b;
    35. this.mode = m;
    36. this.ctl = c;
    37. checkPermission();
    38. }

    使用例子:

    一般我们定义任务的时候是从Runnable来继承,但ForkJionPoll需要定义成ForkJoinTask,在下面的例子中将使用RecursiveTask(继承了ForkJoinTask),会递归的把大任务切分成小任务,直到满足条件为止。例外还有一个特别相似的RecursiveAction,这个没有返回值,前面的RecursiveTask有返回值。

    计算一个 给定数组的累加和。

    1. public class T1_ForkJoinPool {
    2. static int[] nums =new int[1000000];
    3. static final int MAX_NUM=50000;
    4. static Random r =new Random();
    5. static {
    6. for(int i=0;i<nums.length;i++){
    7. nums[i]=r.nextInt(100);
    8. }
    9. int sum = 0;
    10. for (int num : nums) {
    11. sum += num;
    12. }
    13. System.out.println("----"+ sum);
    14. }
    15. static class AddTaskRet extends RecursiveTask<Long> {
    16. private static final long serialVersionUID=1L;
    17. int start,end;
    18. public AddTaskRet(int start, int end) {
    19. this.start = start;
    20. this.end = end;
    21. }
    22. @Override
    23. protected Long compute() {
    24. if(end-start<=MAX_NUM){
    25. long sum =0L;
    26. for(int i=start;i<end;i++){
    27. sum+=nums[i];
    28. }
    29. return sum;
    30. }
    31. int middle =start+(end-start)/2;
    32. AddTaskRet sub1=new AddTaskRet(start,middle);
    33. AddTaskRet sub2= new AddTaskRet(middle,end);
    34. sub1.fork();
    35. sub2.fork();
    36. return sub1.join()+sub2.join();
    37. }
    38. }
    39. public static void main(String[] args) {
    40. T1_ForkJoinPool t1_forkJoinPool=new T1_ForkJoinPool();
    41. ForkJoinPool joinPool =new ForkJoinPool();
    42. AddTaskRet taskRet =new AddTaskRet(0,nums.length);
    43. joinPool.execute(taskRet);
    44. long result=taskRet.join();
    45. System.out.println(result);
    46. }
    47. }

    netty之零拷贝

    1. //FileChannel 文件读写、映射和操作的通道
    2. FileChannel fileChannel = new FileInputStream(fileName).getChannel();
    3. long startTime = System.currentTimeMillis();
    4. //transferTo⽅法⽤到了零拷⻉,底层是sendfile,这里只需要发生2copy2次上下文切换
    5. long transferCount = fileChannel.transferTo(0, fileChannel.size(), socketChannel);

  • 相关阅读:
    LeetCode刷题2:链表篇
    STC单片机19——ds18b20 液晶显示温度
    01【数据库的介绍】
    【Flink集群RPC通讯机制(三)】AkkaRpcActor设计与实现:接收RPC消息以及处理逻辑
    NOIP2006-2018 提高组 初赛试题完善程序题 CSP-S 2019 2020 初赛试题完善程序题
    Flink学习之旅:(三)Flink源算子(数据源)
    centos或aws linux部署java应用,环境搭建shell
    riscv引导程序及仿真记录
    Vue教程,React教程
    基于ASP.NET的Web酒品销售商城平台系统
  • 原文地址:https://blog.csdn.net/u010603891/article/details/126783108