背景
一个月前在线上发生了内存泄露。但很奇怪的是,那次发布只涉及几行代码,且都不涉及内存的主动分配。
我们都知道,如果发生了内存泄露,那么一定会有内存分配这个动作。
但当前的业务代码里并没有“内存分配”这个操作,所以问题八成是出在框架内部。
所修改的代码如下:
from django.urls import path
from django.http.response import HttpResponse
ALIVE_ECHO = HttpResponse('Alive')
urlpatterns = [
path('/', lambda request: ALIVE_ECHO),
]
因为这个URL只需要返回服务可用这个信息,所以每次返回的Response都是一样的。为了避免每次请求都生成一个Response这种无用开销,所以想预先分配一个Response。但正是这个操作导致了服务的内存泄露。
原因
排查的过程很曲折,从gunicorn、wsgi协议一直排查到了django http请求流程,用了Pympler这个工具帮忙排查。
直接说原