美文网首页@IT·互联网程序员
单线程/多线程下的单例

单线程/多线程下的单例

作者: 小甜妮子__ | 来源:发表于2017-04-26 10:47 被阅读176次

    1.一般的单例(单线程下的单例):
    <pre>
    /**

    • 单线程下单例--延迟实例化(lazy)
    • @author wangzhaoli

    */
    public class SettingManager {
    // 静态私有实例
    private static SettingManager settingManager;
    // 私有构造方法
    private SettingManager(){

    }
    // 提供静态共有方法获取唯一实例
    public static SettingManager getInstance(){
        if(settingManager == null){
            settingManager = new SettingManager();
        }
        return settingManager;
    }
    

    }
    </pre>

    2.多线程下使用上述的单例会有严重的问题,一下一一说明多线程下单例的使用方式
    <pre>
    /**

    • 多线程下单例1--synchronized
    • 优点:简单
    • 缺点:严重降低性能;且实际上只有首次实例化对象才需要synchronized
    • @author wangzhaoli

    */
    public class SettingManager {
    private static SettingManager settingManager;
    private SettingManager(){

    }
    // synchronized的作用:每个线程在调用该方法之前要等待其他线程的退出
    public static synchronized SettingManager getInstance(){
        if(settingManager == null){
            settingManager = new SettingManager();
        }
        return settingManager;
    }
    

    }
    </pre>

    <pre>
    /**

    • 多线程下单例2--急切实例化(eagerly)
    • 优点:简单
    • 缺点:对于一次使用几率很小的单例,每次程序运行都启动,并存在整个应用生命周期,会造成不必要的资源浪费
    • @author wangzhaoli

    */
    public class SettingManager {
    private static SettingManager settingManager = new SettingManager();//在JVM类加载时就创建实例,此时任何线程未启动
    private SettingManager(){

    }
    public static SettingManager getInstance(){
        return settingManager;
    }
    

    }
    </pre>

    <pre>
    /**

    • 多线程下单例3--双重检查加锁(考虑性能的有效单例)
    • 优点:同样是使用synchronized,但是只有首次实例化的时候才会使用synchronized,一旦存在实例,不再使用synchronized,提高性能
    • 缺点:繁琐一点
    • @author wangzhaoli

    */
    public class SettingManager {
    // 用volatile修饰的变量,线程在每次使用变量的时候,都会读取变量修改后的最的值。volatile只在JDK1.4及以后的版本可用
    private volatile static SettingManager settingManager;
    private SettingManager(){

    }
    public static SettingManager getInstance(){
        if(settingManager == null){
            // 保证synchronized只是用一次,保证了性能
            synchronized (SettingManager.class) {
                if(settingManager == null){
                    settingManager = new SettingManager();
                }
            }
        }
        return settingManager;
    }
    

    }
    </pre>

    最后说明,区分类的单例和对象的单例:
    类的单例--静态方法调用,适用于一些工具类,引用能够自给自足;
    对象的单例--除了单例对象本身,其他的方法都是有单例对象调用,适用于依赖一些其他类的情况。

    相关文章

      网友评论

        本文标题:单线程/多线程下的单例

        本文链接:https://www.haomeiwen.com/subject/rhcizttx.html