一、继承

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

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

  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. 7-42 jmu-python-找字符 (15 分)

    输入一个字符串及字符,输出第一次出现该字符的位置. 输入格式: 行1:输入字符串 行2:输入一个字符 输出格式: 找到,输出对应位置,格式index=X的, X表示查找到位置 找不到,输出can't ...

  2. JavaWeb中登录验证码生成

    1.页面代码 <html> <head> <title>Title</title> <script type="text/javascr ...

  3. 使用openxml提取word中的文本和图片并转为Html

    使用openxml提取word中的文本和图片 使用 openXml 提取 word 中的 Text 和 Drawing 使用 openXml 将 word 中的文本和图片转为Html 使用 openX ...

  4. Generator的异步编程

    对比下常用的异步处理的方案: 1,回调 我们常说的 “回调地狱”,就是多个异步操作时候,代码多重嵌套,异步之前形成强耦合,如果修改一处,其他地方也是跟着修改.(callback hell). 2,pr ...

  5. Core + Vue 后台管理基础框架0——开篇

    1.背景 最近,打算新开个项目,鉴于团队技术栈,选型.net core + vue,前后端分离.本打算捡现成的轮子的,github上大致逛了逛,总发现这样那样的不太适合心中那些“完美实践”,例如:Ab ...

  6. JS中的reduce()详解

    reduce()作为一个循环使用.接收四个参数:初始值(上一次返回值),当前元素值,当前元素下标,原数组. 应用  作为累加器使用 var a=[4,5,6,7,8] //item代表一次回调的值 初 ...

  7. 第一个爬虫经历----豆瓣电影top250(经典案例)

    因为要学习数据分析,需要从网上爬取数据,所以开始学习爬虫,使用python进行爬虫,有好几种模拟发送请求的方法,最基础的是使用urllib.request模块(python自带,无需再下载),第二是r ...

  8. numpy——>数组拼接np.concatenate

    语法:np.concatenate((a1, a2, ...), axis=0) 1.默认是 axis = 0,也就是说对0轴(行方向)的数组对象,进行其垂直方向(axis=1)的拼接(即数据整行整行 ...

  9. 动手建立jdbc连接

    工具:Idea  Navicat 环境:jdk 1.8  mysql-5.7.27-winx64 创建一个project 打开navicat开启连接. 在idea中导入数据库. 导入好后可以开始连接了 ...

  10. 覆盖io.spring.platform管理的版本号

    使用io.spring.platform时,它会管理各类经过集成测试的依赖版本号.想要覆盖其中某个依赖的版本号个: https://www.cnblogs.com/ld-mars/p/11818252 ...