再谈Java单例

本文深入探讨了Java中单例模式的各种实现方式,包括最简单的单例、延迟创建、线程安全的延迟创建、双重检查锁定及枚举单例模式。通过对比分析,帮助读者理解每种方式的优缺点及其适用场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

网上流传很多java单例的写法,让人看了很是不解,一个简单的单例为何如此复杂,让人都不敢确定自己写的单例可不可用了。其实很多的变体都是基于线程安全考虑的,之所以觉得复杂,是未能搞清楚应用的场景。绝对的线程安全是要付出代价的,我们只要知道我们写的单例模式能满足我们的并发需求,就可以放心的使用了。

  • 最简单的写法

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

 既然是单例模式,就是为了使用的时候能调用到它,迟早要创建,而且创建了也不会销毁,这点内存,可以承担。

  • 延迟创建 

 

public class Singleton {
    private static Singleton singleton = null;
    private Singleton() {
    }
    public static Singleton getSignleton() {
        if (singleton == null){
            singleton = new Singleton();
        }
        return singleton;
    }
}
  • 考虑线程安全的延迟创建

 

public class Singleton {
    private static Singleton singleton = null;
    private Singleton() {
    }
    public static Singleton getSignleton() {
        synchronized (Singleton.class){
            if (singleton == null){
                singleton = new Singleton();
            }
        }
        return singleton;
    }
}
  • 双重检查的延迟创建

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

有大量线程访问的时候,只要有一个线程执行过了,其他都能正确的拿到值,即使不做同步,而出现并发创建多个对象的场景少之又少,为了满足处女座的情节就用这种吧。

  • 枚举单例模式

public class LockContainer {
    private LockContainer() {
    }

    public static LockContainer getInstance() {
        return LockContainerEnum.INSTANCE.getInstance();
    }

    private enum LockContainerEnum {
        INSTANCE;
        LockContainer container = null;

        LockContainerEnum() {
            container = new LockContainer();
        }

        private LockContainer getInstance() {
            return container;
        }
    }
}

注意细节虽好,但不要被细节所束缚,以至于无法抉择。我喜欢用最简单的和枚举。

 

模式是一种常用的设计模式,它保证一个类仅有一个实,并提供一个访问它的全局访问点。在Java中,实现模式的方式有多种,其中最常用的方式是通过静态方法和私有构造函数来实现。 具体实现步骤如下: 1. 将类的构造函数私有化,使得外部无法通过构造函数来创建实。 2. 在类内部创建一个私有静态变量,用于保存唯一的实。 3. 创建一个静态方法,用于获取唯一的实。在该方法中,如果实不存在,则创建一个实并将其保存到静态变量中,最后返回该实。 下面是一个示代码: ```java public class Singleton { private static Singleton instance; private Singleton() {} public static Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } } ``` 另外,Java中读取XML配置文件的方式也有多种,其中比较常用的方式是使用javax.xml.parsers包中的DocumentBuilder类来解析XML文件。具体实现步骤如下: 1. 创建一个DocumentBuilderFactory对象 2. 调用DocumentBuilderFactory对象的newDocumentBuilder()方法创建一个DocumentBuilder对象 3. 调用DocumentBuilder对象的parse()方法读取XML文件,并返回一个Document对象 4. 通过Document对象的方法获取XML文件中的元素和属性信息 下面是一个示代码: ```java import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.Node; import org.w3c.dom.NodeList; public class XMLParser { public static void main(String[] args) { try { DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder builder = factory.newDocumentBuilder(); Document doc = builder.parse("config.xml"); Element root = doc.getDocumentElement(); NodeList nodeList = root.getChildNodes(); for (int i = 0; i < nodeList.getLength(); i++) { Node node = nodeList.item(i); if (node.getNodeType() == Node.ELEMENT_NODE) { Element element = (Element) node; String name = element.getAttribute("name"); String value = element.getTextContent(); System.out.println(name + " = " + value); } } } catch (Exception e) { e.printStackTrace(); } } } ``` 上面的示代码读取了一个名为config.xml的XML配置文件,并输出了其中所有元素的名称和值。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值