如何使单例模式遇到多线程使安全的,正确的。
1.1单例模式之饿汉模式
饿汉模式(立即加载):就是使用类的时候已经将对象创建完毕,常见的实现办法就是直接new 实例化。
立即加载是在方法调用前,实例已经被创建了,实现代码如下:
package testSingleton;
public class MyObject {
private static MyObject myObject = new MyObject();
private MyObject(){}//将构造器私有
public static MyObject getInstance(){
return myObject;
}
}
package testSingleton;
public class MyThread extends Thread {
@Override
public void run() {
System.out.println(MyObject.getInstance().hashCode());
}
}
package testSingleton;
public class Run {
public static void main(String[] args) {
MyThread myThread1 = new MyThread();
MyThread myThread2 = new MyThread();
MyThread myThread3 = new MyThread();
myThread1.start();
myThread2.start();
myThread3.start();
}
}
运行结果:
1251387847
1251387847
1251387847
三个线程打印出来的hashCode是同一个值,说明对象是同一个,也就实现了立即加载型单例设计模式。
1.2 单例模式之懒汉模式
懒汉模式(延迟加载):就是在调用 getInstance() 方法时实例才被创建,常见的实现办法就是在getInstance() 方法中进行new 实例化。
延迟加载是在调用方法时实例才被创建。实现代码如下:
package testSingleton;
public class MyObject {
private static MyObject myObject;
public static MyObject getInstance(){
if(myObject == null){
myObject = new MyObject();
}
return myObject;
}
}
运行结果:
94165517
303479031
303479031
在多线程中,hashCode值不相同,就会出现去除多个实例的情况,与单例模式的初衷违背。
1.3 如何在多线程环境中保证延迟加载型单例设计模式的正确性。
解决方法1: 声明 synchronized 关键字
public static MyObject getInstance(){
}
直接在getInstance()方法上加上 synchronized关键字,将其变为同步方法。
synchronized public static MyObject getInstance(){
try{
if(myObject == null){
Thread.sleep(3000);//模拟在实例创建之前的一些准备工作
myObject = new MyObject();
}
}catch(InterruptedException e){
e.printStackTrace();
}
return myObject;
}
但是该方法运行效率非常低,是同步运行的,下一个线程想要取得对象,则必须等上一个线程释放之后,才可以继续执行。
解决方法2:使用同步代码块
同步方法是对方法整体进行持锁,这对运行效率来讲是不利的。
同步代码块1:
public static MyObject getInstance(){
try{
synchronized(MyObject.class){
if(myObject == null){
Thread.sleep(3000);//模拟在实例创建之前的一些准备工作
myObject = new MyObject();
}
}
}catch(InterruptedException e){
e.printStackTrace();
}
return myObject;
}
这样的同步代码块也可以实现单例模式,但是和同步方法效率一样低。
同步代码2:只对某些重要的代码进行单独同步
private static MyObject myObject ;
public static MyObject getInstance(){
try {
if(myObject == null){
Thread.sleep(3000);//模拟创建实例前的准备工作
synchronized (MyObject.class) {
myObject = new MyObject();
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
return myObject;
}
运行结果:
94165517
1251387847
303479031
虽然效率提高了,但是并不能实现单例模式,即多线程下仍会得到多个实例对象。
解决方法3:使用DCL双重检查锁机制
使用DCL双重检查锁机制来实现多线程环境中延迟加载单例设计模式。
public static MyObject getInstance(){
try{
if(myObject == null){
Thread.sleep(2000);//用来做创建对象前的一些准备工作
synchronized (MyObject.class) {
if(myObject == null){
myObject = new MyObject();
}
}
}
}catch(InterruptedException e){
e.printStackTrace();
}
return myObject;
}
使用DCL双重检查锁机制,成功解决了“懒汉模式” 遇到多线程的问题。DCL也是大多数多线程结合单例模式使用的解决方案。
1.4 使用静态内置类实现单例模式
public class MyObject{
private static class MyObjectHandler{
public static MyObject myObject = new MyObject();
}
private MyObject(){}
public static MyObject getInstance(){
return MyObjectHandler.myObject;
}
}
同样可以在多线程环境中实现单例模式。但是,如果遇到序列化对象时,使用默认的方式运行得到的结果还是多例的。
情况1:反序列化时使用默认方式,会破坏单例模式
package testSingleton;
import java.io.Serializable;
public class MyObject2 implements Serializable{
private static final long serialVersionUID = 1L;
private static class MyObjectHandler{
public static MyObject2 myObject = new MyObject2();
}
private static MyObject2 myObject2 ;
private MyObject2(){}
public static MyObject2 getInstance(){
return MyObjectHandler.myObject;
}
/*默认模式没有该方法
protected Object readResolve() {
System.out.println("调用了readResolve() 方法");
return MyObjectHandler.myObject;
}
*/
}
package testSingleton;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
public class SaveAndRead {
public static void main(String[] args) {
try{
MyObject2 myObject2 = MyObject2.getInstance();
FileOutputStream out = new FileOutputStream(new File("myObject2.txt"));
ObjectOutputStream oot = new ObjectOutputStream(out);
oot.writeObject(myObject2);
oot.close();
out.close();
System.out.println(myObject2.hashCode());
}catch(FileNotFoundException e){
e.printStackTrace();
}catch(IOException e){
e.printStackTrace();
}
try{
FileInputStream in = new FileInputStream(new File("myObject2.txt"));
ObjectInputStream oi = new ObjectInputStream(in);
MyObject2 myObject2 = (MyObject2)oi.readObject();
System.out.println(myObject2.hashCode());
oi.close();
in.close();
}catch(FileNotFoundException e){
e.printStackTrace();
}catch(IOException e){
e.printStackTrace();
}catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
运行结果:
865113938
1283928880
破坏了单例模式,解决方法就是在MyObject2中加入 readResolve() 方法,并指定要返回对象myObject的生成方式。
情况2: 在Singleton中 定义readResolve() 方法,可以防止反序列化时破坏单利模式
package testSingleton;
import java.io.Serializable;
public class MyObject2 implements Serializable{
private static final long serialVersionUID = 1L;
private static class MyObjectHandler{
public static MyObject2 myObject = new MyObject2();
}
private static MyObject2 myObject2 ;
private MyObject2(){}
public static MyObject2 getInstance(){
return MyObjectHandler.myObject;
}
//加入了readResolve()方法
protected Object readResolve() {
System.out.println("调用了readResolve() 方法");
return MyObjectHandler.myObject;
}
}
运行结果:
865113938
调用了readResolve() 方法
865113938
实现了单例模式
解析:为什么在MyObject2类中加入了readResolve()方法就可以防止反序列化时破坏单例模式呢?
对象序列化过程通过ObjectOutputStream 和ObjectInputStream 来实现,下面看看ObjectInputStream中的readObject() 方法时如何执行的。
这里只给出ObjectInputStream 的readObject() 方法的调用栈:
readObject() ---> readObject0() ---> readOrdinaryObject() --> checkResolve()
下面只给出重点代码块,readOrdinaryObject() 方法的代码:
/**
* Reads and returns "ordinary" (i.e., not a String, Class,
* ObjectStreamClass, array, or enum constant) object, or null if object's
* class is unresolvable (in which case a ClassNotFoundException will be
* associated with object's handle). Sets passHandle to object's assigned
* handle.
*/
private Object readOrdinaryObject(boolean unshared)
throws IOException
{
if (bin.readByte() != TC_OBJECT) {
throw new InternalError();
}
ObjectStreamClass desc = readClassDesc(false);
desc.checkDeserialize();
Class<?> cl = desc.forClass();
if (cl == String.class || cl == Class.class
|| cl == ObjectStreamClass.class) {
throw new InvalidClassException("invalid class descriptor");
}
Object obj;
try {
obj = desc.isInstantiable() ? desc.newInstance() : null;
} catch (Exception ex) {
throw (IOException) new InvalidClassException(
desc.forClass().getName(),
"unable to create instance").initCause(ex);
}
passHandle = handles.assign(unshared ? unsharedMarker : obj);
ClassNotFoundException resolveEx = desc.getResolveException();
if (resolveEx != null) {
handles.markException(passHandle, resolveEx);
}
if (desc.isExternalizable()) {
readExternalData((Externalizable) obj, desc);
} else {
readSerialData(obj, desc);
}
handles.finish(passHandle);
if (obj != null &&
handles.lookupException(passHandle) == null &&
desc.hasReadResolveMethod())
{
Object rep = desc.invokeReadResolve(obj);
if (unshared && rep.getClass().isArray()) {
rep = cloneArray(rep);
}
if (rep != obj) {
// Filter the replacement object
if (rep != null) {
if (rep.getClass().isArray()) {
filterCheck(rep.getClass(), Array.getLength(rep));
} else {
filterCheck(rep.getClass(), -1);
}
}
handles.setObject(passHandle, obj = rep);
}
}
return obj;
}
isInstantiable :如果一个serializable / externalizable 的实现类可以在运行时被实例化,那么该方法就返回true。
desc.newInstance : 该方法通过反射的方式调用无参构造方法金建一个对象。
这就是为什么序列化会破坏单例了。答案:序列化会通过反射调用无参构造方法创建一个新的对象。
如何防止序列化/反序列化破坏单例模式:
防止序列化破坏单例模式: 解决方法就是在Singleton类中定义readResolve() 方法。如上述的例子。
原因:
hasReadResolveMethod() : 如果在实现了serializable 或着 externalizable 接口的类中 包含readResolve() 方法,就返回true;
invokeReadResolve() : 通过反射的方式调用要被反序列化的类中的eadResolve() 方法。
所以:只要在Singleton类中定义readResolve() 方法,并且在该方法中指定要返回对象的生成策略,就可以防止单例模式被破坏了。