单例模式创建的三种方式

一、单例模式的好处

1.对于频繁使用的对象,可以省略new操作花费的时间,这对于那些重量级对象而言,是非常可观的一笔系统开销;

2.由于new操作的次数减少,因而对系统内存的使用频率也会降低,这将减轻GC压力,缩短GC停顿时间。

二、创建方式

1)饿汉式

1 public class Singleton {

2     private Singleton(){

3         System.out.println("Singleton is create”);

4     }

5     private static Singleton instance = new Singleton();

6     public static Singleton getInstance() {

7         return instance;

8     }

9 }

使用以上方式创建单例有几点必须特别注意:

因为我们要保证系统中不会有人意外创建多余的实例,因此

1.我们把Singleton的构造函数设置为private。这点非常重要,这就警告所有的开发人员,不能随便创建这个类的实例,从而有效避免该类被错误的创建。

2.instance对象必须是private并且static的。如果不是private,那么instance的安全性无法得到保证。一个小小的意外就可能使得in-stance变成null。

3.因为工厂方法getInstance()必须是static的,因此对应的instance也必须是static。

优点:这个单例的性能是非常好的,因为getInstance()方法只是简单地返回instance,并没有任何锁操作,因此它在并行程序中,会有    良好的表现。

缺点:Singleton实例在什么时候创建是不受控制的。对于静态成员instance,它会在类第一次初始化的时候被创建。这个时刻并不一定是getInstance()方法第一次被调用的时候。任何对Singleton方法或者字段的引用,都会导致类初始化,并创建instance实例,但是类初始化只有一次,因此instance实例永远只会被创建一次

2)懒汉式

01 public class LazySingleton {

02     private LazySingleton() {

03         System.out.println("LazySingleton is create”);

04     }

05     private static LazySingleton instance = null;

06     public static synchronized LazySingleton getInstance() {

07         if (instance == null)

08             instance = new LazySingleton();

09         return instance;

10     }

11 }

注意点:

1.为了防止对象被多次创建,我们不得不使用synchronized进行方法同步。

优点:精确控制instance的创建时间,它只会在in-stance被第一次使用时,创建对象,这种实现的好处是,充分利用了延迟加载,只在真正需要时创建对象

缺点:并发环境下加锁,竞争激烈的场合对性能可能产生一定的影响。

3)使用内部类

01 public class StaticSingleton {

02     private StaticSingleton(){

03         System.out.println("StaticSingleton is create”);

04     }

05     private static class SingletonHolder {

06         private static StaticSingleton instance = new StaticSingleton();

07     }

08     public static StaticSingleton getInstance() {

09         return SingletonHolder.instance;

10     }

11 }

优点:

同时拥有前两种方式的优点

1.首先getInstance()方法中没有锁,这使得在高并发环境下性能优越

2.只有在getInstance()方法被第一次调用时,StaticSingleton的实例才会被创建(这种方法巧妙地使用了内部类和类的初始化方式)

3.内部类SingletonHolder被申明为private,这使得我们不可能在外部访问并初始化它。而我们只可能在getInstance()内部对SingletonHolder类进行初始化,利用虚拟机的类初始化机制创建单例。

时间: 2024-08-05 10:55:33

单例模式创建的三种方式的相关文章

于Unity3D动态创建对象和创建Prefab三种方式的原型对象

u3d在动态创建的对象,需要使用prefab 和创建时 MonoBehaviour.Instantiate( GameObject orignal) 需要的对象为原型. 文提供三种方式获得prefab对象. 方式一:使用脚本的public字段 直接在Project视图里找到做好的prefab,将其拖拽到指定脚本的指定public GameObject 字段. 方式二:Resource类 1.在Assets目录下的任何位置创建一个名为resources的目录.将做好的prefab放到这个目录下,p

js 类创建的 三种方式

第一种方式:  简单的 JavaScript 类(使用工厂方法) 1. 首先 通过原型继承创建一个对象 第二种方式: 使用构造函数来定义 /**原型对象是类的唯一标识: * 当且仅当两个对象继承自同一个原型对象时, 它们才属于对一个类的实例. * 而构造函数则不能作为类的标识, 两个构造函数的prototype属性可能指向同一个原型对象, * 那么这两个构造函数创建的实例就属于同一个类. * */ function Range(from, to){ this.from = from; this.

控制器创建的三种方式

1 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 2 { 3 // 1.创建UIWindow 4 self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]]; 5 self.window.backgroundColor = [UIC

单例模式实现的三种方式

1.什么是单例模式 单例模式:基于某种方法实例化多次得到实例是同一个2.为何用单例模式 当实例化多次得到的对象中存放的属性都一样的情况,应该将多个对象指向同一个内存,即同一个实例3.如何用''' # 单例模式实现方式一:# import settings## class Mysql:# __instacne=None## def __init__(self,ip,port):# self.ip=ip# self.port=port## @classmethod# def from_conf(cl

05_创建线程三种方式对比

[采用实现Runnable.Callable接口方式创建多线程] [优点]1.线程只是实现了Runable接口或者Callable接口,还可以继承其它类. 2.在这种方式下,多个线程可以共享同一个target对象,比较适合多个相同线程来处理同一份资源的情况. [缺点] 编程略复杂,如果需要访问当前线程,则必须使用Thread currentThread()方法. [采用继承Thread类来创建多线程] [优点] 编写简单,如果需要访问当前线程,只需使用this即可获得当前线程. [缺点] 因为继

49、多线程创建的三种方式之继承Thread类

继承Thread类创建线程 在java里面,开发者可以创建线程,这样在程序执行过程中,如果CPU空闲了,就会执行线程中的内容. 使用Thread创建线程的步骤: 1.自定义一个类,继承java.lang包下的Thread类 2.重写run方法 3.将要在线程中执行的代码编写在run方法中 4.创建上面自定义类的对象 5.调用start方法启动线程 package com.sutaoyu.Thread; //1.自定义一个类,继承java.lang包下的Thread类 class MyThread

python实现单例模式的三种方式及相关知识解释

python实现单例模式的三种方式及相关知识解释 模块模式 装饰器模式 父类重写new继承 单例模式作为最常用的设计模式,在面试中很可能遇到要求手写.从最近的学习python的经验而言,singleton实现的四种方法都是python的重要特征,反过来也刚好是几种特征的最佳实现.(比如你平常开发中很难遇到几个需要写元类的地方)如果不能随手写出某种实现,说明你对于那种实现的概念还没有完全掌握.最近场通过写装饰器模式的singleton来复习装饰器概念. 1. module实现 #模块实现 from

java创建线程的三种方式及其对比

Java中创建线程主要有三种方式: 一.继承Thread类创建线程类 (1)定义Thread类的子类,并重写该类的run方法,该run方法的方法体就代表了线程要完成的任务.因此把run()方法称为执行体. (2)创建Thread子类的实例,即创建了线程对象. (3)调用线程对象的start()方法来启动该线程. package com.thread; public class FirstThreadTest extends Thread{ int i = 0; //重写run方法,run方法的方

Spring创建对象的三种方式以及创建时间

创建对象的三种方式: 1.采用默认的构造函数创建 2.采用静态工厂方法 1.写一个静态工厂方法类 public class HelloWorldFactory { public static HelloWorld getInstance(){ return new HelloWorld(); } } 2.在spring的配置文件中进行声明 <bean id="helloWorld2" class="com.itheima05.spring.bean.HelloWorld