Android 设计模式之面向对象的六大原则

在日常开发过程中时常需要用到设计模式,但是设计模式有23种,如何将这些设计模式了然于胸并且能在实际开发过程中应用得得心应手呢?和我一起跟着《Android源码设计模式解析与实战》一书边学边应用吧!


今天我们要讲的是面向对象的六大原则

单一职责原则

就一个类而言,应该仅有一个引起它变化的原因。简单来说,一个类中应该是一组相关性很高的函数、数据的封装。
  • 我们在App中往往会用到很多的公共方法,比如获取系统的时间,这个功能可能在App中的很多地方都要用到。这个时候我们一般会单独写个工具类TimeUtils,把处理时间有关的方法都放到这个类里面,这样就能减少重复代码,App的结构会更加清晰。当需要添加其他跟时间有关的方法时,就可以都加到这个TimeUtils类里面,这就是我们平时遵循的单一职责原则。

开闭原则

软件中的对象(类、模块、函数等),应该对于扩展是开放的,而对于修改是封闭的。
  • 我们在软件开发过程中就要考虑到后续的扩展和修改。比如说,我们在开发一款类似于universal-image-loader的图片加载框架,可能一开始我们的功能比较简单,图片缓存只有内存缓存。当我们新版本需要添加SD卡缓存时,就要注意尽可能的减少对原来代码的修改,因为这样很可能会引入新的bug。而要做到开闭原则,一般有2种途径,一是通过继承原有的类;二是通过抽象和接口。后面我们会拿书中的图片加载框架来具体说明。

里氏替换原则

所有引用基类的地方必须能透明的使用其子类。通俗的说,就是只要父类能出现的地方子类就可以出现,而且替换为子类以后不会出现任何错误或异常。反过来就不行了,子类出现的地方父类不一定能适应。
  • 要实现里氏替换原则,一般需要一个抽象的父类,父类中定义了子类的公共方法,子类继承或是实现父类以后扩展不同的功能,这样以来可以实现根据不同的需要来应用对应的子类,从而达到应用不同的功能的目的,程序的扩展性大大增强。同时这也体现了开闭原则,即当需要增加新功能时,只要继承或实现父类,实现新增的功能就达到了扩展的目的,而不是直接修改原来的代码,也即对扩展开放,对修改封闭。

依赖倒置原则

依赖倒置原则在Java中的表现就是:模块间的依赖通过抽象发生,实现类之间不发生直接的依赖关系,其依赖关系是通过接口或抽象类产生的。
  • 一句话就是依赖抽象而不依赖具体的实现。比如上面我们说开发一个图片加载框架,那我们肯定会根据单一职责原则划分不同的模块,比如网络加载模块,图片缓存模块,以及主模块等。主模块肯定会调用图片缓存模块,如果我们调用的是图片缓存模块的具体实现,那么当我们修改图片模块时就很可能要对应修改主模块,这就是耦合了。一个比较好的做法是将图片缓存模块抽象出来,而主模块调用这个抽象即可,这样也就是依赖抽象了。

接口隔离原则

类间的依赖关系应该建立在最小的接口上。
  • 接口隔离原则就是让客户端依赖的接口尽可能的小。就是在上面提到的依赖倒置(依赖抽象而不是实现)原则的基础上,增加一个最小化依赖的原则。说白了就是在依赖接口的基础上依赖尽可能少的接口。
  • 这里举个例子:
<!--将图片写入SD卡-->
public void put(String url, Bitmap bitmap) {
    FileOutputStream fileOutputStream = null;
    try {
        fileOutputStream = new FileOutputStream(SDPath+url);
        bitmap.compress(Bitmap.CompressFormat.PNG, 100, fileOutputStream);
    }catch (FileNotFoundException e) {
        e.printStackTrace();
    }finally {
        CloseUtils.closeQuietly(fileOutputStream);
    }
}

<!--关闭工具类-->
public final class CloseUtils {

    private CloseUtils() { }

    /**
     * 关闭Closeable对象
     * @param closeable
     */
    public static void closeQuietly(Closeable closeable) {
        if (null != closeable) {
            try {
                closeable.close();
            }catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}
  • 上述例子中的CloseUtils的closeQuietly方法的原理就是依赖于Closeable抽象而不是具体实现(依赖倒置原则),并且建立在最小化依赖的基础上,它只要知道这个对象是可以关闭的就行了,其他的一概不用关心,这就是接口隔离原则

迪米特原则

一个对象应该对其他的对象有最少的了解
  • 通俗的讲,一个类应该对自己需要耦合或调用的类知道得最少,调用者或是依赖者只要知道它需要的方法即可。要做到这个原则,需要我们对各个模块之间的功能进行很好的区分和分配,把相互之间的依赖和耦合减到最少。

以上就是面向对象的六大原则。

下面我们通过书中的图片加载框架ImageLoader的例子介绍这些原则的具体应用。
  • 首先是ImageLoader类
public class ImageLoader {
    // 图片缓存,依赖接口,而不是具体实现
    // 如果改为MemoryCache mImageCache = new MemoryCache();就不能定制图片缓存的实现,扩展性大大降低,耦合度也会大大提高
    ImageCache mImageCache = new MemoryCache();
    // 线程池,线程数量为CPU的数量
    ExecutorService mExecutorService = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());

    // 注入缓存,对扩展开放,对修改关闭
    public void setImageCache(ImageCache cache) {
        mImageCache = cache;
    }

    /**
     * 显示图片
     * @param imageUrl
     * @param imageView
     */
    public void displayImage(String imageUrl, ImageView imageView) {
        Bitmap bitmap = mImageCache.get(imageUrl);
        if (bitmap != null) {
            imageView.setImageBitmap(bitmap);
            return;
        }
        // 图片没有缓存,提交到线程池下载
        submitLoadRequest(imageUrl, imageView);
    }

    /**
     * 下载图片
     * @param imageUrl
     * @param imageView
     */
    private void submitLoadRequest(final String imageUrl, final ImageView imageView) {
        imageView.setTag(imageUrl);
        mExecutorService.submit(new Runnable() {
            @Override
            public void run() {
                Bitmap bitmap = downloadImage(imageUrl);
                if (bitmap == null) {
                    return;
                }
                if (imageUrl.equals(imageView.getTag())) {
                    imageView.setImageBitmap(bitmap);
                }
                mImageCache.put(imageUrl, bitmap);
            }
        });
    }

    /**
     * 下载图片
     * @param imageUrl
     * @return
     */
    private Bitmap downloadImage(String imageUrl) {
        Bitmap bitmap = null;
        try {
            URL url = new URL(imageUrl);
            HttpURLConnection connection = (HttpURLConnection) url.openConnection();
            bitmap = BitmapFactory.decodeStream(connection.getInputStream());
            connection.disconnect();
        }catch (Exception e) {
            e.printStackTrace();
        }
        return bitmap;
    }
  • 然后是图片缓存接口以及内存缓存
<!--图片缓存接口-->
public interface ImageCache {
    public Bitmap get(String url);
    public void put(String url, Bitmap bitmap);
}

<!--内存缓存的实现-->
public class MemoryCache implements ImageCache{
    private LruCache<String, Bitmap> mMemoryCache;

    public MemoryCache() {
        //初始化LRU缓存
        initImageCache();
    }
    
    private void initImageCache() {
        // 计算可使用的最大内存
        final int maxMemory = (int) (Runtime.getRuntime().maxMemory()/1024);
        // 取四分之一的可用内存作为缓存
        final int cacheSize = maxMemory / 4;
        mMemoryCache = new LruCache<String, Bitmap>(cacheSize) {
            @Override
            protected int sizeOf(String key, Bitmap bitmap) {
                return bitmap.getRowBytes() * bitmap.getHeight() / 1024;
            }
        };
    }


    @Override
    public Bitmap get(String url) {
        return mMemoryCache.get(url);
    }

    @Override
    public void put(String url, Bitmap bitmap) {
        mMemoryCache.put(url, bitmap);
    }
}
ImageLoader类中的逻辑比较直观,六大原则的体现主要在图片缓存的处理上。
  • 将图片缓存单独出去而不是写在ImageLoader一起,体现了单一职责原则
  • ImageLoader中依赖的是图片缓存的接口而不是具体的实现,体现了开闭原则、里氏替换原则和依赖倒置原则。
  • ImageLoader类和MemoryCache类之间只依赖ImageCache接口,也可以说体现了接口隔离原则
  • ImageLoader类只需要知道MemoryCache类的put方法和get方法,其他的实现一概不管,也体现了迪米特原则
当然以上的ImageLoader还只是初版,还有很多有待优化的地方,比如可以把下载的逻辑单独出去,可以增加更多的定制功能等。

六大原则能让我们的代码结构更加合理,扩展性更好,让程序更稳定灵活。这些原则往往都是共同作用的,比如上面例子中的CloseUtils满足了单一职责原则,closeQuietly方法运用了依赖倒置,并且遵循接口隔离原则。赶紧用六大原则优化你的代码吧!!!
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容