一、继承

不恰当的使用继承导致的最大的一个特征就是高耦合。

是否使用继承需要考虑三个点:

  1. 父类只是给子类提供服务,并不涉及子类的业务逻辑
  2. 层级关系明显,功能划分清晰,父类和子类各做各的。
  3. 父类的所有变化,都需要在子类中体现,也就是说此时耦合已经成为需求

万不得已不要用继承,优先考虑组合等方式。

  • 如果只是共享接口,我们可以使用协议;

    @protocol ptc <NSObject>
    - (void)do;
    @end @interface A : NSObject<ptc>
    @end @implementation A
    - (void)do
    {
    }
    @end @interface B : NSObject<ptc>
    @end
    @implementation B
    - (void)do
    {
    }
    @end
  • 如果希望共用一个方法的部分实现,但希望根据需要执行不同的其他行为,我们可以使用代理或者 AOP;

    @protocol ptc <NSObject>
    - (void)do;
    @end @interface A : NSObject
    @property (nonatomic, weak) id<ptc>delegate;
    @end @implementation A
    - (void)func
    {
    ...
    [self.delegate do];
    ...
    }
    @end @interface B : NSObject <ptc>
    @end @implementation B
    - (void)do
    {
    }
    @end
  • 如果是添加方法,我们可以优先使用类别;

  • 如果是为了使用一个类的很多方法,我们可以使用组合来实现。

    @interface A : NSObject
    - (void)methodA;
    @end @interface B : NSObject
    -(void)methodB;
    @end // 定义 C 以及其需要的 methodA,methodB
    @interface C : NSObject
    {
    A * __a;
    B * __b;
    }
    - (id)initWithA:(A *)a b:(B *)b;
    - (void)methodA;
    - (void)methodB;
    @end @implementation ClassC
    - (id)initWithA:(A *)a b:(B *)b
    {
    __a = [[A alloc] initWithA:a]; // [A copy];
    __b = [[B alloc] initWithB:b]; // [B copy];
    }
    - (void)methodA
    {
    [__a methodA];
    }
    - (void)methodB
    {
    [__b methodB];
    }
    @end

如果只是出于代码复用的目的而不区分类别和场景,就采用继承是不恰当的。当你发现你的继承超过 2 层的时候,你就要好好考虑是否这个继承的方案了,第三层继承正是滥用的开端。

二、多态

使用场景:

  1. 父类有部分public的方法是不需要,也不允许子类覆重
  2. 父类有一些特别的方法是必须要子类去覆重的,在父类的方法其实是个空方法
  3. 父类有一些方法是可选覆重的,一旦覆重,则以子类为准
  4. 父类有一些方法即便被覆重,父类原方法还是要执行的

通常带来的问题:

  • 容易使得一个对象引入原本不属于它的业务逻辑
  • 调用时机或忘记调用

解决方案:

  • 面向接口编程(Interface Oriented Programming, IOP)

在决定是否采用多态时,要有一个清晰的角色概念,做好角色细分,不要角色混乱。

三、封装

将相关的一堆函数和一堆对象放在一起,只留给外部程序员操作方式,而不暴露具体执行细节。

带来的问题:

  1. 制约了并行程度
  2. 数据部分就是数据部分,执行部分就是执行部分,不同类的东西放在一起是不合适的

四、内容来源

casatwy & 跳出面向对象思想(一) 继承

跳出面向对象思想(二) 多态

跳出面向对象思想(三) 封装

iOS架构师之路:慎用继承

iOS 继承的更多相关文章

  1. ios 继承UITableViewController,更改tableview样式

    // 继承UITableViewController,更改tableview样式 - (instancetype)initWithStyle:(UITableViewStyle)style { ret ...

  2. ios 继承关系图

  3. IOS入门之Swift语言(一)

    经过不断的努力,小哥也买了台苹果设备,终于可以开始我的IOS之旅了,说来确实令人苦恼,为了学习IOS我这着贫农阶级,省了几个月的零花钱,外加向亲朋好友求救,最终痛下心扉,卖了台MAC pro128G版 ...

  4. iOS项目开发日常之创建文件(协议、类、分类、扩展)

    iOS项目开发过程中,是以不断创建文件的形式进行着的. 创建得比较频繁的文件类型是: 这两个类型中创建的文件有:子类.分类.扩展.协议四种文件,如下:    这四类文件是频繁创建的,我们来看一下各自分 ...

  5. 字符串流sstream[part1/基本知识]

    C++中的输入输出分为三种:基于控制台的I/O,即istream.ostream.iostream:基于文件的I/O,即ifstream.ofstream.fstream:基于字符串的I/O,即ist ...

  6. C++ stringstream介绍,使用方法与例子

    From: http://www.usidcbbs.com/read-htm-tid-1898.html C++引入了ostringstream.istringstream.stringstream这 ...

  7. stringstream

    C++引入了ostringstream.istringstream.stringstream这三个类,要使用他们创建对象就必须包含sstream.h头文件. istringstream类用于执行C++ ...

  8. C++ 系列:iostream 的用途与局限

    转载自http://www.cnblogs.com/Solstice/archive/2011/07/17/2108715.html 本文主要考虑 x86 Linux 平台,不考虑跨平台的可移植性,也 ...

  9. 30分钟学会Objective-C

    注: 本文首发于我的个人博客:https://evilpan.com/2019/04/05/objc-basics/ 请原谅我的标题党.但是如果你有其他语言的学习经验,要学习Objective-C的语 ...

随机推荐

  1. linux入门系列17--邮件系统之Postfix和Dovecot

    前文演示了通过Samba和NFS实现文件共享,本篇演示使用Postfix和Dovecot在局域网实现电子邮件收发系统. 电子邮件系统是我们日常生活和工作中非常重要的一个网络服务,在windows下收发 ...

  2. SQL语句中in 与 exists的区别

    SQL语句中in 与 exists的区别 SQL中EXISTS检查是否有结果,判断是否有记录,返回的是一个布尔型(true/false); IN是对结果值进行比较,判断一个字段是否存在于几个值的范围中 ...

  3. audioContext.decodeAudioData 返回null 错误

    此问题并不是100%出现.没想到国外大神已经有处理此问题的经验 原贴地址: https://stackoverflow.com/questions/10365335/decodeaudiodata-r ...

  4. ARC中__bridge, __bridge__transfer, __bridge_retained 关系

    总结于 IOS Tuturial 中 ARC两章,详细在dropbox pdf 文档. Toll-Free Bridging 当你在 Objective-C 和 Core Foundation 对象之 ...

  5. WSGI-mini-web框架服务器

    前期准备: 安装python环境安装pycharm安装MySQL数据库安装pymsql创建一个学生表,存入数据我们只是实现一个非常简单的web服务,前端页面不会专门做页面文件,会在代码中以具体命令的形 ...

  6. Java easyui 下拉框默认选中第一个

    html代码: <tr> <td> <div style="margin-bottom:5px">计价方式:   <%--下拉框默认选中第 ...

  7. java开发——Cloneable接口、clone()方法和深浅拷贝

    1.实现Cloneable接口表明该类的对象是允许克隆的. 2.允许克隆的意思是:可以调用clone()方法. 3.深拷贝还是浅拷贝,取决于如何重写Object的clone()方法. 4.原对象和克隆 ...

  8. 【10】openlayers 视图view

    创建地图: //View对象代表地图的简单2D视图 //创建view let view = new ol.View({ center:[109,34],//视图的初始中心 maxZoom:18,//最 ...

  9. 【Win10】我们无法更新系统保留的分区

      前言 笔者是一个萌新,这个方案也是慢慢摸索出来的,有更好的方案欢迎大家提出 前段时间用公司电脑发现win10新版本还行,回家升级自己的电脑却提示“我们无法更新系统保留的分区”.(O_o)?? 笔者 ...

  10. JS中的call()方法和apply()方法用法总结(挺好 转载下)

    最近又遇到了JacvaScript中的call()方法和apply()方法,而在某些时候这两个方法还确实是十分重要的,那么就让我总结这两个方法的使用和区别吧. 1. 每个函数都包含两个非继承而来的方法 ...