1. Github 地址 https://github.com/valyala/fasthttp

  2. fastHttp 服务端的处理请求的过程

    1. 工作过程

    2. 主要代码

      1. 设置监听地址 server.go

        func (s *Server) ListenAndServe(addr string) error {
        ln, err := net.Listen("tcp4", addr)
        if err != nil {
        return err
        }
        if tcpln, ok := ln.(*net.TCPListener); ok {
        return s.Serve(tcpKeepaliveListener{
        TCPListener: tcpln,
        keepalive: s.TCPKeepalive,
        keepalivePeriod: s.TCPKeepalivePeriod,
        })
        }
        return s.Serve(ln)
        }
      2. 初始化协程池并启动,当接收到请求后,调 wp.Serve 交给协程池去处理请求 server.go

      func (s *Server) Serve(ln net.Listener) error {
      
      	.....
      
      	wp := &workerPool{
      WorkerFunc: s.serveConn,
      MaxWorkersCount: maxWorkersCount,
      LogAllErrors: s.LogAllErrors,
      Logger: s.logger(),
      connState: s.setState,
      }
      wp.Start() for {
      if c, err = acceptConn(s, ln, &lastPerIPErrorTime); err != nil {
      wp.Stop()
      return err
      }
      s.setState(c, StateNew)
      atomic.AddInt32(&s.open, 1)
      if !wp.Serve(c) {
      atomic.AddInt32(&s.open, -1)
      s.writeFastError(c, StatusServiceUnavailable,
      "The connection cannot be served because Server.Concurrency limit exceeded")
      c.Close()
      s.setState(c, StateClosed)
      }
      }
      ......
      }
      1. 获取请求的句柄 server.go

        func acceptConn(s *Server, ln net.Listener, lastPerIPErrorTime *time.Time) (net.Conn, error) {
        for {
        c, err := ln.Accept()
        ......
        return c, nil
        }
        }
      2. 协程池处理请求 workerPool.go

        // 从协程池中获取某个协程对应的 句柄channel,然后将 3 中获取到的 请求句柄推入channel
        func (wp *workerPool) Serve(c net.Conn) bool {
        ch := wp.getCh()
        if ch == nil {
        return false
        }
        ch.ch <- c
        return true
        } // 这个是协程池工作最重要的部分,获取协程池中协程对应的channel
        func (wp *workerPool) getCh() *workerChan {
        var ch *workerChan
        createWorker := false wp.lock.Lock()
        ready := wp.ready
        n := len(ready) - 1
        // 获取协程句柄channel失败,如果可以新建协程的话就进行新建,如果不可用新建的话,返回的句柄channel为nil,本次请求被拒绝服务
        if n < 0 {
        if wp.workersCount < wp.MaxWorkersCount {
        createWorker = true
        wp.workersCount++
        }
        } else {
        // 获取协程句柄channel 成功
        ch = ready[n]
        ready[n] = nil
        wp.ready = ready[:n]
        }
        wp.lock.Unlock() if ch == nil {
        if !createWorker {
        return nil
        }
        // 新建协程句柄,且为之创建协程
        vch := wp.workerChanPool.Get()
        ch = vch.(*workerChan)
        go func() {
        wp.workerFunc(ch)
        wp.workerChanPool.Put(vch)
        }()
        }
        return ch
        } func (wp *workerPool) workerFunc(ch *workerChan) {
        var c net.Conn
        var err error
        for c = range ch.ch {
        //协程的句柄channel 出现nil的时候 当前协程就退出了
        if c == nil {
        break
        } // wp.WorkerFunc是在初始化协程池的时候注册的方法,是 s.serveConn
        if err = wp.WorkerFunc(c); err != nil && err != errHijacked {
        ......
        }
        }
        ......
        if !wp.release(ch) {
        break
        }
        }
        .....
        // 协程的句柄channel 出现nil的时候 当前协程就退出了,退出后将work协程数自减
        wp.workersCount--
        ......
        } // 请求处理完之后 将当前协程的句柄channel 放入协程池的ready切片中,待下次获取协程的句柄channel的时候进行复用(复用这个句柄channel就相当于复用了对应的协程)
        func (wp *workerPool) release(ch *workerChan) bool {
        ch.lastUseTime = time.Now()
        wp.lock.Lock()
        if wp.mustStop {
        wp.lock.Unlock()
        return false
        }
        wp.ready = append(wp.ready, ch)
        wp.lock.Unlock()
        return true
        }
      3. 实际处理请求的方法 server.go

        func (s *Server) serveConn(c net.Conn) (err error) {
        .....
        //上下文对象复用并初始化上下文
        ctx := s.acquireCtx(c)
        ......
        //执行在初始化server的时候自定义的 逻辑
        if continueReadingRequest {
        s.Handler(ctx)
        }
        .....
        // 处理http响应
        }
  3. 高效的原因

    1. fastHttp 协程池 详情见 2.2.2 、2.2.4
    2. fastHttp 对象复用
      1. 协程channle对象 复用 详见2.2.4 的 getCh() 函数
      2. 上下文对象复用 详见2.2.5

注:

本文为自己对fastHttp的理解,如有疑问,欢迎一起讨论交流

如需转载请注明出处:https://www.cnblogs.com/zhuchenglin/p/14358612.html

fastHttp服务端处理请求的过程的更多相关文章

  1. Query通过Ajax向PHP服务端发送请求并返回JSON数据

    Query通过Ajax向PHP服务端发送请求并返回JSON数据 服务端PHP读取MYSQL数据,并转换成JSON数据,传递给前端Javascript,并操作JSON数据.本文将通过实例演示了jQuer ...

  2. jQuery通过Ajax向PHP服务端发送请求并返回JSON数据

    SON(JavaScript Object Notation) 是一种轻量级的数据交换格式.易于人阅读和编写,同时也易于机器解析和生成.JSON在前后台交互的过程中发挥着相当出色的作用.请接着往下看教 ...

  3. Request对象 --web浏览器向web服务端的请求

    一]Request对象常用方法        1)StringBuffer getRequestURL()            url表示访问web应用的完整路径            2)Stri ...

  4. 服务端CURL请求

    服务端与服务端之间,也存在接口编程. 比如我们网站服务端,需要发送短信.发送邮件.查询快递等,都需要调用第三方平台的接口. 1.php中发送请求 ①file_get_contents函数 :传递完整的 ...

  5. 基于jFinal建立简单的服务端-接收请求并返回指定内容

    本菜鸡是一名弱弱的测试工程师,最近完成了一个支付相关的项目,项目工作中,需要建立一个模拟支付宝的网关,主要是接收请求并返回数据.作为一名没有丝毫开发经验的菜鸡,初期入门相当费劲,主要还是思维上的转变. ...

  6. Visual Studio 2015 Bowser Link的功能不停的向服务端发送请求

    Visual Studio 2015新建的mvc项目 默认在每个视图上生成一些JavaScript脚本

  7. 在java服务端判断请求是来自哪个终端

    在servlet中,我们可以获取到HttpServletRequest,然后通过HttpServletRequest的getHeader("User-Agent")方法获取请求头中 ...

  8. ajax对服务端发送请求

    //兼容处理获取ajax对象 var req = ''; if (window.XMLHttpRequest)    req = new XMLHttpRequest(); else    req = ...

  9. nextjs 服务端渲染请求参数

    Post.getInitialProps = async function (context) { const { id } = context.query const res = await fet ...

随机推荐

  1. java有关静态代理及动态代理的实现

    引言: 代理是基本的设计模式之一,它为你提供额外的或不同的操作而插入的用来代替实际对象的对象(委托对象).这些对象通常涉及到与实际对象的通信,因此代理通常充当中间人的角色.--JAVA编程思想 相关的 ...

  2. 关于HashSet

    HashSet存储数据原理: 当HashSet调用add方法时,有返回值,返回值是boolean类型,表示是否添加成功(如果对象不存在,则添加成功,否则添加失败) 但是,添加的过程并不是一个个去遍历去 ...

  3. Centos7 编译安装PHP7.2

    yum install wget 在 /usr/local/src 目录下载php源码包 wget http://cn2.php.net/distributions/php-7.2.4.tar.gz ...

  4. mysql词法分析和语法分析

    如果没有命中查询缓存,就要开始真正执行语句了.首先,MySQL 需要知道你要做什么,因此需要对 SQL 语句做解析.分析器先会做"词法分析".你输入的是由多个字符串和空格组成的一条 ...

  5. Oracle RedoLog-二进制格式分析,文件头,DML,DDL

    上篇文章,简单介绍了 RedoLog 是什么,以及怎么从 Oracle Dump 二进制日志.接下来,分析下 Redo Log 二进制文件的格式,主要包括:文件头,重做日志头,DML-INSERT 操 ...

  6. Windows下使用Graalvm将Javafx应用编译成exe

    1 背景 Graalvm是Oracle推出的一款新型虚拟机,其中一个吸引人的功能是:它可以将Java代码编译成各个平台的本地代码,这些平台包括:linux.macOS.windows.iOS.andr ...

  7. Second_week_mofangzhen

    第二周 奇数阶魔方阵 一.上节回顾 1.数组的基本操作 数组:若干个相同类型变量的集合. 声明:数据类型 数组名称[]; (在栈内存分配空间,存储的是数组的引用地址.数组首元素在堆内存 中的地址) 初 ...

  8. JavaScript中的原型、原型链、原型模式

    今天,咱来聊聊JavaScript中的原型跟原型链 原型跟原型模式 这一块的知识,主要是设计模式方面的. 首先,我们知道JavaScript是面向对象的.既然是面向对象,那它自然也有相应的类跟对象等概 ...

  9. 【Python】PDF转WORD

    注意,下文中的PDF文档是纯文字格式,而且非扫描版的PDF文件. 如果是扫描版或者带有图片的.可能转起来会出现排版异常并且图片无法保存到.doc文件中. 正文开始: 需要安装依赖包 pdfminer3 ...

  10. 【Samba】共享服务器的搭建和相关权限设置

    1.查看防护墙 [root@zhang~ ]# /etc/init.d/iptables status   iptables:Firewall is not running.   如果没有关闭的话将他 ...