Java单例设计模式,多线程下同步

单例模式的特点:

  1. 只创建一次
  2. 私有的属性
  3. 公有的访问方法

单例模式的分类:

  1. 懒汉(LazySingleton);默认不自动实例化,等到用的时候根据当前情况实例化,并且返回当前对象;
  2. 恶汉(EagerSingleton);在类第一次加载的时候强制实例化;

用处:

         一般用在配置文件的读取,有些东西只需要加载一次,以后重复使用。

示例代码:

package myThread;  
  
/** 
 * 单例模式,懒汉模式 
 *  
 * @author Champion.Wong 
 *  
 */  
public class LazySingleton {  
  
    private static LazySingleton singleton = null;  
  
    /** 
     * 私有的构造方法,防止被多次实例化 
     *  
     */  
    private LazySingleton() {  
    }  
  
    /** 
     * 静态工厂,返回此类的唯一一个实例 判断,如果没有被实例化,则先初始化再返回 
     *  
     * @return LazySingleton 
     */  
    public static LazySingleton getInstance() {  
        if (null == singleton)  
            singleton = new LazySingleton();  
        return singleton;  
    }  
}  


 

package myThread;  
  
/** 
 * 单例模式,饿汉模式 
 * @author Champion.Wong 
 * 
 */  
public class EagerSingleton {  
  
    private static EagerSingleton singleton = new EagerSingleton(); // 强制初始化,在类加载的时候就创建好了对象   
  
    /** 
     * 私有的构造方法,防止被多次实例化 
     *  
     */  
    private EagerSingleton() {  
    }  
  
    public static EagerSingleton getInstance() {  
        return singleton;  
    }  
}  


 

 

前边说过了,单例只创建一次,来做个测试,在单例的类中加入一个属性,当这个类在实例化的时候,会将当前的时间赋给这个属性。我们创建两个单例类的对象,然后分别拿出它们类中的这个属性进行对比。

所以,在EagerSingleton中加入属性:createdTime

 

package myThread;  
  
/** 
 * 单例模式,饿汉模式 
 * @author Champion.Wong 
 * 
 */  
public class EagerSingleton {  
  
    private static EagerSingleton singleton = new EagerSingleton(); // 强制初始化,在类加载的时候就创建好了对象   
  
    /** 
     * 私有的构造方法,防止被多次实例化 
     *  
     */  
    private EagerSingleton() {  
        createdTime = System.currentTimeMillis();  
    }  
  
    public static EagerSingleton getInstance() {  
        return singleton;  
    }  
      
    private long createdTime = 0L; //此类的创建时间,在类实例化的时候赋值   
    public long getCreatedTime() {  
        return createdTime;  
    }  
}  


 

然后,写个Test:

 

package myThread;  
  
public class Test_Singleton {  
  
    public static void main(String[] args) {  
        EagerSingleton singleton1 = EagerSingleton.getInstance();  
        try {  
            Thread.sleep(2000); //让当前线程睡眠2000ms,这样,两个对象实例化的时间就差2s   
        } catch (InterruptedException e) {  
            e.printStackTrace();  
        }  
        EagerSingleton singleton2 = EagerSingleton.getInstance();  
          
        System.out.println("Singleton1:" + singleton1.getCreatedTime());  
        System.out.println("Singleton2:" + singleton2.getCreatedTime());  
    }  
  
}  


 

最后控制台输出的内容为:

两个对象的实例化的时间相同,证明单例设想成功!


当多个线程访问单例类的时候,必须要保证同时只能由一个线程访问,当这个线程访问结束了,其它线程才能访问。这是就要给单例的访问方法加个同步锁(synchronized)。

修改后的LazySingleton为:

 

package myThread;  
  
/** 
 * 单例模式,懒汉模式 
 *  
 * @author Champion.Wong 
 *  
 */  
public class LazySingleton {  
  
    private static LazySingleton singleton = null;  
  
    /** 
     * 私有的构造方法,防止被多次实例化 
     *  
     */  
    private LazySingleton() {  
    }  
  
    /** 
     * 静态工厂,返回此类的唯一一个实例 判断,如果没有被实例化,则先初始化再返回 
     *  
     * @return LazySingleton 
     */  
    synchronized public static LazySingleton getInstance() {  
        if (null == singleton)  
            singleton = new LazySingleton();  
            System.out.println(Thread.currentThread().getName() + System.currentTimeMillis());  
        try {  
            Thread.sleep(900); //假设每个线程访问的时候都需要耗费一定的时间   
        } catch (InterruptedException e) {  
            e.printStackTrace();  
        }  
        return singleton;  
    }  
}  


 

做个测试,起两个线程,没个线程中分别调用单例类的访问方法,每次访问的时候输出访问的时间,判断时间是否相等。如果相等,同步失败,如果不等,同步成功!

 

package myThread;  
  
public class Test_Thread_Singleton {  
  
    /** 
     * @param args 
     */  
    public static void main(String[] args) {  
        SingletonThread st = new SingletonThread();  
        SingletonThread st1 = new SingletonThread();  
        st.start();  
        st1.start();  
    }  
  
}  
  
class SingletonThread extends Thread {  
    private LazySingleton singleton = null;  
    public void run() {  
        while (!this.isInterrupted()) {  
            try {  
                Thread.sleep(1000);  
            } catch (InterruptedException e) {  
                e.printStackTrace();  
            }  
            singleton = LazySingleton.getInstance();  
        }  
    }  
}  


 

看控制台的输出内容:

时间差不多都不一致,同步成功!限制了同时访问。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值