面向对象六大原则及单例模式

设计模式

面向对象的六大原则

单一职责原则

单一职责原则 (SRP) 是指就一个类而言,应该仅有一个引起它变化的原因

简单而言就是一个类应该只有一项职责,而不是具有多项职责,比如一个类既负责图片缓存的处理同时还负责显示图片,实际上应该拆分成两个类,一个类负责图片的缓存,另外一个类负责图片显示。如果一个类兼具太多的职责不仅导致了耦合性,而且在一个职责发生变化的时候还可能削弱其它的职责功能。

开闭原则

开闭原则 (OCP) 是指软件中的对象对于修改应该是封闭的,对于扩展应该是开放的。

如果一个类为了实现新的功能不断的对类中的原有代码进行修改和增加,不仅可能引入 Bug,还有可能会导致类越来越庞大,比如一个图片的缓存类需要实现内存缓存、SD 卡缓存、两种方式混合的缓存方法,在图片显示类中需要自由选择何种方式进行缓存显示。比较好的一种方式是:由于三种缓存方式实际上基本功能一致,所以可以定义一个接口,然后在图片显示类中义一个接口用于指向三个类实例化的对象,那么当需要采用哪种方式去进行缓存的时候,只需要使用 set 方法进行依赖注入将接口指向相应方式的对象即可,并且如果要实现其它不同的缓存方式只需要对接口进行实现即可。这样实现的代码耦合性弱扩展性强。

里氏替换原则

里氏替换原则是指所有引用基类的地方必须能透明地使用其子类的对象

一个基类的子类拥有基类的属性和方法(私有的除外),所以在大多数情况下基类能干的子类都能做,这样可以保证很好的扩展性,因为可以在基类的基础上进行扩展实现不同功能的子类。因此里氏替换原则有利于提高扩展性,同时为开闭原则提供了保障。

依赖倒置原则

依赖倒置原则是用于解耦的一种方式,主要有以下几个关键点:

  1. 高层模块不应该依赖底层模块,两者都应该依赖其抽象
  2. 抽象不应该依赖细节
  3. 细节应该依赖抽象

第一点是指当高层的模块使用底层的模块时候,不应该直接使用底层模块类的具体对象,而应该使用其接口或者是抽象类,这样可以保证其扩展性,也就是说高层模块与底层模块之间应该通过接口发生联系,而不应该存在直接关联。

接口隔离原则

接口隔离原则是指类间的关系应该建立在最小的接口上

最小的接口实际上就是抽象的一种表达,一个接口下面可能可能会实现很多种接口,或者是很多层级接口,要对这些接口相同的功能部分进行操作的时候只需要对最顶层的接口操作即可,譬如当关闭输入输出流的时候,Java 中有很多种流,字节流、字符流、缓冲流。这个时候为了减少依赖、耦合性以及增加扩展性,我们只需要利用 Cloaseable 接口指向各种流的对象进行关闭操作即可。

迪米特原则

迪米特原则是指一个对象应该对其它对象有最少的了解

一个类应该尽可能少的利用到其它类完成相同的任务,这样可以降低耦合性

单例模式

定义

所谓单例也就是说在一个类在系统中只存在一个实例,并且可以自行实例化向系统提供这个实例

使用场景

适用于某个类有且仅有一个对象的场景,避免创建多个对象消耗过多的资源。

  1. 构造函数不对外开放,一般为 private

  2. 通过一个静态方法或者枚举返回单例类对象

  3. 确保单例类对象有且只有一个,尤其是在多线程环境下

  4. 确保单例类对象在反序列化时不会重新构建对象

也即是说单例模式的对象必须由该类的静态方法进行实例化和提供,并且不能出现多个对象。

优缺点
  1. 优点
  • 单例模式在内存中只存在一个实例,减少了内存的开支。
  • 减少了系统的性能开销,当一个对象的产生需要较多的资源的时候,这个时候可以通过产生一个单例对象,然后永驻内存来解决。
  • 单例模式可以避免对资源的多重占用。
  • 单例模式可以在系统设置全局的访问点,优化和共享资源访问
  1. 缺点
  • 单例模式一般没有接口,扩展很困难。
  • 单例对象如果持有 Context,那么很容易引发内存泄漏,此时传递给单例对象的 Context 最好是 Application Context
常用的实现方式
  1. 饿汉模式
public class CEO {
    private static CEO sCeo = new CEO();

    private CEO() {
        super();
    }

    public static CEO newInstance() {
        return sCeo;
    }
}

该种方式实现的单例模式当类被加载的时候就会初始化一个 CEO 对象,然后外部可以通过 newInstance 静态方法进行获取。

由于单例模式需要类能够自行进行实例化,所以返回值一定是类变量以及通过静态方法进行返回。

  1. 懒汉模式
public class SingleInstance {
    private static SingleInstance sSingleInstance;

    private SingleInstance() {
        super();
    }

    public static synchronized SingleInstance newInstance() {
        if (sSingleInstance != null) {
            sSingleInstance = new SingleInstance();
        }
        return sSingleInstance;
    }
}

采用懒汉模式实现的单例模式可以在使用的时候才将对象实例化,但是由于每次调用 newInstance 方法的时候都会进行同步(比不需要同步的慢 100 倍),所以造成了不必要的同步开销,不建议使用。

  1. Double Check Lock
public class SingleInstance {
    private static SingleInstance sSingleInstance = null;

    private SingleInstance() {
        super();
    }

    public static SingleInstance newInstance() {
        if (sSingleInstance != null) {
            synchronized (SingleInstance.class) {
                if (sSingleInstance != null) {
                    sSingleInstance = new SingleInstance();
                }
            }
        }
        return sSingleInstance;
    }
}

第一次的判断避免了在对象非空情况下进行同步导致不必要开销的问题,第二次判断是由于可能存在线程 A,B 同时判断了对象为空,然后依次进入同步块中,如果这个时候不进行判断则可能导致创建出两个对象出来,所以需要进行第二次判断。

这个模式存在的一个问题是 mSingleInstance = new SingleInstance() 不是原子操作,其分为三个部分:给实例对象分配内存;调用构造函数,初始化成员字段;将实例对象指向分配的内存空间。并且后两步的执行顺序是不确定的,所以可能出现 A 线程执行完第三步,没有执行完第二步的情况下,程序切换至 B 线程,B 线程判断当前对象非空取走对象,但由于对象的成员字段没有初始化完成,所以可能出现错误。

解决办法是在 sInstance 前加上 volatile 关键字。

  1. 静态内部类
public class SingleInstance {
    private SingleInstance() {
        super();
    }

    public static SingleInstance newInstance() {
        return SingleHolder.sSingleInstance;
    }

    private static class SingleHolder {
        private static final SingleInstance sSingleInstance = new SingleInstance();
    }
}

采用这种方式实现的单例模式很好的避免了 DCL 中可能出现的问题,由于内部类只有在使用它的成员以及方法的时候才会进行载入,所以可以做到使用的时候才实例化对象,而且能够确保线程安全。

  1. 枚举单例
public enum SingletonEnum {
    INSTANCE;

    public void doSomething() {
        System.out.println("do sth.");
    }
}

在任何情况下枚举实例都是一个单例,而且创建过程是线程安全的。

  1. 容器实现单例
public class SingletonManager {
    private static Map<String, Object> objectMap = new HashMap<>();

    private SingletonManager() {
    }

    public static void registerService(String key, Object instance) {
        if (!objectMap.containsKey(key)) {
            objectMap.put(key, instance);
        }
    }

    public static Object getService(String key) {
        return objectMap.get(key);
    }
}

采用容器实现的单例模式可以对多种对象的单例进行管理,例如 Android 当中的 getSystemService 就是这样实现的单例模式。

总结
  1. 前四种方式实现的单例模式存在在反序列化(反射执行无参构造函数)的情况下可能会重新创建一个对象,为了避免这种情况的发生,我们需要重写 readResolve 方法,这样在进行反序列化的时候就会执行这个方法获取对象实例。
private Object readResolve()throws ObjectStreamException{
        return sSingleInstance;
}
  1. 单例模式的核心在于将构造函数进行私有化,并且通过一个静态方法返回唯一的对象实例,在这个获取的过程当中需要保证线程安全、防止反序列化导致生成实例对象等问题。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容

  • 单例模式(SingletonPattern)一般被认为是最简单、最易理解的设计模式,也因为它的简洁易懂,是项目中最...
    成热了阅读 4,222评论 4 34
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,585评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,560评论 18 399
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,852评论 6 13
  • 一瞬间失落到极点,各种坏想法涌来,冷静几分钟也就平缓了许多,想法幼稚,这是什么大事。
    违愿阅读 152评论 0 0