首页 > 多态---父指针指向子类对象(父类引用指向子类对象)

多态---父指针指向子类对象(父类引用指向子类对象)

 

我们都知道,面向对象程序设计中的类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今天就来看一下OC中类的三大特性:

一、封装

封装就是对类中的一些字段,方法进行保护,不被外界所访问到,有一种权限的控制功能,Java中有四种访问权限修饰符:

1
public,default,protected,private

访问权限依次递减,这样我们在定义类的时候,哪些字段和方法不想暴露出去,哪些字段和方法可以暴露,可以通过修饰符来完成,这就是封装,下面来看一个例子吧:

Car.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//  Car.h  
//  05_ObjectDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import   
   
@interface Car : NSObject{  
    //这个属性就是对外进行保密的相当于private,所以我们需要在外部访问的话,必须定义get/set方法  
    //默认的是private的,但是我们可以使用@public设置为public属性的,那么在外部可以直接访问:person->capcity = 2.8;  
    //当然我们一般不这么使用,因为这会破坏封装性,这种用法相当于C中的结构体中权限  
    //一共四种:@public,@protected,@private,@package,这个和Java中是相同的  
@public  
    float _capcity; //油量属性  
}  
   
- (void)run:(float)t;  
   
@end

这里我们可以看到,OC中也是有四种访问权限修饰符:

1
@public、@protected、@private、@package

其中默认的修饰符是@private

但是这里要注意的是:OC中的方法是没有修饰符的概念的,这个和Java有很大的区别,一般都是公开访问的,即public的,但是我们怎么做到让OC中的一个方法不能被外界访问呢?

OC中是这么做的,如果想让一个方法不被外界访问的话,只需要在.m文件中实现这个方法,不要在头文件中进行定义,说白了就是:该方法有实现,没定义,这样外界在导入头文件的时候,是没有这个方法的,但是这个方法我们可以在自己的.m文件中进行使用。

为什么要介绍这点知识呢?因为在后面我们会说到单利模式,到时候就会用到这个知识点了。

二、继承

继承是类中的一个重要的特性,他的出现使得我们没必要别写重复的代码,可重用性很高。当然OC中的继承和Java中是一样的,没多大区别,这里在看一个例子吧:

首先来看一下父类:Car

Car.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//  
//  Car.h  
//  06_ExtendDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import   
   
@interface Car : NSObject{  
    NSString *_brand;  
    NSString *_color;  
}  
   
- (void)setBrand:(NSString *)brand;  
- (void)setColor:(NSString *)color;  
- (void)brake;  
- (void)quicken;  
   
@end

在Car类中定义了两个属性,以及一些方法

Car.m

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
//  
//  Car.m  
//  06_ExtendDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Car.h"  
   
@implementation Car  
- (void)setBrand:(NSString *)brand{  
    _brand = brand;  
}  
- (void)setColor:(NSString *)color{  
    _color = color;  
}  
- (void)brake{  
    NSLog(@"刹车");  
}  
- (void)quicken{  
    NSLog(@"加速");  
}  
@end

方法的实现

在来看一下子类:

Taxi.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
//  
//  Taxi.h  
//  06_ExtendDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Car.h"  
   
@interface Taxi : Car{  
    NSString *_company;//所属公司  
}  
   
//打印发票  
- (void)printTick;  
   
@end

看到Taxi类继承了父类Car,这里需要导入父类的头文件,然后在Taxi类中多了一个属性和方法

Taxi.m

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
//  
//  Taxi.m  
//  06_ExtendDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Taxi.h"  
   
@implementation Taxi  
   
- (void)printTick{  
    [super brake];  
    [self brake];  
    NSLog(@"%@出租车打印了发票,公司为:%@,颜色为:%@",_brand,_company,_color);  
}  
   
@end

 

对方法的实现,这里我们看到实现文件中是不需要导入父类Car的头文件的,因为可以认为,Taxi.h头文件中已经包含了Car的头文件了。而且,这里可以使用super关键字来调用父类的方法,同时这里我们也是可以用self关键字来调用,这里看到其实这两种方式调用的效果是一样的,当我们在子类重新实现brake方法的时候(Java中的重写概念),那么这时候super关键字调用的还是父类的方法,而self调用的就是重写之后的brake方法了。同样,我们也是可以使用父类中的属性。

再看一下另外一个子类:

Truck.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//  
//  Truck.h  
//  06_ExtendDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Car.h"  
//卡车类继承Car  
@interface Truck : Car{  
    float _maxWeight;//最大载货量  
}  
   
//覆盖父类的方法brake  
//优先调用子类的方法  
- (void)brake;  
   
- (void)unload;  
   
@end

这里就自己定义了一个brake方法,这时候就会覆盖父类中的brake方法了。

Truck.m

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
//  
//  Truck.m  
//  06_ExtendDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Truck.h"  
   
@implementation Truck  
   
- (void)brake{  
    [super brake];  
    NSLog(@"Truck类中的brake方法");  
}  
   
- (void)unload{  
    [super brake];//调用父类的方法  
    [self brake];//也是可以的  
    NSLog(@"%@的卡车卸货了,载货量:%.2f,汽车的颜色:%@",_brand,_maxWeight,_color);  
}  
   
@end

这里就可以看到,我们会在brake方法中调用一下父类的brake方法,然后在实现我们自己的逻辑代码。

好了,继承就说这么多了,其实封装和继承两个特性没什么难度的,很容易理解的,下面在来看一下最后一个特性:多态

三、多态

多态对于面向对象思想来说,个人感觉是真的很重要,他对以后的编写代码的优雅方式也是起到很重要的作用,其实现在很多设计模式中大部分都是用到了多态的特性,Java中的多态特性用起来很是方便的,但是C++中就很难用了,其实多态说白了就是:定义类型和实际类型,一般是基于接口的形式实现的,不多说了,直接看例子吧:

打印机的例子

抽象的打印机类Printer

Printer.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
//  
//  Printer.h  
//  07_DynamicDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import   
   
@interface Printer : NSObject  
   
- (void) print;  
   
@end

就是一个简单的方法print

Printer.m

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
//  
//  Printer.m  
//  07_DynamicDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Printer.h"  
   
@implementation Printer  
   
- (void)print{  
    NSLog(@"打印机打印纸张");  
}  
   
@end

实现也是很简单的

下面来看一下具体的子类

ColorPrinter.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
//  
//  ColorPrinter.h  
//  07_DynamicDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Printer.h"  
   
//修改父类的打印行为  
@interface ColorPrinter : Printer  
- (void)print;  
@end

ColorPrinter.m

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
//  
//  ColorPrinter.m  
//  07_DynamicDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "ColorPrinter.h"  
   
@implementation ColorPrinter  
   
- (void)print{  
    NSLog(@"彩色打印机");  
}  
   
@end

在看一下另外一个子类

BlackPrinter.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
BlackPrinter.m
//  
//  BlackPrinter.m  
//  07_DynamicDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "BlackPrinter.h"  
   
@implementation BlackPrinter  
   
- (void)print{  
    NSLog(@"黑白打印机");  
}  
   
@end

这里我们在定义一个Person类,用来操作具体的打印机

Person.h 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
Person.m
//  
//  Person.m  
//  07_DynamicDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import "Person.h"  
   
@implementation Person  
   
/* 
- (void) printWithColor:(ColorPrinter *)colorPrint{ 
    [colorPrint print]; 
  
- (void) printWithBlack:(BlackPrinter *)blackPrint{ 
    [blackPrint print]; 
 */  
   
- (void) doPrint:(Printer *)printer{  
    [printer print];  
}  
   
@end

再来看一下测试代码:

main.m

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
//  
//  main.m  
//  07_DynamicDemo  
//  
//  Created by jiangwei on 14-10-11.  
//  Copyright (c) 2014年 jiangwei. All rights reserved.  
//  
   
#import   
   
#import "Person.h"  
#import "BlackPrinter.h"  
#import "ColorPrinter.h"  
   
int main(int argc, const charchar * argv[]) {  
    @autoreleasepool {  
           
        Person *person =[[Person alloc] init];  
           
        ColorPrinter *colorPrint = [[ColorPrinter alloc] init];  
        BlackPrinter *blackPrint = [[BlackPrinter alloc] init];  
           
        //多态的定义  
        /* 
        Printer *p1 = [[ColorPrinter alloc] init]; 
        Printer *p2 = [[BlackPrinter alloc] init]; 
          
        [person doPrint:p1]; 
        [person doPrint:p2]; 
         */  
           
        //通过控制台输入的命令来控制使用哪个打印机  
        int cmd;  
        do{  
            scanf("%d",&cmd);  
            if(cmd == 1){  
                [person doPrint:colorPrint];  
            }else if(cmd == 2){  
                [person doPrint:blackPrint];  
            }  
        }while (1);  
           
    }  
    return 0;  
}

下面就来详细讲解一下多态的好处

上面的例子是一个彩色打印机和黑白打印机这两种打印机,然后Person类中有一个操作打印的方法,当然这个方法是需要打印机对象的,如果不用多态机制实现的话(Person.h中注释的代码部分),就是给两种打印机单独定义个操作的方法,然后在Person.m(代码中注释的部分)中用具体的打印机对象进行操作,在main.m文件中,我们看到,当Person需要使用哪个打印机的时候,就去调用指定的方法:

1
2
[person printWithBlack:blackPrint];//调用黑白打印机  
[person printWithColor:colorPrint];//调用彩色打印机

这种设计就不好了,为什么呢?假如现在又有一种打印机,那么我们还需要在Person.h中定义一种操作这种打印机的方法,那么后续如果在添加新的打印机呢?还在添加方法吗?那么Person.h文件就会变得很臃肿。所以这时候多态就体现到好处了,使用父类类型,在Person.h中定义一个方法就可以了:

1
- (void) doPrint:(Printer *)printer;

这里看到了,这个方法的参数类型就是父类的类型,这就是多态,定义类型为父类类型,实际类型为子类类型

1
2
3
- (void) doPrint:(Printer *)printer{  
    [printer print];  
}

这里调用print方法,就是传递进来的实际类型的print方法。

1
2
3
4
5
Printer *p1 = [[ColorPrinter alloc] init];  
Printer *p2 = [[BlackPrinter alloc] init];  
           
[person doPrint:p1];  
[person doPrint:p2];

这里的p1,p2表面上的类型是Printer,但是实际类型是子类类型,所以会调用他们自己对应的print方法。

从上面的例子中我们就可以看到多态的特新很是重要,当然也是三大特性中比较难理解的,但是在coding的过程中,用多了就自然理解了,没必要去刻意的理解。

搜索CocoaChina微信公众号:CocoaChina
微信扫一扫
订阅每日移动开发及APP推广热点资讯

公众号:CocoaChina
我要投稿  收藏文章
29
上一篇:imageIO完成渐进加载图片
下一篇:怎样做一个iOS App的启动分层引导动画?

相关资讯

  • 如何在iOS上创建矢量图形
  • iOS开发--处理不等高TableViewCell的小花招
  • 小白如何晋级入门级iOS开发者
  • iOS runtime实战应用:成员变量和属性
  • 解决3D Touch导致系统相册崩溃的问题
  • iOS开发编码建议与编程经验(持续更新中)
  • iOS开发调试技巧总结(持续更新中)
  • iOS开发-CALayer的探究应用
  • Unrecognized Selector Sent to Instance问题
  • 深入浅出-iOS函数式编程的实现 && 响应式编程概念
我来说两句
发表评论
您还没有登录!请登录或注册
所有评论(1)
11月的淡然2016-03-09 14:18:00
不错~
  回复

热门资讯

综合评论

相关帖子

sinaweixinmail回到顶部

转载于:https://www.cnblogs.com/iOS-mt/p/5601202.html

更多相关:

  • 来源:公众号|计算机视觉工坊(系投稿)作者:仲夏夜之星「3D视觉工坊」技术交流群已经成立,目前大约有12000人,方向主要涉及3D视觉、CV&深度学习、SLAM、三维重建、点云后处理、自动驾驶、CV入门、三维测量、VR/AR、3D人脸识别、医疗影像、缺陷检测、行人重识别、目标跟踪、视觉产品落地、视觉竞赛、车牌识别、硬件选型、学术交流、...

  • 点云PCL免费知识星球,点云论文速读。文章:Real-Time LIDAR-Based Urban Road and Sidewalk Detection for Autonomous Vehicles作者:Ern˝o Horváth  , Claudiu Pozna ,and Miklós Unger编译:点云PCL代码:http...

  • 文章:Semantic Histogram Based Graph Matching for Real-Time Multi-Robot Global Localization in Large Scale Environment作者:Xiyue Guo, Junjie Hu, Junfeng Chen, Fuqin Deng, T...

  • 点云PCL免费知识星球,点云论文速读。文章:Robust Place Recognition using an Imaging Lidar作者:Tixiao Shan, Brendan Englot, Fabio Duarte, Carlo Ratti, and Daniela Rus编译:点云PCL(ICRA 2021)开源代码:...

  • 文章:A Survey of Calibration Methods for Optical See-Through Head-Mounted Displays作者:Jens Grubert , Yuta Itoh, Kenneth Moser编译:点云PCL本文仅做学术分享,如有侵权,请联系删除。欢迎各位加入免费知识星球,获取PD...

  • 因为函数参数是按值传递的,所以要想改变变量,必须传递地址。 二级指针实际上就是指针变量的地址,如果传递二级指针,函数声明必须写**。 (void**)&必须是本质上就是指针变量的地址才可以做这样的转换,并不是说把一个一级指针也可以转换,void**的本质是标识一个二级指针。 &data就是(默认数据类型 **)&data,(void...

  • 文章目录1. 解决问题2. 应用场景3. 实现如下:C++实现C语言实现4. 缺点 1. 解决问题 在工厂方法模式中,我们卖衣服。此时我们为每一种衣服创建不同的工厂,帽子有一个工厂专门创建,裤子有一个工厂专门创建,T恤有一个工厂专门创建。这样的方式保证了代码设计的开闭原则(对扩展开发,对修改关闭),解决了简单工厂模式中暴露的...

  • 转载于:http://blog.csdn.net/u012819339/article/details/50654764   实体作品请参看优酷视频。 若以上链接点击无效请把该链接地址复制到浏览器地址栏 http://v.youku.com/v_show/id_XODYzODczNzQ4.html 说明: 该作品为arvik于2014...

  • - (void)viewDidLoad {[super viewDidLoad];NSLog(@"我在玩手机");NSLog(@"手机没电了");[self chargeMyIphone:^{NSLog(@"出门逛街");}];NSLog(@"我在看电视"); }-(void)chargeMyIphone:(void(^)(void...

  • http://stackoverflow.com/questions/150446/how-do-i-detect-when-someone-shakes-an-iphone 我的实现(基于Eran Talmor): 没必要application.applicationSupportsShakeToEdit = YES; Set th...

  • 英语的重要性,毋庸置疑!尤其对广大职场人士,掌握英语意味着就多了一项竞争的技能。那,对于我们成人来说,时间是最宝贵的。如何短时间内在英语方面有所突破,这是我们最关心的事情。英语学习,到底有没有捷径可以走,是否可以速成?周老师在这里明确告诉大家,英语学习,没有绝对的捷径走,但是可以少走弯路。十多年的教学经验告诉我们,成功的学习方法可以借...

  • 展开全部 其实IDLE提供了一个显32313133353236313431303231363533e78988e69d8331333365663438示所有行和所有字符的功能。 我们打开IDLE shell或者IDLE编辑器,可以看到左下角有个Ln和Col,事实上,Ln是当前光标所在行,Col是当前光标所在列。 我们如果想得到文件代码...

  • 前言[1]从 Main 方法说起[2]走进 Tomcat 内部[3]总结[4]《Java 2019 超神之路》《Dubbo 实现原理与源码解析 —— 精品合集》《Spring 实现原理与源码解析 —— 精品合集》《MyBatis 实现原理与源码解析 —— 精品合集》《Spring MVC 实现原理与源码解析 —— 精品合集》《Spri...

  • 【本文摘要】【注】本文所述内容为学习Yjango《学习观》相关视频之后的总结,观点归Yjango所有,本文仅作为学习之用。阅读本节,会让你对英语这类运动类知识的学习豁然开朗,你会知道英语学习方面,我们的症结所在。学习英语这类运动类知识,需要把握四个原则第一,不要用主动意识。第二,关注于端对端第三,输入输出符合实际情况第四,通过多个例子...

  • 点云PCL免费知识星球,点云论文速读。文章:RGB-D SLAM with Structural Regularities作者:Yanyan Li , Raza Yunus , Nikolas Brasch , Nassir Navab and Federico Tombari编译:点云PCL代码:https://github.co...