关于Java优质代码的那些事

      以前别人告诉我,代码都是粘贴复制,然后写多了,就有了自己的思想,然后1,2年过去了,我的代码质量并没有什么提高,问了一些博客园里的前辈,前辈们都是语重心长的说:"少年,多看书呀!",我只当是敷衍我。知道我无意中看到一篇博客面试感悟----一名3年工作经验的程序员应该具备的技能,然后结实了大师哥:5月的仓颉,然后拜读了他的另一篇文章:给Java程序猿们推荐一些值得一看的好书,我一口气买了他文章里说的《Java多线程编程核心技术》,《Effective Java中文版》,《深入分析Java Web技术内幕》,《大型网站技术架构 核心原理与案例分析》,《Spring源码深度解析》,没有全部买下,主要是外文书理论比较强烈,我理解能力和兴趣不太感冒咬文嚼字。虽然还没看完,但是我坚持着,我一下子感觉到什么叫做优质代码。

       有人说,去看java源代码呀,可能个人性格使然吧,不是那种拔尖的人才,能慢慢咀嚼枯燥无味的源码,还是靠依赖大神的文章带着我去看源码,这样进度慢点,但是我觉得适合我,我能学到东西,有所沉淀。扯远了,这边随笔的目的,主要是告我自己,这些代码,我以后也可以尝试着用,代码看起来不是那么low。

 

1.使用 AtomicBoolean 高效并发处理 “只初始化一次” 的功能要求:

 可能会AtomicBoolean 这个对我来说,我都不知道这个啥意思,

对于官方的说明是:

可以用原子方式更新的 boolean 值。有关原子变量属性的描述,请参阅 java.util.concurrent.atomic 
包规范。AtomicBoolean 可用在应用程序中(如以原子方式更新的标志),但不能用于替换 Boolean。

换一句话说,Atomic就是原子性的意思,即能够保证在高并发的情况下只有一个线程能够访问这个属性值。

假设不使用AtomicBoolean ,代码如下:

public static volatile initialized  = false;

public void init(){
        if( initialized  == false ){
            initialized  = true;
            // 这里初始化代码....
        }
}
View Code

然后使用后的效果就比较明显:

1 private AtomicBoolean done_ = new AtomicBoolean(false);
2 
3     public void init()
4     {
5         if( done_.compareAndSet(false, true) )
6         {
7             // 这里放置初始化代码....
8         }
9     }
View Code

 

2.尽可能不要在For遍历中创建对象引用

很显然,创建对象意味着需要分配内存,一把小心,内存就爆了,哈哈,解决方法,我摘了师兄的

 1 for (int i = 1; i <= count; i++)
 2 {
 3     Object obj = new Object();    
 4 }
 5 
 6 //优化方法
 7 
 8 Object obj = null;
 9 for (int i = 0; i <= count; i++)
10 {
11     obj = new Object();
12 }

 

 3.要不要初始化HashMap的容量

 1    public static void main(String[] args) {
 2 
 3         long l1 = System.currentTimeMillis();
 4         for (int i=0;i<10000000;i++){
 5          //   Map<String, String> FileName = new HashMap<String, String>();//26  23  32
 6        //     Map<String, String> FileName = new HashMap<String, String>(5);//86 41 29
 7             Map<String, String> FileName = new HashMap<String, String>(18);//25 29 24
 8         }
 9 
10 
11         System.out.println(System.currentTimeMillis()-l1);
12     }

我测试了3种情况,一种是默认的int capacity=16;一种是小于16,一种是大于16,跑了3次,所耗时间如右侧注释,得出结论如下,

如果给定的capacity<16时反而会增加创建对象的所需时间;

            >=16时,给定一个初始化值,显然比默认的更加快。

 

 4.要不要初始化HashMap的内容

 1     public static void main(String[] args) {
 2 
 3         long l1 = System.currentTimeMillis();
 4         //1044
 5      /*  for (int i=0;i<10000000;i++){
 6             Map<String, String> FileName = new HashMap<String, String>(){
 7                 {
 8                     put("1", "66");
 9                     put("2", "deptAnnualSummary");
10                 }
11             };
12         }*/
13 
14        //1107
15         /*for (int i=0;i<10000000;i++){
16             Map<String, String> FileName = new HashMap<String, String>();
17             FileName.put("1", "66");
18             FileName.put("2", "deptAnnualSummary");
19         }*/
20 
21 
22         System.out.println(System.currentTimeMillis()-l1);
23     }

如上2种代码,一种是初始化的时候给HashMap赋值,一种则是我们常常看到的那种,我也反复测了几次性能,初始化赋值显然要优于另一种.

第一种方式一气呵成,定义了一个匿名内部类(Anonymous Inner Class),然后匿名内部类中再实例化一个代码块,即实例初始化块 (instance initializer block),在类构造时执行这个块。

推荐第一种

5.Java Stream for each如何用index

eg:循环List ,取第一个,并修改它属性,在不破坏streams 的优雅编码风格前提下,略感棘手,那么现在解决方法来了

1 List<MemberFollowUpRecordVO> params = Lists.newArrayList();
2 IntStream.range(0, params.size())
3   .forEach(idx ->
4     query.bind(
5       idx,
6       params.get(idx)
7     )
8   )
9 ;

 

 

 

6. 待续,我会持续更新,对自己代码质量有提交的代码段子,在这个随笔里面的,系希望有人扶正我的错误表达

 

posted @ 2017-04-11 15:54  谢幕ゾ华丽  阅读(405)  评论(2编辑  收藏  举报