Android中的设计模式之单例模式
参考
- 《设计模式:可复用面向对象软件的基础 》3.5 Singleton 单件--对象创建型模式
- 《设计模式解析》(第二版)第21章 Singleton模式和Double-Checked Locking模式
- 《Android源码设计模式解析与实战》第2章 单例模式
意图
保证一个类仅有一个实例,并提供一个访问它的全局访问点。
动机
对一些类来说,只有一个类实例是很重要的,比如一个公司CEO只有一位,一个帝国只有一个皇帝。
单例模式是应用最广的模式之一,单例对象的类必须保证只有一个实例存在。如在一个应用中,应该只有一个ImageLoader实例,这个ImageLoader中又含有线程池,缓存系统,网络请求等,很消耗资源,因此,没有理由让它构造多个实例。
保持只有一个单例,好的办法是让类自身负责保存它的唯一实例。
适用性
- 当类只能有一个实例而且客户可以从一个众所周知的访问点访问它。
- 当这个唯一实例应该是通过子类化可扩展的,并且客户应该无需更改代码就能使用一个扩展的实例。
定义
确保一个类只有一个实例,而且自行实例化并向整个系统提供这个实例
角色
- Client 高层客户端
- Singleton 单例类
Java实现单例模式的关键点
- 构造函数不对外开放,一般为private;
- 通过一个静态方法或者枚举返回单例类对象;
- 确保单例类的对象有且只有一个,尤其是在多线程环境下;
- 确保单例类对象在反序列化时不会重新构建对象;
实现方式
饿汉模式
public class Singleton { private void Singleton(){} private static Singleton instance =new Singleton(); public static Singleton getInstance(){ return instance; } }
加载类的时候就构造了单例,有个问题是万一客户端一直都不用这个单例呢,岂不是浪费资源
懒汉模式
public class Singleton { private void Singleton(){} private static Singleton instance; public static synchronized Singleton getInstance(){ if (instance==null) { instance=new Singleton(); } return instance; } }
这里getInstance()方法添加了synchronized关键字,也就是getInstance()是一个同步方法,目的是保证在多线程情况下单例对象唯一的手段。这里会有个问题,即使instance已经被初始化,每次调用getInstance()方法都会进行同步,这样会消耗不必要的资源。
Double Check Lock(DCL)实现单例
public class Singleton { private void Singleton(){} private static Singleton instance=null; public static Singleton getInstance(){ if (instance==null) { synchronized (Singleton.class){ if(instance==null){ instance=new Singleton(); } } } return instance; } }
因为getInstance()可能同时会有多个客户端调用,DCL方式实现单例模式的优点是既能在需要时才初始化单例,又能保证线程安全,且单例对象初始化调用后调用getInstance不进行同步锁。
- 第一次检查单例有没有实例化, getInstance()并没有同步,所以不会太耗时,如果为空则进入第二次检查,否则返回实例
- 第二次检查就是构造实例了,这次必须同步,就是在同步的情况下,判断实例为不为空,为空就构造。
这种实现方式很流行,很多框架都用这种
静态内部类单例模式
public class Singleton { private void Singleton(){} public static Singleton getInstance(){ return SingletonHolder.sInstance; } /** * 静态内部类 */ private static class SingletonHolder{ private static final Singleton sInstance=new Singleton(); } }
DCL虽然在一定程度上解决了资源消耗,多余的同步,线程安全等问题,但是,他还是在某些情况下出项失效的问题。
静态内部类单例模式,当第一次加载Singleton类时并不会初始化sInstance,只有在第一次调用Singleton的getInstance方法时才会导致sInstance被初始化。因此,第一次调用getInstance方法会导致虚拟机加载SinglentonHolder类,这种方式不仅能确保线程安全,也能够保证单例对象的唯一性,同时也延迟了单例的实例化,所以这个是推荐使用的单例模式实现方式。
枚举单例
public enum SingletonEnum { INSTANCE; public void dosomeThing(){ System.out.println("do sth."); } }
《Effective Java》推荐的用法,用枚举
使用容器实现单例模式
public class SingletonManager { private static Map<String,Object> objMap=new HashMap<String,Object>(); private SingletonManager(){} public static void registerService(String key,Object instance){ if (!objMap.containsKey(key)) { objMap.put(key, instance); } } public static Object getService(String key){ return objMap.get(key); } }
在程序的初始,将多种单例类型注入到一个统一的管理类中,在使用时根据key获取对象对应类型的对象。
优点
- 减少内存开支,特别是在一个对象频繁的创建和销毁时。
- 单例模式可以避免对资源的多重占用
- 单例模式可以在系统中设置全局的访问点,优化和共享资源访问
缺点
- 单例模式一般没有接口,扩展很困难
- 单例对象如果持有Context,容易发生内容泄露,最少congtext最好是Application级别的
应用例子1 Android开发封装的OkHttp3 Client单例
// todo 这个很常见了,后面贴个代码
应用例子2 Android中的LayoutInflater
// todo 待研究,后面更
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Android 使用Sqlite
如何使用 继承SQLiteOpenHelper,通过SQLiteOpenHelper可以方便的管理SQLiteDatabase public class MyDataBaseHelper extends SQLiteOpenHelper { public final static String CREATE_TABLE_SQL="create table dict (_id integer primary key autoincrement,word text,detail text)"; public MyDataBaseHelper(Context context, String name, int version) { super(context, name, null, version); } @Override public void onCreate(SQLiteDatabase db) { // 第一次使用数据库是自动建表 db.execSQL(CREATE_TABLE_SQL); } /** * 升级数据是调用 */ @Override public void onUp...
- 下一篇
3.6 自定义View (3.6.1)
本文对应项目的码云地址:https://gitee.com/wanchuanxy/AndroidHeroesTest/tree/master/3/SystemWidget Android给我们提供了丰富的组件库来创建丰富的UI效果,同时也提供了非常方便的拓展方法。通过继承Android的系统组件,我们可以非常方便地拓展现有功能,在系统组件的基础上创建新的功能,甚至可以直接自定义一个控件,实现Android系统控件所没有的功能。自定义控件作为Android中一个非常重要的功能,一直以来都被初学者认为是代表高手的象征。其实,自定义View并没有想象中的那么难,与其说是在自定义一个View,不如说是在设计一个图形,只有站在一个设计者的角度上,才可以更好地创建自定义View。我们不能机械地记忆所有绘图的API,而是要让这些API为你所用,结合现实中绘图的方法,甚至是PhotoShop的技巧,才能设计出更好的自定义View。 适当地使用自定义View,可以丰富应用程序的体验效果,但滥用自定义View则会带来适得其反的效果。一个让用户觉得熟悉得控件,才是一个好的控件。如果一味追求炫酷的效果而创建自...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
-
Docker使用Oracle官方镜像安装(12C,18C,19C)
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- CentOS8编译安装MySQL8.0.19
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- MySQL8.0.19开启GTID主从同步CentOS8
- CentOS7,8上快速安装Gitea,搭建Git服务器
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
推荐阅读
最新文章
- Linux系统CentOS6、CentOS7手动修改IP地址
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- CentOS6,CentOS7官方镜像安装Oracle11G
- SpringBoot2整合Redis,开启缓存,提高访问速度
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- Hadoop3单机部署,实现最简伪集群