1、考虑用静态工厂方法代替构造器

类的一个实例,通常使用类的公有的构造方法获取。也可以为类提供一个公有的静态工厂方法(不是设计模式中的工厂模式)来返回类的一个实例。例如:


//将boolean类型转换为Boolean类型
public static valueOf(boolean b) {
return b ? Boolean.TRUE : Boolean.FALSE;
}

使用静态工厂方法代替构造器的优势:

  • 静态工厂方法有名称,更易读。静态工厂方法能够使用方法名称进行自注释,来描述被返回的对象。例如:BigInteger.probablePrime的静态方法表示返回的BigInteger为素数。当一个类需要多个带有相同签名(参数列表仅在参数类型的顺序上有所不同)的构造器时,应使用静态工厂方法并且慎重的选择名称以便突出它们的区别。

  • 静态工厂方法能够为重复的调用返回相同的对象。将创建好的对象缓存起来,重复利用。

  • 静态工厂方法可以返回原返回类型的任何子类型的对象。例如:在基于接口的框架(通过接口来引用对象)中,为了隐藏类的实现,通常会使用API返回对象的实例。由于接口中不能有静态方法,通常把静态工厂方法放在实现了接口的不可实例化类(private构造函数)中。

  • 创建参数化类型实例时,静态工厂方法使代码变的更简洁。(java 8加入泛型的推导后,优势不再)


Map<String, List<String>> map = new HashMap<String, List<String>>(); // java 1.7之前必须这样写,前后两次都要写泛型列表
Map<String, List<String>> map = new HashMap<>(); // java 1.8可以这样写 //静态工厂方法
//HashMap类中加入
public static <K, V> HashMap<K, V> newInstance() {
return new HashMap<K, V>();
}
//使用
Map<String, List<String>> map = HashMap.newInstance();

缺点:

  • 类如果不含有public或protected型的构造器,则不能被继承。

2、遇到多个构造器参数时考虑用构建器

当有多个构造器参数(其中一些为可选参数)时,代码的编写通常有几种方式:

  • 使用重叠的构造器,根据可选参数提供不同的构造器。

  • 使用JavaBean模式:先调用一个无参的构造器来创建对象,然后使用setter方法来设置必要的参数。

当参数很多时,使用重叠的构造器代码的编写将很繁琐,难以阅读,同时调用时容易出错。而是用JavaBean模式,因为对象的构造过程被分到了几个调用中,在构造过程中JavaBean处于不一致状态。试图使用(特别是在多线程中)不一致状态的对象,将导致错误并且很难调试。并且JaveBean模式阻止了将类做成不可变的可能。

这时可以考虑使用构建器(Builder模式),不直接生成想要的对象,而是通过Builder对象来构造对象。

代码:


public class AlertDialog { private final String title; private final String message; private final boolean cancelable; private AlertDialog(Builder builder) {
this.title = builder.title;
this.message = builder.message;
this.cancelable = builder.cancelable;
} public static class Builder { private final String title; private String message; private boolean cancelable; public Builder(String title) {
this.title = title;
} public Builder setMessage(String message) {
this.message = message;
return this;
} public Builder setCancelable(boolean cancelable) {
this.cancelable = cancelable;
return this;
} public AlertDialog create() {
return new AlertDialog(this);
}
} public String toString() {
return "title: " + title + ", message: " + message + ", cancelable: " + cancelable;
} public static void main(String[] args) {
AlertDialog dialog = new AlertDialog.Builder("地点")
.setMessage("知识的荒漠")
.setCancelable(true).create();
System.out.println(dialog);
}
}

Builder模式的优点:

  • 比JavaBean更安全,比重叠的构造器更易于阅读和编写

  • 适用于参数较多,且大多数参数可选的情况

3、用私有构造器或枚举类型强化Singleton属性

实现Singleton的几种方式:

方式一:饿汉方式,线程安全


public class Singleton {
private static final Singleton singleton = new Singleton(); private Singleton() {} public static Singleton getInstance() {
return singleton;
}
}

这种方式基于classloader机制避免了线程同步的问题。但singleton在类被装载时立即实例化,没有实现类的延时加载(lazy loading)。另外这种方式可以通过反射机制(借助Class对象的setAccessible方法)来生成对象。

方式二:懒汉方式,线程不安全


public class Singleton {
private static final Singleton singleton = null; private Singleton() {} public static Singleton getInstance() {
if(singleton == null) {
singleton = new Singleton();
}
return singleton;
}
}

这种方式实现了类的延时加载,但致命的是多线程下不安全。

方式三:双重校验锁,线程安全


public class Singleton {
private volatile static Singleton singleton = null;
private Singleton() {}
public static Singleton getInstance() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}

方式四:枚举,线程安全,可避免反序列化时重新创建对象,推荐使用


public enum Singleton {
INSTANCE; private Singleton() {}
.....
} //使用
Singleton single = Singleton.INSTANCE;

这种方式可自由序列化,可有效避免反序列化时创建多个对象。保证只有一个实例(即使反射机制也无法多次实例化一个枚举量),线程安全。缺点是使用的人较少(java1.5后才有enum类型),失去了类的一些特性。单元素的枚举型是实现Singleton的最佳方式

4、通过私有构造器强化不可实例化的能力

编写一些只包含静态方法或静态域的类(工具类)时,可以使用private构造器来使类不能被实例化。这些类不能被继承。(因为子类的构造器都要显示或隐式调用父类的构造器)。自己编写的工具类要使用这种方法


public class UtilityClass {
private UtilityClass() {
throw new AssertionError(); //断言,确保不会在类内部被调用
}
.....
}

5、避免创建不必要的对象

重用对象而不是每次需要时创建一个相同的新对象,这可以显著程序提高性能。对于不可变的对象应始终重用,对于可变的对象应尽量重用。

例如:


String str1 = "abc";
String str2 = new String("abc");
String str3 = new String("abc");

内存模型:

new String("abc")每次执行都会创建一个新的String实例,在堆中分配一个新的空间。而使用String str1 = "abc"每次调用都将指向常量池中同一个常量,不会产生新的实例。

对于可变对象,那些一旦计算出来就不再变化的子对象或常量可单独提取到静态域中。


class Person {
private final Date birthDate;
private static final Date BOOM_STATE; //第一次初始化后将不改变,所有对象共用
private static final Date BOOM_END; static {
Calendar gmtCal = Calendar.getInstance(TimeZone.getTimeZone("GMT"));
gmtCal.set(1946, Calendar.JANUARY, 1, 0, 0, 0);
BOOM_STATE = gmtCal.getTime();
gmtCal.set(1965, Calendar.JANUARY, 1, 0, 0, 0);
BOOM_END = gmtCal.getTime();
} public boolean isBabyBoomer() {
return birthDate.compareTo(BOOM_STATE) >= 0 && birthDate.compareTo(BOOM_END) < 0;
}
}

当心无意识的自动装箱,要优先使用基本类型而不是包装类

例如:计算所有int正值的和


Long sum = 0L; //应使用基本类型long
for(long i=0; i<Integer.MAX_VALUE; i++ ) {
sum += i; //自动装箱,将创造2^32个多余的Long实例,降低性能
}
System.out.println(sum);

6、消除过期的对象引用

过期引用是指永远也不会再被解除的引用。在java中内存泄漏是隐藏的(无意识的对象保持)。如果一个对象的引用被无意识的保留下来,那么垃圾回收机制不会回收这个对象及这个对象所持有的所有对象。

消除过期引用最好的方法是让包含该引用的变量结束其生命周期。只要类自己管理内存,就应该小心内存泄漏问题。

	public class Stack {
private Object[] elements;
private int size = 0;
private static final int DEFAULT_INITIAL_CAPACITY = 16; public Stack() {
elements = new Object[DEFAULT_INITIAL_CAPACITY];
} public void push(Object o) {
ensureCapacity();
elements[size++] = o;
} public Object pop() {
if(size == 0)
throw new EmptyStackException();
Object result = elements[--size];
elements[size] = null; //【避免内存泄漏】
return result;
} private void ensureCapacity() {
if(elements.length == size) {
elements = Arrays.copyOf(elements, 2*size + 1);
}
}
}

对于垃圾回收器而言,elements数组中的所有对象的引用都同等有效,它无法区分哪些可以回收哪些不能回收。

内存泄漏的常见来源:

  • 数组

  • 缓存。使用WeakHashMap代表缓存,对于复杂的缓存必须使用java.lang.ref

  • 监听器及其回调。要及时取消注册,可以使用弱引用

7、避免使用终结方法

终结方法finalizer()为类Object中的方法,当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。

  • 终结方法是不可预测的,通常不要直接调用。

  • Java语言规范不保证终结方法会被立即执行,所以不应该依赖终结方法来更新重要的持久状态。

  • 终结方法可用作安全网或终止非关键的本地资源(记住调用super.finalize())

Effective java笔记(一),创建与销毁对象的更多相关文章

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

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

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

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

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

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

  4. 《Effective Java》读书笔记 - 2.创建和销毁对象

    Chapter 2 Creating and Destroying Objects item 1:Consider static factory methods instead of construc ...

  5. 【Effective Java读书笔记】创建和销毁对象(一):考虑使用静态工厂方法代替构造器

    类可以提供一个静态方法,返回类的一个静态实例,如Boolean包装类的一个获取实例的静态方法 public static Boolean valueOf(boolean b) { return (b ...

  6. 【Java基础】创建和销毁对象

    Num1:考虑用静态工厂方法代替构造器 对于类而言,常见的方法是提供一个公有的构造器,但其实还有一种方法叫做静态工厂方法(static factory method),它只是一个返回类的实例静态方法. ...

  7. Effective Java 读书笔记之一 创建和销毁对象

    一.考虑用静态工厂方法代替构造器 这里的静态工厂方法是指类中使用public static 修饰的方法,和设计模式的工厂方法模式没有任何关系.相对于使用共有的构造器来创建对象,静态工厂方法有几大优势: ...

  8. Effective Java 学习笔记之创建和销毁对象

    一.考虑用静态工厂方法代替构造器 1.此处的静态工厂方法是指返回指为类的对象的静态方法,而不是设计模式中的静态工厂方法. 2.静态工厂方法的优势有: a.使用不同的方法名称可显著地表明两个静态工厂方法 ...

  9. 《Effective Java》读书笔记(一)之创建和销毁对象

    最近在研读<Effective Java>一书,读书不做点笔记,感觉很容易就忘掉,于是用本篇博客来记录阅读此书的笔记. 郑重声明: 由于是<Effective Java>一书的 ...

  10. Effective Java学习笔记--创建和销毁对象

    创建和销毁对象 一.静态工厂方法代替构造器 静态工厂方法的优缺点 优点: 1.可以自定义名称(可以将功能表述的更加清晰) 2.不必每次调用都创建新的对象(同一对象重复使用) 3.返回的类型可以是原返回 ...

随机推荐

  1. xss和sql注入原理学习

    8.4 Web跨站脚本攻击 8.4.1  跨站脚本攻击的原理(1) 跨站脚本在英文中称为Cross-Site Scripting,缩写为CSS.但是,由于层叠样式表 (Cascading Style ...

  2. iOS -- CocoaPods

    CocoaPods 是什么? CocoaPods 是一个负责管理 iOS 项目中第三方开源库的工具.CocoaPods 的项目源码在 GitHub( https://github.com/CocoaP ...

  3. 关于javascript中的this关键字

    this是非常强大的一个关键字,但是如果你不了解它,可能很难正确的使用它. 下面我解释一下如果在事件处理中使用this. 首先我们讨论一下下面这个函数中的this关联到什么. function doS ...

  4. 技术笔记:XMPP之openfire+spark+smack

    在即时通信这个领域目前只找到一个XMPP协议,在其协议基础上还是有许多成熟的产品,而且是开源的.所以还是想在这个领域多多了解一下. XMPP协议:具体的概念我就不写了,毕竟这东西网上到处是.简单的说就 ...

  5. 浅谈浏览器http的缓存机制

    针对浏览器的http缓存的分析也算是老生常谈了,每隔一段时间就会冒出一篇不错的文章,其原理也是各大公司面试时几乎必考的问题. 之所以还写一篇这样的文章,是因为近期都在搞新技术,想“回归”下基础,也希望 ...

  6. DoraCMS 源码知识点备注

    项目需要研究了下DoraCMS这款开源CMS,真心做的不错:).用的框架是常用的express 4 + mongoose,代码也很规范,值得学习. 源码中一些涉及到的小知识点备注下: https:// ...

  7. Maven实战:pom.xml与settings.xml

    pom.xml与settings.xml pom.xml与setting.xml,可以说是Maven中最重要的两个配置文件,决定了Maven的核心功能,虽然之前的文章零零碎碎有提到过pom.xml和s ...

  8. 前端构建大法 Gulp 系列 (二):为什么选择gulp

    系列目录 前端构建大法 Gulp 系列 (一):为什么需要前端构建 前端构建大法 Gulp 系列 (二):为什么选择gulp 前端构建大法 Gulp 系列 (三):gulp的4个API 让你成为gul ...

  9. 玩转Windows服务系列汇总

    玩转Windows服务系列汇总 创建Windows服务 Debug.Release版本的注册和卸载及其原理 无COM接口Windows服务启动失败原因及解决方案 服务运行.停止流程浅析 Windows ...

  10. [异常解决] How make ubuntu use Google Search

    1.Download the hosts file fromhttps://laod.cn/hosts/2016-google-hosts.html [1] 2.Write a bash shell ...