枚举实现单例模式

                                    枚举实现单例模式

 

     枚举单例(Enum Singleton)是实现单例模式的一种新方式,尽管单例模式在java中已经存在很长时间了,但是枚举单例相对来说是一种比较新的概念,枚举这个特性是在Java5才出现的,这篇文章主要讲解关于为什么我们应该使用枚举来实现单例模式,它与传统方式实现的单例模式相比较又有哪些优势?

 

枚举写反简单

      写法简单这是它最大的优点,如果你先前写过单例模式,你应该知道即使有DCL(double checked locking) 也可能会创建不止一个实例,尽管在Java5这个问题修复了(jdk1.5在内存模型上做了大量的改善,提供了volatile关键字来修饰变量),但是仍然对新手来说还是比较棘手。对比通过double checked locking 实现同步,枚举单例那实在是太简单了。如果你不相信那么对比下面代码,分别为传统的用double checked locking实现的单例和枚举单例。

枚举实现:

      下面这段代码就是声明枚举实例的通常做法,它可能还包含实例变量和实例方法,但是为了简单起见,我并没有使用这些东西,仅仅需要小心的是如果你正在使用实例方法,那么你需要确保线程安全(如果它影响到其他对象的状态的话)。默认枚举实例的创建是线程安全的,但是在枚举中的其他任何方法由程序员自己负责。

/**
* Singleton pattern example using Java Enumj
*/
public enum EasySingleton{
    INSTANCE;
}

你可以通过EasySingleton.INSTANCE来访问,这比调用getInstance()方法简单多了。

double checked locking 实现法:

      下面代码就是用double checked locking 方法实现的单例,这里的getInstance()方法要检查两次,确保是否实例INSTANCE是否为null或者已经实例化了,这也是为什么叫double checked locking 模式。

/**
* Singleton pattern example with Double checked Locking
*/
public class DoubleCheckedLockingSingleton{
     private volatile DoubleCheckedLockingSingleton INSTANCE;
 
     private DoubleCheckedLockingSingleton(){}
 
     public DoubleCheckedLockingSingleton getInstance(){
         if(INSTANCE == null){
            synchronized(DoubleCheckedLockingSingleton.class){
                //double checking Singleton instance
                if(INSTANCE == null){
                    INSTANCE = new DoubleCheckedLockingSingleton();
                }
            }
         }
         return INSTANCE;
     }
}

你可以使用 DoubleCheckedLockingSingleton.getInstance()来获取实例。

      从创建一个lazy loaded thread-safe单例来看,它的代码行数与枚举相比,后者可以全部在一行内完成,因为枚举创建的单例在JVM层面上也能保证实例是thread-safe的。

      人们可能会争论有更好的方式去写单例用来替换duoble checked locking 方法,但是每种方法有他自己的优点和缺点,象我很多时候更愿初始化通过类加载静态字段,如下所示,但是记住他不是lazy loaded形式的单例。

      静态工厂实现法:

这是我最喜欢的一种方式来实现单例模式,因为单例是静态的final变量,当类第一次加载到内存中的时候就初始化了,所以创建的实例固然是thread-safe。

/**
* Singleton pattern example with static factory method
*/
 
public class Singleton{
    //initailzed during class loading
    private static final Singleton INSTANCE = new Singleton();
 
    //to prevent creating another instance of Singleton
    private Singleton(){}
 
    public static Singleton getSingleton(){
        return INSTANCE;
    }
}

你可以调用Singleton.getSingleton()获取实例。

枚举自己处理序列化

      传统单例存在的另外一个问题是一旦你实现了序列化接口,那么它们不再保持单例了,因为readObject()方法一直返回一个新的对象就像java的构造方法一样,你可以通过使用readResolve()方法来避免此事发生,看下面的例子:

//readResolve to prevent another instance of Singleton
private Object readResolve(){
    return INSTANCE;
}

     这样甚至还可以更复杂,如果你的单例类维持了其他对象的状态的话,因此你需要使他们成为transient的对象。但是枚举单例,JVM对序列化有保证。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Elasticsearch是一个开源的分布式搜索和分析引擎,用于处理大规模数据集的实时搜索和分析。它基于Lucene库构建而成,提供了强大的全文搜索、分布式性能和可扩展性。 在简单运用Elasticsearch之前,你需要先安装和配置Elasticsearch。安装完成后,你可以使用Elasticsearch提供的RESTful API进行索引数据、搜索和聚合操作。 下面是一个简单的例子,演示如何使用Elasticsearch进行索引和搜索操作: 1. 导入Elasticsearch客户端库(例如elasticsearch-py): ```python from elasticsearch import Elasticsearch ``` 2. 创建Elasticsearch客户端实例: ```python es = Elasticsearch('http://localhost:9200') ``` 3. 创建索引: ```python index_name = 'my_index' mapping = { "properties": { "title": {"type": "text"}, "content": {"type": "text"} } } es.indices.create(index=index_name, body={"mappings": mapping}) ``` 4. 索引文档: ```python document = { "title": "Elasticsearch Introduction", "content": "Elasticsearch is a distributed search engine." } es.index(index=index_name, body=document) ``` 5. 搜索文档: ```python query = { "query": { "match": {"content": "distributed"} } } result = es.search(index=index_name, body=query) print(result['hits']['hits']) ``` 以上是一个简单的示例,演示了如何使用Elasticsearch进行索引和搜索操作。你可以根据自己的需求进一步深入学习和了解Elasticsearch的其他功能和用法。 关于枚举实现单例模式的问题,你可以使用Python中的枚举(Enum)来实现单例模式。以下是一个示例: ```python from enum import Enum class Singleton(Enum): INSTANCE = 1 # 使用方式 instance = Singleton.INSTANCE ``` 在这个示例中,我们定义了一个枚举类Singleton,它只有一个枚举成员INSTANCE。通过访问INSTANCE成员,我们可以获取到单例对象的实例。 需要注意的是,Python的枚举类并不是传统意义上的单例模式,它只是提供了一种简洁明了的方式来实现单例。如果你需要更复杂的单例模式实现,可以考虑使用装饰器、元类或其他方式来实现
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

东方鲤鱼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值