题记
前段时间的一次面试,面试官问我最近有读什么书,我答了云云,又问有什么收获或者令你印象深刻的地方,无话可说。似乎读过的书只记得书名,没有任何感悟体会,也无从证明。故写一些小记。
编程规约
-
序列化类新增属性并兼容升级时,serialVersionUID不要修改。serialVersionUID的作用是反序列化时,会判断当前类和反序列化对象的版本是否一致,一致才能反序列化。但是我们要兼容新增属性,则不用修改serialVersionUID,新增的属性会在反序列化中赋初值。
-
POJO对象都要写toString()方法,便于排查问题。
-
ArrayList的subList和ArrayList并不是一个类,因此不能够强转。
-
集合转数组方法toArray(T[] array)要带参的,若不带参只能返回Object[],强转会导致失败,而且T[] array这个数组的大小要和集合的大小保持一致,不一致也不会错,但非常不推荐。
-
数组转集合的方法Arrays.asList()返回的集合也是Arrays的一个内部类,并不能支持修改等操作,它是一种适配器模式,只改变了接口,底层仍是数组。如修改了array[0],那么list.get(0)也会跟着变化。
-
PECS (Producer Extends Consumer Super)
频繁读取(支持get,不支持add) 固定上边界 <? extends T>
频繁插入(支持add,不支持get) 固定下边界 <? extends T> -
线程池创建不能使用Executor,要用ThreadPoolExecutor,这样线程池运行规则更明确,也可以减少OOM的发生(因为Executor返回的线程池里,FixedThreadPool和SingleThreadPool的请求队列长度是Integer.MAX_VALUE,CachedThreadPool和ScheduledThreadPool的线程最大数也是Integer.MAX_VALUE,都容易造成内存溢出。)
-
SimpleDateFormat是将时间格式化的类,但是它是线程不安全的,在并发情况下,需要加锁或者结合ThreadLocal来使用。
-
switch的default是在没有匹配成功时执行,必须要有break,否则会往下执行直到遇到break。而且每个switch都必须有一个default,哪怕default是空。
异常日志
-
不能在finally里使用return,这样try里的return就无效了。
-
日志文件要保存15天以上,便于排查以周为单位的问题。
-
日志级别要清晰,生产环境禁止输出debug日志,有选择的输出info日志,warn日志记录潜在的问题,error日志只记录系统逻辑错误,异常等错误信息。
单元测试
-
好的单元测试要像AIR一样无声无息但必不可少。遵循三个原则Automatic,Independent,Repeatable。
-
单元测试基本目标:语句覆盖率70%,核心模块语句覆盖率和分支覆盖率100%。
安全规约
- 用户敏感数据禁止直接展示,必须进行脱敏,比如手机号176****1667。
MySQL数据库
-
表达是与否概念的字段命名为is_XXX,类型为unsigned tinyint,1表示是,0表示否。
-
数据库中的表名,字段名都要全部小写。表名不为复数,只表示表内实体内容,不表示表内实体的数量。
-
小数保存的类型为decimal,不能用float和double,他俩存在精度损失的问题,在比较大小的时候可能返回错误的结果。
-
不使用外键和级联更新,这种逻辑应该在应用层处理。比如A表中的id是主键,关联B表中的外键id,若更新A.id则会同时更新B.id,这就是级联更新。外键和级联更新适合单机低兵法,不适合高并发,外键还影响插入速度,因此尽量不使用。
-
POJO的类属性不能加is,数据库的必须加。
-
SUM()方法在数据为空时也返回NULL,可能会出现NPE问题,可以用这种SQL语句处理:
SELECT IF(ISNULL(SUM(score.score)),0,SUM(score.score)) FROM score;
工程结构
- 高并发服务器建议调小time_wait超时时间,默认是240s,高并发情况下可能会有过多的time_wait状态的线程。同时调大fd,Linux系统的fd数是1024,每个连接也会占用一个fd,因此要调高该值。
网友评论