第一章 走向灵活软件之路-----面向对象的六大原则

(单一原则)一个类而言,应该仅有一个引起它变化的原因。简单来说,一个类应该是一组相关性很高的函数、数据的封装。就想秦小波在<设计模式之禅>中说的:“这是一个备受争议却又及其重要的原则。只要你想和别人争执、怄气或者吵架,这个原则屡试不爽的”。因为单一原则的划分界限并不是总是难么清晰,很多时候都是需要靠个人经验来界定。当然,最大的问题就是对职责的定义,什么是类的职责,以及怎么划分类的职责。

对于计算机技术,通常只单纯地学习理论知识并不能很好地领会其深意,只有自己动手实践,并在实际运用中发现问题、解决问题、思考问题,才能够将知识吸收到自己的脑海中。下面以我的朋友小明的事情说起。

自从Android系统发布以来,小明是Android的铁杆粉丝,于是在大学期间一直保持着对Android的关注,并且利用课余时间做一些小项目,锻炼自己的实战能力。毕业后,如愿地加入了心仪的公司,并且投入到了自己热爱的Android开发行业中、将爱好、生活、事业融为一体,小眀的第一份工作也算是顺风顺水,一切尽在掌握中。

在经历过一周的适应期以及熟悉公司产品、开发规范之后,开发工作就正式开始了。小明的主管是一个经验丰富的技术专家,对于小明的工作并不是很满意,尤其是最薄弱的面向对象设计,而Android开发又是使用Java语言,程序中的抽象、接口、六大原则、23中设计模式等名词把小明弄得晕头转向,自己也就查到自己的问题所在。于是主管决定先让小明做一个小项目来锻炼这方面的能力。正所谓养兵千日用兵一时,磨刀不误砍柴工。小明的开发之路刚刚开始。

在经历了一番思考之后,主管挑选了使用范围广、难度也适中的图片加载器(ImageLoader)作为小明的训练项目。既然要训练小明的面向对象设计,那么就必须考虑到扩展性,灵活性,而检测这一切是否符合需求的最好途径就是开源。用户不断地提出需求、反馈问题,小明的项目需要不断升级瞒住用户需求,并且要保证系统的稳定性、灵活性。在主管根小明说了着一特殊任务之后,小明第一次感到了压力。“生活不易啊!”年仅22的小明发出了这样的感叹!

挑战总是要面对的,何况是从来不服输的小明。主管的要求很简单,要实现图片加载,并且要将图片缓存起来。在分析了需求之后,小明一下就放心下来了吗,“这么简单,原来我还是以为很难啦......”小明胸有成竹地喃喃自语到。在经历了10分钟的编码之后,小明写下了如代码。


/**

* Created by ${whq} on 2020/3/27

*/

public class ImageLoader {

    //    图片缓存

    LruCachelruCache;

ExecutorService executorService =Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());

Handler handler =new Handler(Looper.getMainLooper());

public ImageLoader() {

        initImageCache();

}

    private void initImageCache() {

//        计算可使用的最大内存

        final int maxMenory =(int) (Runtime.getRuntime().maxMemory() /1024);

//        取四分之一的可用内存作为缓存

        final int cacheSize =maxMenory /4;

lruCache =new LruCache(cacheSize) {

            @Override

            protected int sizeOf(String key,Bitmap bitmap) {

                return bitmap.getRowBytes() *bitmap.getHeight() /1024;

}

        };

}

    public void dispalyImage(final String url,final ImageView imageView) {

        imageView.setTag(url);

executorService.submit(new Runnable() {

            @Override

            public void run() {

                Bitmap bitmap = downloadImage(url);

if (bitmap ==null) {

                    return;

}

                if (imageView.getTag().equals(url)) {

                    updataImageView(imageView,bitmap);

}

                lruCache.put(url,bitmap);

}

        });

}

    private void updataImageView(final ImageView imageView,final Bitmap bitmap) {

        handler.post(new Runnable() {

            @Override

            public void run() {

                imageView.setImageBitmap(bitmap);

}

        });

}

    private Bitmap downloadImage(String imageUrl) {

        Bitmap bitmap =null;

try {

            URL url =new URL(imageUrl);

final HttpURLConnection connection =(HttpURLConnection) url.openConnection();

bitmap =BitmapFactory.decodeStream(connection.getInputStream());

} catch (MalformedURLException e) {

            e.printStackTrace();

} catch (IOException e) {

            e.printStackTrace();

}

        return bitmap;

}

}

并且使用Git软件进行版本控制,将工程托管到github上,伴随着git push命名的完成.,ImageLoader0.1版本就正式发布了!如此短的时间内就完成了这个任务,而且还是一个开源项目,小明暗暗自喜,并幻想着待会儿被主管称赞。

在给主管报告了ImageLoader的发布消息的几分钟之后,主管就把小明叫到了会议室。这下小明纳闷了,怎么夸人还需要到会议室。”小明,你的ImageLoader耦合太严重了!简直就没设计可言,更不要说扩展性,灵活性了。所有的功能都写在一个类里怎么行啦,这样随着功能的增多,ImageLoader类会越来越大,代码页越来越复杂,图片加载系统越来越脆弱......“这简直就是当头棒喝,小明的脑海里已经听不清楚下面说的内容了,只是觉得自己之前没有考虑清楚就匆忙完成任务,而且把任务想的太简单了。

”你还是把ImageLoader拆分一下,把各个功能独立出来,让他们瞒住单一职责原则。“主管最后说道。小明是个聪明人,敏锐的捕捉到了单一职责原则这个关键词,他用Google搜索了一些资料之后,总算对单一原则有了一些认识,于是打算对ImageLoader进行一次重构。这次小明不敢草率,也是先画了一幅UML图,如图1-1所示:

ImageLoader代码修改如下:

/**

* Created by ${whq} on 2020/3/27

*/

public class ImageLoaderNew {

    ImageCache imageCache =new ImageCache();

ExecutorService executorService =Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());

Handler handler =new Handler(Looper.getMainLooper());

private void updataImage(final ImageView imageView,final Bitmap bitmap) {

        handler.post(new Runnable() {

            @Override

            public void run() {

                imageView.setImageBitmap(bitmap);

}

        });

}

    public void dispalyImage(final String url,final ImageView imageView) {

        imageView.setTag(url);

executorService.submit(new Runnable() {

            @Override

            public void run() {

                Bitmap bitmap = downloadImage(url);

if (bitmap ==null) {

                    return;

}

                if (imageView.getTag().equals(url)) {

                    updataImage(imageView,bitmap);

}

                imageCache.put(url,bitmap);

}

        });

}

    private Bitmap downloadImage(String imageUrl) {

        Bitmap bitmap =null;

try {

            URL url =new URL(imageUrl);

final HttpURLConnection connection =(HttpURLConnection) url.openConnection();

bitmap =BitmapFactory.decodeStream(connection.getInputStream());

} catch (MalformedURLException e) {

            e.printStackTrace();

} catch (IOException e) {

            e.printStackTrace();

}

        return bitmap;

}

}


/**

* Created by ${whq} on 2020/3/27

*/

public class ImageCache {

    LruCachebitmapLruCache;

public ImageCache() {

        initImageCache();

}

    private void initImageCache() {

        final int maxMemory =(int) (Runtime.getRuntime().maxMemory() /1024);

final int cacheSize =maxMemory /4;

bitmapLruCache =new LruCache(cacheSize) {

            @Override

            protected int sizeOf(String key,Bitmap value) {

                return value.getRowBytes() *value.getHeight() /1024;

}

        };

}

    public void put(String url,Bitmap bitmap) {

        bitmapLruCache.put(url,bitmap);

}

    public Bitmap get(String url) {

        return bitmapLruCache.get(url);

}

}

如图1-1和上述代码所示,小明将ImageLoader一拆为二,ImageLoader只负责图片的加载逻辑,而ImageCache只负责图片缓存的逻辑,这样ImageLoader的代码量变少了,职责也清晰了;当与缓存相关的逻辑需要改变时,不需要修改ImageLoader类,而图片加载的逻辑需要修改时也不会影响到图片缓存逻辑。主管在审核了第一次重构之后,对工作给予表扬,大致意思是结构变得清晰很多,但是可扩展还是比较欠缺。虽然没有得到主管的完全肯定,但也颇有进度,再考虑到自己确实有所收获,原本沮丧的心里也略微地好转了起来。

从上述的例子中我们能够体会到,单一职责所表达出的用意就是”单一“二字。正如上文所说,如何划分一个类、一个函数的职责,每个人都有自己的看法,这需要更具个人经验、具体的业务逻辑而定。但是,它也有一些基本的指导原则,例如,两个完全不一样的功能就应该放在一个类中。一个类中应该是一组相关性很高的函数、数据的封装。工程师可以不断地审视自己的代码,根据具体的业务、功能对类进行相对应拆分,这是程序员优化代码迈出的第一步。

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

推荐阅读更多精彩内容

  • SOLID+迪米特原则 在应用开发的过程中,最难的往往不是开发的工作,而是在后续的升级维护过程中让应用系统能够灵活...
    MarcoHorse阅读 3,106评论 1 2
  • 1、面向对象的六大原则 1、单一职责原则 简单的说,一个类应该是一组相关性很高的函数和数据的封装,也就是一个类一个...
    Dane_404阅读 492评论 0 0
  • Xutils3.0技术分享1.这个技术分享的目的1.首先要让大家了解Xutil3.0是什么Xtuils3.0的前身...
    wodezhuanshu阅读 3,110评论 5 9
  • 7.1 压缩图片 一、基础知识 1、图片的格式 jpg:最常见的图片格式。色彩还原度比较好,可以支持适当压缩后保持...
    AndroidMaster阅读 2,500评论 0 13
  • Volley框架 Volley是Google官方出的一套小而巧的异步请求库,该框架封装的扩展性很强,支持HttpC...
    void_Zhao阅读 10,700评论 2 2