原创:小姐姐味道(微信公众号ID:xjjdog),欢迎分享,转载请保留出处。java
部门新来了个架构师,BAT背景,住在三环,开宝立刻班,有车位。git
小伙话很少,但一旦说话斩钉截铁,带着没法撼动的自信。缘由就是,有他着数亿高并发经验,每一秒钟的请求,都是其余企业运行一年也没法企及的。这就让人很是羡慕,毕竟他靠这个比我赚的钱要多。程序员
俗话说,要想在公司不出事故,那就不要写代码。干活多了容易出事,一身轻松无人问津,这就是现实。安全
但有时候仍是要当作果的。新来的研发领导不懂技术,但他懂技术指标,因此就统计你们提交git的数量,若是git活动是一片绿色如A股,那就算过关了。bash
架构师思来想去,决定领一个并发量最高的需求:统计接口的平均响应时间和启动以来的请求数。微信
为何说它的并发量高呢?这是由于,它是统计全部接口的,天然比每个接口的请求量都要大。AOP代码一包,每一个接口都得从他这里走一圈。架构
该咱们的架构师上场了。代码如图。并发
架构师说,个人代码不须要作注释。所谓的注释,都是给垃圾代码用的。我深觉得是,他明显是受到了Netflix
公司的影响。高并发
程序考虑到了高并发场景,使用了线程安全的ConcurrentHashMap
,而后每次经过监控key取出相应的数据,而后在value上递增。这么简单的代码,确实不须要增长什么注释。测试
做为项目里并发量最高的代码,出于对高级架构师的信任,咱们并不须要作什么代码review,也不须要作什么测试。你们都很忙,代码您呐,到线上遛一遛吧。
我建议你先找一找代码的问题,若是你发现了问题,那就比架构师还厉害;若是你没发现,也不证实你比架构师弱,没有什么好伤心的。
下面插一副图,阻断一下思惟。
装B遭雷劈,线上运行一段时间后,内存溢出了。
你们吵吵个没完,毕竟xjjdog
说过,内存溢出问题的排查周期很长,大约平均须要40天左右才能解决问题。在你们开始论证的时候,架构师偷偷的启动了Eclipse MAT。MAT用来分析内存问题是很是合适的,但前提是你须要把堆栈给捣鼓下来。
架构师会用jmap
,最主要的是权限大,因而本身搞了一份拷贝到线下分析。
我能理解到他的心情,毕竟问题定位到本身的代码不是一件什么值得高兴的事情。他发现内存的堆里面,满满的全是MonitorKey
和MonitorValue
。
Monitor$MonitorKey@15aeb7ab
复制代码
我和架构师关系比较好,因而他问我:我们的接口是否是特别的多?
我说:不是啊,你别看访问量大,就这么个狗屁业务能有多少接口?几百个撑了天了。
他说:我在堆里发现了几千万个...
说完他就不言语了,由于他发现里面有很多是同样的接口。必定是参数的缘由,因此他在代码里加了这个,把?
后面的给截断了。
key = key.split("\\?")[0];
复制代码
结果发布到线上,过不了多久内存又溢出了。此次终于引发了大牛们的注意,通过你们的分析,发现代码是忘了给MonitorKey
重写equals
和hashCode
方法了。
我不由脸红起来。做为好朋友,我不该该让他出这个丑。但我又是隐隐快乐的,由于他工资比我高。
因此这就是一个很大的问题。不少同窗对HashMap的知识点对答如流,甚至还专门记忆了红黑树。但换一个方式去问,却又一脸懵逼。
其中一种问法是这样的:一个普通的对象,可以做为HashMap的key么?
答案显然是能够的,但须要注意重写hashCode和equals方法。若是忘记重写的话,大几率会形成内存泄漏。
很不幸,现实中忘记的案例不少。大牛架构师也会中招。
代码重写hashCode和equals方法后,线上就再也没发生过内存溢出。
等等,还没完。毕竟是架构师,仅仅这样一个bug仍是证实不了水平的。架构师写的bug,确定非比寻常。
这种事出现的多了,研发领导对技术的权威性就再也不是那么感冒。咱们决定从并发量最高的代码开始,进行一下代码review。
很不幸,架构师的visit
代码出现问题了。虽然问题不是很大,但它毕竟是个问题。
visit方法,首先拿出了key,而后判空,再塞值。这明显不是一个原子操做。
线程1:获取key为a的值
线程2:获取key为a的值
线程1:a为null,生成一个b
线程2:a为null,生成一个c
线程1:保存a=b
线程2:保存a=c
复制代码
此时,B丢了。
业务能够忍受,但严谨的技术大牛们忍受不了,提出了修改的意见。
架构师说,给visit方法加个synchronized
不就成了。
public synchronized void visit(String url, String desc, long timeCost) 复制代码
我说不行。有更优雅的写法,效率更高。那就是使用putIfAbsent
方法,代码改动以下:
MonitorKey key = new MonitorKey(url, desc);
MonitorValue value = monitors.putIfAbsent(key, new MonitorValue());
value = monitors.get(key);
value.count.getAndIncrement();
value.totalTime.getAndAdd(timeCost);
value.avgTime = value.totalTime.get() / value.count.get();
复制代码
你们就这两种方式争论了起来。
技术总监托着腮想了半天,看了看争的面红耳赤的同窗们,说:这就是我不放心大家的缘故。线上环境要尽可能保持稳定性,作最小的变动。既然加个synchronized
就可以很容易简单解决的问题,为啥不直接用呢?下面这种代码改动太大,有风险。
总监接着把头转向我:这个BUG非比寻常,为了让你们引觉得戒,你来作整个事故的复盘。把问题的排查和获得的教训分享给你们,让你们向这种至简的架构看齐。咱们日常的工做中,也要尽可能以结果导向为主,用什么手段无所谓,能漂亮把事情办好就行。
这就是此篇文章的由来,我虚心受教,同时也明白本身的工资是涨不上去了。
你要是点个赞或者友情三连,或许还能安慰我一下下。
做者简介:小姐姐味道 (xjjdog),一个不容许程序员走弯路的公众号。聚焦基础架构和Linux。十年架构,日百亿流量,与你探讨高并发世界,给你不同的味道。个人我的微信xjjdog0,欢迎添加好友,进一步交流。