教你借助设计模式限制实例化数量 | 带你学《Java面向对象编程》之七十三

简介: 开发过程中,如果遇到了某些要求,需要限制实例化类的数量时,需要借助本节介绍到的单例与多例设计模式来实现。

上一篇:带你领略类图以外的世界 | 带你学《Java面向对象编程》之七十二
【本节目标】
通过阅读本节内容,你将进一步利用所学知识来一步步实现某些开发要求,了解到单例模式与多例模式的实现原理与作用。

单例设计模式(多例设计模式)主要是一种控制实例化对象产生个数的设计操作。

单例设计

如果说有一个程序类,假设该程序类的定义如下:

class Singleton {
    public void print() {
        System.out.println("www.mldn.cn") ;
    }
}
public class JavaDemo{
    public static void main(String args[]) {
        Singleton instanceA = new Singleton() ;
        Singleton instanceB = new Singleton() ;
        Singleton instanceC = new Singleton() ;
        instanceA.print() ;
        instanceB.print() ; 
        instanceC.print() ; 
    }
}

但是由于某些要求,现在要求Singleton这个类只允许提供有一个实例化对象。那么此时首先控制构造方法因为所有的新的实例华对象产生了,那么一定要调用我们的构造方法,如果“没有”构造方法,那么就自然无法产生实例化对象了。
范例:构造方法私有化

class Singleton {
    private Singleton() {}    //构造方法私有化
    public void print() {
        System.out.println("www.mldn.cn") ;
    }
}
public class JavaDemo{
    public static void main(String args[]) {
        Singleton  instance = null ;    //声明对象
        instance = new Singleton() ;   //错误:Singleton()可以在Singleton中访问private
    }
}

但是现在是有严格要求的:必须产生有一个实例化对象。所以现在需要考虑去产生一个实例化对象交给客户端去调用。那么这个时候的分析如下:
1、private访问权限的主要特点在于:不能在类外部访问,但是可以在类本身调用,所以现在可以考虑在类的内部调用构造;

class Singleton {
    private Singleton instance = new Singleton() ;
    private Singleton() {}    //构造方法私有化
    public void print() {
        System.out.println("www.mldn.cn") ;
    }
}
public class JavaDemo{
       public static void main(String args[]) {
            Singleton  instance = null ;    //声明对象
            //instance = new Singleton() ;   //错误:Singleton()可以在Singleton中访问private
       }
}

2、此时Singleton类内部的instance属于一个普通属性,而普通属性是在有实例化对象产生之后才会被调用的,那么这个时候外部无法产生实例化对象,所以这个属性就不能访问到了,那么就必须要考虑如何在没有实例化对象的时候获取此属性,那么只有一个static属性可以访问。

class Singleton {
    static Singleton instance = new Singleton() ;
    private Singleton() {}    //构造方法私有化
    public void print() {
        System.out.println("www.mldn.cn") ;
    }
}
public class JavaDemo{
       public static void main(String args[]) {
            Singleton  instance = null ;    //声明对象
            instance = Singleton.instance ;   
            instance.print() ;   //www.mldn.cn
       }
}

3、类中的属性应该封装后使用,所以理论上此时的instance需要被封装起来,那么就需要通过static方法获得。

class Singleton {
    private static Singleton instance = new Singleton() ;
    private Singleton() {}    //构造方法私有化
    public static Singleton getInstance() {
          return instance ;
    }
    public void print() {
        System.out.println("www.mldn.cn") ;
    }
}
public class JavaDemo{
       public static void main(String args[]) {
            Singleton  instance = null ;    //声明对象
           instance = Singleton.getInstance() ;   
            instance.print() ;       //www.mldn.cn

       }
}

4、整个代码从头强调的是只有一个实例化对象,但是这个对象依然可以被重新使用。所以需要保证此时Singleton类内部的instance无法再次实例化,那么应该使用final定义。

class Singleton {
    private static final Singleton INSTANCE = new Singleton() ;
    private Singleton() {}    //构造方法私有化
    public static Singleton  getInstance() {
          return INSTANCE ;
    }
    public void print() {
        System.out.println("www.mldn.cn") ;
    }
}
public class JavaDemo{
       public static void main(String args[]) {
            Singleton  instance = null ;    //声明对象
           instance = Singleton.getInstance() ;   
            instance.print() ;       //www.mldn.cn

       }
}

在很多情况下有些类是不需要重复产生对象的,例如:如果一个程序启动,那么现在肯定需要有一个类负责保存有一些程序加载的数据信息。

image.png
图一 单例设计

对于单例设计模式也分为两种:懒汉式、饿汉式。在之前所定义的都属于饿汉式。在系统加载类的时候就会自动提供Singleton类的实例化,而懒汉式是在第一次使用的时候进行实例化对象处理。

范例:将单例修改为懒汉式

class Singleton {
    private static Singleton instance  ;
    private Singleton() {}    //构造方法私有化
    public static Singleton  getInstance() {
          if(instance == null) {    //第一次使用
              instance = new Singleton() ;       //实例化对象
          }
          return instance ;
    }
    public void print() {
        System.out.println("www.mldn.cn") ;
    }
}
public class JavaDemo{
       public static void main(String args[]) {
            Singleton  instance = null ;    //声明对象
           instance = Singleton.getInstance() ;   
            instance.print() ;       //www.mldn.cn
       }
}

面试题:请编写一个Singleton程序,并说明其主要特点?

  • 代码如上,可以把懒汉式(后面需要考虑到线程同步问题)和饿汉式都写上;
  • 特点:构造方法私有化,类内部提供static方法获取实例化对象,这样不管外部如何操作,永远都只有一个实例化对象提供。

多例设计

与单例设计对应的还有一个多例设计,单例设计指的是只保留有一个实例化对象,而多例设计指的是可以保留有多个实例化对象。例如:如果现在定义一个描述性别的类。那么该对象只有两个:男、女。或者描述颜色基色的类,可以使用红色、绿色、蓝色。这种情况下就可以使用多例设计来解决。
范例:实现多例设计

class Color {    //定义描述颜色的类
    private static final Color RED= new Color("红色") ;
    private static final Color GREEN= new Color("绿色") ;
    private static final Color BLUE= new Color("蓝色") ;
    private String title;
    private Color(String title) {     //构造方法私有化
         this.title = title ;
    }
    public static Color getInstance(String color) {
        switch(color) {
            case "red": return RED ;
            case "green": return GREEN ;
            case "blue": return BLUE ;
            default: return null ;
        }
    }
    public String toString() {
        return this.title ;
    } 
}
public class JavaDemo{
   public static void main(String args[]) {
       Color c =Color.getInstance("green") ;
       System.out.println(c) ;
   }      //绿色
}

多例设计与单例设计的本质是相同的,一定都会在内部提供有static方法以返回实例化对象。

想学习更多的Java的课程吗?从小白到大神,从入门到精通,更多精彩不容错过!免费为您提供更多的学习资源。
本内容视频来源于阿里云大学

下一篇:迅速了解多例模式竞争者-枚举 | 带你学《Java面向对象编程》之七十四
更多Java面向对象编程文章查看此处

相关文章
|
26天前
|
设计模式 消息中间件 搜索推荐
Java 设计模式——观察者模式:从优衣库不使用新疆棉事件看系统的动态响应
【11月更文挑战第17天】观察者模式是一种行为设计模式,定义了一对多的依赖关系,使多个观察者对象能直接监听并响应某一主题对象的状态变化。本文介绍了观察者模式的基本概念、商业系统中的应用实例,如优衣库事件中各相关方的动态响应,以及模式的优势和实际系统设计中的应用建议,包括事件驱动架构和消息队列的使用。
|
1月前
|
Java 开发者
Java 面向对象编程
总之,Java 的面向对象编程为开发者提供了一种有效的编程范式,帮助他们构建出高质量、可维护的软件系统。理解和掌握面向对象的概念和原则是成为优秀 Java 开发者的重要基础。
48 3
|
1月前
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
38 4
|
1月前
|
设计模式 缓存 数据库连接
深入理解PHP的面向对象编程与设计模式####
本文旨在通过实例解析PHP中面向对象编程的核心概念及其在实际开发中的应用,特别是单例模式、工厂模式和观察者模式等常见设计模式。文章将展示这些模式如何帮助开发者构建更加灵活、可维护的代码结构,以及它们在解决具体问题时的优势。 ####
|
2月前
|
设计模式 Java 程序员
[Java]23种设计模式
本文介绍了设计模式的概念及其七大原则,强调了设计模式在提高代码重用性、可读性、可扩展性和可靠性方面的作用。文章还简要概述了23种设计模式,并提供了进一步学习的资源链接。
54 0
[Java]23种设计模式
|
1月前
|
设计模式 JavaScript Java
Java设计模式:建造者模式详解
建造者模式是一种创建型设计模式,通过将复杂对象的构建过程与表示分离,使得相同的构建过程可以创建不同的表示。本文详细介绍了建造者模式的原理、背景、应用场景及实际Demo,帮助读者更好地理解和应用这一模式。
|
2月前
|
设计模式 监控 算法
Java设计模式梳理:行为型模式(策略,观察者等)
本文详细介绍了Java设计模式中的行为型模式,包括策略模式、观察者模式、责任链模式、模板方法模式和状态模式。通过具体示例代码,深入浅出地讲解了每种模式的应用场景与实现方式。例如,策略模式通过定义一系列算法让客户端在运行时选择所需算法;观察者模式则让多个观察者对象同时监听某一个主题对象,实现松耦合的消息传递机制。此外,还探讨了这些模式与实际开发中的联系,帮助读者更好地理解和应用设计模式,提升代码质量。
Java设计模式梳理:行为型模式(策略,观察者等)
|
2月前
|
设计模式 Java
Java设计模式
Java设计模式
37 0
|
1天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
3天前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。

热门文章

最新文章