关于饿汉式、懒汉式见字不觉想起:
有二女如斯:一个是有心羡的男士追求的时候,才答应对方做自己的男朋友,有如懒汉(女汉哈);另一个是永远有一个备胎,随时需要的时候都可以当做男朋友,有如饿汉。此例虽劣,有点意思。
1.-- 懒汉式
package com.niewj.basic.multi.patterns;
/**
* 懒汉式单例
* Created by weijun.nie on 2017/10/12.
*/
public class Singleton1 {
private static Singleton1 instance;
private Singleton1() {
System.out.println("实例初始化...");
}
public static Singleton1 getInstance() {
if (instance == null) {
instance = new Singleton1();
}
return instance;
}
public static void main(String[] args) {
// TODO -多线程时会有可能有多个实例
for (int i = 0; i < 10; i++) {
new Thread(new Runnable() {
public void run() {
System.out.println(Singleton1.getInstance());
}
}, "" + i).start();
}
}
}
实例初始化...
实例初始化...
com.niewj.basic.multi.patterns.Singleton1@4f9d1ec9
com.niewj.basic.multi.patterns.Singleton1@33bb4072
com.niewj.basic.multi.patterns.Singleton1@33bb4072
实例初始化...
com.niewj.basic.multi.patterns.Singleton1@5e99dd12
com.niewj.basic.multi.patterns.Singleton1@33bb4072
com.niewj.basic.multi.patterns.Singleton1@5e99dd12
com.niewj.basic.multi.patterns.Singleton1@5e99dd12
com.niewj.basic.multi.patterns.Singleton1@33bb4072
com.niewj.basic.multi.patterns.Singleton1@5e99dd12
com.niewj.basic.multi.patterns.Singleton1@5e99dd12
可见:多线程getInstance() ……懒汉式…… 可能会产生多个实例
2.-- 饿汉式
package com.niewj.basic.multi.patterns;
/**
* 懒汉式单例
* Created by weijun.nie on 2017/10/12.
*/
public class Singleton2 {
public static int STATUS = 1;
private static Singleton2 instance = new Singleton2(); //只会被初始化一次
private Singleton2() {
System.out.println("实例初始化...");
}
public static Singleton2 getInstance() {
return instance;
}
public static void main(String[] args) {
System.out.println(Singleton2.STATUS);
}
}
实例初始化...
1
可见,饿汉式在未用getInstance() 获取示例之前,即便是使用其他的静态字段,也会实例化构造方法,不好的地方,就是做不到精确控制实例化时间,但另一方面 不存在多线程问题, 这是他的好处。
3. 懒汉式-延迟初始化
public static synchronized Singleton1 getInstance(){...}
但是加锁就会有资源竞争,会有排队,但至少是线程安全。权衡使用。
4. 静态内部类方式-延迟初始化
package com.niewj.basic.multi.patterns;
/**
* 懒汉式-安全延迟
* Created by weijun.nie on 2017/10/12.
*/
public class SingletonBetter {
private SingletonBetter() {
System.out.println("实例初始化...");
}
public static SingletonBetter getInstance() {
return InnerInitUtil.instance;
}
/**
* 静态内部类:
*/
private static class InnerInitUtil {
private static SingletonBetter instance = new SingletonBetter(); //只会被调用一次
}
public static void main(String[] args) {
// 无论取多少次,都是初始化好的。
for (int i = 0; i < 100; i++) {
new Thread(new Runnable() {
public void run() {
System.out.println(SingletonBetter.getInstance());
}
}, "" + i).start();
}
}
}
无论 SingletonBetter.getInstance() 取多少次,InnerInitUtil .Instance 都是InnerInitUtil初次加载的时候初始化一次赋值的。