走向面向对象的六大原则——单一职责原则

写在前面#


新的主题确定了,这一次准备总结Android设计模式,其实更准确的说应该叫做JAVA设计模式,这注定是一个浩大的工程,有你们陪着,我很快乐。

面向对象编程的六大原则


优化代码的第一步,单一职责原则###

单一职责原则的英文名称是Single Responsibility Principle,缩写是SRP。它的定义是:就一个类而言,应该仅有一个引起它变化的原因。简单来说,一个类应该是一组相关性非常高的函数、数据的封装
我们通过一个实际的Android例子进行讲解。

相信很多同学在学习安卓的过程中都有尝试写一个属于自己的ImageLoader,那么通常新手所做的ImageLoader应该是什么样呢?

public class ImageLoader {    
    // 图片缓存    
    LruCache<String, Bitmap> mImageCache;    
    // 线程池,线程数为CPU所允许的数量    
    ExecutorService mExecutorService =     Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors()); 
    public ImageLoader() {    
        initImageCache();
    }   
    private void initImageCache() {        
        // 获取APP可使用的最大内存       
        int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);        
        // 获取四分之一大小作为缓存空间          
        int cacheMemorySize = maxMemory / 4;        
        mImageCache = new LruCache<String, Bitmap>(cacheMemorySize) {      
            //Sizeof方法的作用只要是定义缓存中每项的大小,当我们缓存进去一个数据后,     
            //当前已缓存的Size就会根据这个方法将当前加进来的数据也加上,便于统计当    
            // 前使用了多少内存,如果已使用的大小超过maxSize就会进行清除动作;           
             @Override            
            protected int sizeOf(String key, Bitmap bitmap) {                
                return bitmap.getRowBytes() * bitmap.getHeight() / 1024;            
            }        
        };    
    }    
    public void displayImg(final String url, final ImageView imageView) {
        Bitmap bitmap = mImageCache.get(url);
        if (bitmap != null) {    imageView.setImageBitmap(bitmap);    return;
    }        
        // View中的setTag(object)表示给View添加一个格外的数据,以后可以用getTag()将这个数据取出来。        
        imageView.setTag(url);        
        //在子线程中完成加载图片和缓存图片        
        mExecutorService.submit(new Runnable() {            
            @Override            
            public void run() {                
                Bitmap bitmap = downloadImg(url);                
                if (bitmap == null) return;                
                if (imageView.getTag().equals(url)) {                    
                    imageView.setImageBitmap(bitmap);               
                }                
                mImageCache.put(url, bitmap);           
             }        
        });    
    }    
    private Bitmap downloadImg(String imageUrl) {        
        Bitmap bitmap = null;        
        try {            
            URL url = new URL(imageUrl);            
            HttpURLConnection connection = (HttpURLConnection) url.openConnection();            
            bitmap = BitmapFactory.decodeStream(connection.getInputStream());            
            connection.disconnect();        
        } catch (MalformedURLException e) {            
            e.printStackTrace();        
        } catch (IOException e) {           
            e.printStackTrace();        
        }        
            return bitmap;    
        }
   }

稍微懂一些的朋友就要说了,这样的代码耦合性太强了,简直没有设计可言,更不要说拓展性和灵活性了,随着ImageLoader功能的增多,ImageLoader会越来越大,代码越来越复杂,图片加载系统就越来越脆弱了!那么,作为一个新手,该怎么改进呢?

public class ImageLoader {
    // 图片缓存
    ImageCache mImageCache = new ImageCache();
    // 线程池,线程数为CPU所允许的数量
    ExecutorService mExecutorService = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());
    public void displayImg(final String url, final ImageView imageView) {
        Bitmap bitmap = mImageCache.get(url);
        if (bitmap != null) {
            imageView.setImageBitmap(bitmap);
            return;
        }
        // View中的setTag(object)表示给View添加一个格外的数据,以后可以用getTag()将这个数据取出来。
        imageView.setTag(url);
        //在子线程中完成加载图片和缓存图片
        mExecutorService.submit(new Runnable() {
            @Override
            public void run() {
                Bitmap bitmap = downloadImg(url);
                if (bitmap == null) return;
                if (imageView.getTag().equals(url)) {
                    imageView.setImageBitmap(bitmap);
                }
                mImageCache.put(url, bitmap);
            }
        });
    }
    private Bitmap downloadImg(String imageUrl) {
        Bitmap bitmap = null;
        try {
            URL url = new URL(imageUrl);
            HttpURLConnection connection = (HttpURLConnection) url.openConnection();
            bitmap = BitmapFactory.decodeStream(connection.getInputStream());
            connection.disconnect();
        } catch (MalformedURLException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return bitmap;
    }}

public class ImageCache {
    // 图片缓存
    LruCache<String, Bitmap> mImageCache;

    public ImageCache() {
        initImageCache();
    }

    private void initImageCache() {
    // 获取APP可使用的最大内存
    int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);
    // 获取四分之一大小作为缓存空间
    int cacheMemorySize = maxMemory / 4;
    mImageCache = new LruCache<String, Bitmap>(cacheMemorySize) {
        /**
         * Sizeof方法的作用只要是定义缓存中每项的大小,当我们缓存进去一个数据后,
         * 当前已缓存的Size就会根据这个方法将当前加进来的数据也加上,便于统计当
         * 前使用了多少内存,如果已使用的大小超过maxSize就会进行清除动作;
         */
        @Override
        protected int sizeOf(String key, Bitmap bitmap) {
            return bitmap.getRowBytes() * bitmap.getHeight() / 1024;
            }
        };
    }

    public void put(String url, Bitmap bitmap) {
        mImageCache.put(url, bitmap);
    }

    public Bitmap get(String url) {
        return mImageCache.get(url);
    }
}

如上述代码所示,将一个ImageLoader拆分成了两个,ImageLoader只负责图片加载的逻辑,而ImageCache只负责处理图片缓存。这样一来ImageLoader的代码变少了,逻辑也清晰了;当缓存逻辑需要修改时,就不需要再改变ImageLoader中的代码了。
从上述的例子中我们可以看出,单一职责原则的关键就在于单一二字,正如上文所述,一个类应该是一组相关性非常高的函数、数据的封装。每个人根据自己的经验,看法和具体的业务逻辑去划分职责,个性很强。但是,它也有一些基本的知道原则。比如两个完全不一样的功能就不应该放在一个类之内。
一个类应该是一组相关性非常高的函数、数据的封装。工程师可以不断审视自己的代码,根据具体的业务、功能对类进行对应的拆分,这是程序优化迈出的第一步!这也是为什么会有MVC,MVP等各种设计模式出现的原因!


希望阅读本章内容之后你能够有所收获!感谢阅读与喜欢!

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

推荐阅读更多精彩内容