go-fasthttp源码分析

架构

listener->server->workerpool

workerpool中有两种缓存:

  • wp.ready,缓存未退出worker,

  • worker退出后用sync.pool缓存channel

worker就是工作协程,它的存活期由maxIdleWorkerDuration控制,因此每次使用后都用lastUseTime记录。 workerpool每maxIdleWorkerDuration时间会使用“least recently used”策略清理掉空闲时间超过maxIdleWorkerDuration的,因为ready是按使用顺序排的(每次worker执行完workerFunc会append到最后),因此只要找到第一个时间超过maxIdleWorkerDuration的,把它和之前的workerChan都停掉、置nil即可。

  • 问题:为什么有了worker的缓存,还需要缓存channel?

回答:fasthttp号称“Zero memory allocations in hot paths”,channel的make也算是allocation,可能也会耗时吧。

listener可以是net.Listen生成,也可以是reuseport.Listen生成,对多核机器性能表现更好。

解码

把server.ServeConn()作为workerpool的WorkerFunc传给worker

s.serveConn(c)主要就是从pool中取一个ctx,然后用bufio.Reader|bytePool封装conn,然后http request.readLimitBody

通过b, err := r(bufio.Reader).Peek(n)来每次重新header.Parse(b),从请求行开始解析,最终readRawHeaders读到两个\n,再一股脑拷贝到RequestHeader.rawHeaders中。

  • 问题:为什么不用bufio.Reader.ReadBytes([]byte("\r\n\r\n"))直接读到headers结束呢?

回答:Peek是返回bufio.Reader.buffer的slice,没有memory allocation,比ReadBytes性能要好。

  • 问题:为什么不直接连续Peek到两个\r\n就结束?

回答:这段代码就是这个思路。

转载于:https://my.oschina.net/chuqq/blog/1574533

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值