最近准备写点Javase的东西,希望可以帮助大家写出更好的代码。

1、给不可实例化的类提供私有构造器

比如:每个项目中都有很多工具类,提供了很多static类型的方法供大家使用,谁也不希望看到下面的代码:

  1. TextUtils textUtils = new TextUtils();
  2. if(textUtils.isDigitsOnly("123"))
  3. {
  4. //doSometing
  5. }else
  6. {
  7. //doSomething
  8. }

自己写个工具类,总有人喜欢先初始化个实例在调用方法,然后还附带一个警告:The static method isDigitsOnly(CharSequence) from the type TextUtils should be accessed in a static way 。 你建议他使用类名.方法,人家还不乐意,我又没出错,干嘛要改,错了你负责么。所以最好的方式,让他没办法new实例。

为工具类添加私有构造器:

  1. public class TextUtils {
  2. private TextUtils() { /* cannot be instantiated */ }

这是android的TextUtils的源码,这样就可以了,让他妹的初始化实例~,当然你也可以在私有方法里面扔个异常。

  1. public class TextUtils
  2. {
  3. private TextUtils()
  4. {
  5. /* cannot be instantiated */
  6. throw new UnsupportedOperationException("cannot be instantiated");
  7. }
  8. }

对于异常的使用,一尽量使用Java提供的异常类,这样可以使你的API比较易读和易懂。

2、正确使用String,避免创建不必要的对象

很多人面试的时候都遇到过这样的问题:String s = new String("abc");请问创建了几个对象。也从侧面说明了这是个反面的代码写法:

a、String s = new String("abc");“abc”本身就是一个String的实例,所以new String创建了不必要的String实例

b、如果改写成 String s = "abc",不仅只创建了一个实例,而且在同一台VM中,对于“abc”(字符串的字面常量)还会重用。

3、优先使用基本类型,Java提供了8种基本类型,以及对应的装箱基本类型,且在Java1.5 提供了自动装箱和解箱操作,虽然方便了代码的编写,但是如果不注意,可能带来不好的效果。

看下面的代码:

  1. long start = System.nanoTime();
  2. Long sum = 0L;
  3. for (long i = 0; i < Integer.MAX_VALUE; i++)
  4. {
  5. sum += i;
  6. }
  7. System.out.println(sum);
  8. System.out.println(System.nanoTime() - start);//20995956735

如果你观察了内存,会发现,一直GC一直在内存回收,并且计算时间需要20多秒,如果我说这段代码有个bug,导致代码运行很慢,以及耗费内存,你能找到吗?

下面我修改下代码:

  1. long start = System.nanoTime();
  2. long sum = 0l;
  3. for (long i = 0; i < Integer.MAX_VALUE; i++)
  4. {
  5. sum += i;
  6. }
  7. System.out.println(sum);
  8. System.out.println(System.nanoTime() - start);//5029758632

这次运行不会出现GC一直回收内存,且速度也只需要5秒左右,可能眼神不好的,没有发现哪个地方修改了。

问题就出在自动装箱、解箱上。第一次的程序sum为Long类型,在计算sum+=i;时会把sum自动解箱成long sum 然后运算,运算完成后,再装箱成Long sum,导致程序构造了大约2的32次方个多余Long实例。所以各位且用且严谨。

4、对于自己管理内存的类,一定要清除不必要的对象引用,防止内存泄漏

看下面的代码:

  1. package com.zhy._01;
  2.  
  3. import java.util.Arrays;
  4.  
  5. /*
  6. * 使用数组模拟栈
  7. */
  8. public class MyStack
  9. {
  10. private static final int DEFAULT_INIT_SIZE = 10;
  11.  
  12. private Object[] eles = new Object[DEFAULT_INIT_SIZE];
  13. /**
  14. * 当前栈顶索引
  15. */
  16. private int currentIndex;
  17.  
  18. /**
  19. * 弹栈
  20. *
  21. * @return
  22. */
  23. public Object pop()
  24. {
  25. if (currentIndex == 0)
  26. throw new ArrayIndexOutOfBoundsException("stack is empty");
  27. return eles[--currentIndex];
  28. }
  29.  
  30. /**
  31. * 压栈
  32. *
  33. * @param o
  34. */
  35. public void push(Object o)
  36. {
  37. ensureCapacity();
  38. eles[currentIndex++] = o;
  39. }
  40.  
  41. private void ensureCapacity()
  42. {
  43. if (eles.length == currentIndex)
  44. {
  45. eles = Arrays.copyOf(eles, currentIndex * 2 + 1);
  46. }
  47. }
  48.  
  49. }

代码中存在一个地方,导致了内存泄漏,你可以发现不?

  1. return eles[--currentIndex];

这行代码导致,如果栈增长了特别大,然后调用多次pop弹栈,虽然currentIndex小了,但是栈始终保持中之前pop出的过期对象的引用,这就导致了内存泄漏。如果不注意甚至最终造成OOM。

应该改为:

  1. /**
  2. * 弹栈
  3. *
  4. * @return
  5. */
  6. public Object pop()
  7. {
  8. if (currentIndex == 0)
  9. throw new ArrayIndexOutOfBoundsException("stack is empty");
  10. Object tmp = eles[--currentIndex];
  11. eles[currentIndex] = null ;
  12. return tmp ;
  13. }

当然了,不要因为担心内存泄漏,在每个变量使用完成后都添加xxx=null,对于消除过期引用的最好方法,就是让包含该引用的变量结束生命周期,而不是显示的清空。一般情况下,对于类自己管理的内存,应当警惕。

好了,就到这里,这些内容都是我个人觉得值得知道,且在项目中会常遇到的,希望可以帮助到大家,嘿嘿,求评论,求赞。

Java进阶 创建和销毁对象的更多相关文章

  1. 《Effective Java》—— 创建与销毁对象

    本篇主要总结的是<Effecticve Java>中关于创建和销毁对象的内容. 比如: 何时以及如何创建对象 何时以及如何避免创建对象 如何确保及时销毁 如何管理对象销毁前的清理动作 考虑 ...

  2. 和我一起学Effective Java之创建和销毁对象

    前言 主要学习创建和销毁对象: 1.何时以及如何创建对象 2.何时以及如何避免创建对象 3.如何确保它们能够适时地销毁 4.如何管理对象销毁之前必须进行的清理动作 正文 一.用静态工厂方法代替构造器 ...

  3. Effective Java - [2. 创建与销毁对象]

    让对象的创建与销毁在掌控中. Item 1: 使用静态工厂方法而非使用构造函数 public static Boolean valueOf(boolean b) { return b ? Boolea ...

  4. 【读书笔记】《Effective Java》——创建和销毁对象

    Item 1. 考虑用静态工厂方法替代构造器 获得一个类的实例时我们都会采取一个共有的构造器.Foo x = new Foo(): 同时我们应该掌握另一种方法就是静态工厂方法(static facto ...

  5. Effective Java笔记一 创建和销毁对象

    Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...

  6. [Effective Java]第二章 创建和销毁对象

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  7. 《Effect Java》学习笔记1———创建和销毁对象

    第二章 创建和销毁对象 1.考虑用静态工厂方法代替构造器 四大优势: i. 有名称 ii. 不必在每次调用它们的时候都创建一个新的对象:   iii. 可以返回原返回类型的任何子类型的对象: JDBC ...

  8. effective java读书小记(一)创建和销毁对象

    序言 <effective java>可谓是java学习者心中的一本绝对不能不拜读的好书,她对于目标读者(有一点编程基础和开发经验)的人来说,由浅入深,言简意赅.每一章节都分为若干的条目, ...

  9. Effective Java(1)-创建和销毁对象

    Effective Java(1)-创建和销毁对象

随机推荐

  1. hdu 4715

    #include<stdio.h> #include<string.h> int prime[1100000],p[1000000],ans; void pri() {     ...

  2. TWinControl的消息覆盖函数大全(41个WM_函数和31个CM_函数,它的WndProc就处理鼠标(转发)、键盘(取消拖动)、焦点、和WM_NCHITTEST一共4类消息)

    注意,这些函数只有Private一种形式(也就是不允许覆盖,但仍在动态表格中): 其中TWinControl对TControl有10个消息进行了覆盖(红色标记),其中有2个是WM_消息,8个是CM_消 ...

  3. XML序列化反序列化—常用类

    public class XMLSerializer    {        #region (public) xml序列化        /// <summary>        /// ...

  4. java之jvm学习笔记十三(jvm基本结构)

    java之jvm学习笔记十三(jvm基本结构) 这一节,主要来学习jvm的基本结构,也就是概述.说是概述,内容很多,而且概念量也很大,不过关于概念方面,你不用担心,我完全有信心,让概念在你的脑子里变成 ...

  5. 图像编程学习笔记1——bmp文件结构处理与显示

    文本内容转载自<数字图像处理编程入门>,代码为自己实现 1.1图和调色板的概念 如今Windows(3.x以及95,98,NT)系列已经成为绝大多数用户使用的操作系统,它比DOS成功的一个 ...

  6. 在web网页中正确使用图片格式

    今天又看了一遍淘宝平四分享的PPT,以前转载网址:http://blog.sina.com.cn/s/blog_995c1f6301017fd2.html

  7. 从零开始,使用python快速开发web站点(2)

    书接上文.http://blog.csdn.net/i7788/article/details/10306595 首先是数据库的搭建,这里的django的数据模型搭建十分easy. no sql.ju ...

  8. 高效合并两个有序数组(Merge Sorted Array)

    Given two sorted integer arrays nums1 and nums2, merge nums2 into nums1 as one sorted array. Note: Y ...

  9. poj3281(最大流)

    传送门:Dining 题意:一些牛,一些食物,一些饮料,每头牛都有其喜欢的几种食物和几种饮料,求最多能给多少头牛即找到食物又找到饮料~也就是有多少个 牛---食物---饮料 的匹配,而且满足一一匹配, ...

  10. WebService 通过POST方式访问时候,因 URL 意外地以“/方法名”结束,请求格式无法识别 解决办法

    因URL意外地以“/方法名”结束,请求格式无法识别. 执行当前Web请求期间生成了未处理的异常.可以使用下面的异常堆栈跟踪信息确定有关异常原因和发生位置的信息. 解决方法:在webservice的we ...