Objective-C--- 多态 、 协议
1 编写交通工具程序
1.1 问题
本案例需要创建一个TRTransportation类,类中有一个方法叫print的方法,该方法默认输出 “显示交通工具信息”,这个类作为父类,派生出三个子类TRTaxi的士类、TRBus巴士类和TRBike 自行车类。TRTaxi的士类覆盖了父类的print方法,改成自己的输出,"交通工具为的士";TRBus巴士类覆盖了父类的print方法,改成自己的输出,"交通工具为巴士";TRBike 自行车类覆盖了父类的print方法,改成自己的输出,"交通工具为单车"。
在主程序中,创建三个交通工具,使用多态输出交通工具信息。
1.2 步骤
实现此案例需要按照如下步骤进行。
步骤一:定义TRTransportation类
首先,在Day04工程中新添加TRTransportation.h文件,用于定义新的类TRTransportation。
代码如下所示:
- #import <Foundation/Foundation.h>
- @interface TRTRansportation : NSObject
- -(void)print;
- @end
在上述代码中,为TRTransportation类添加一个方法print,用于输出信息“显示交通工具信息”到控制台。
然后,在类TRTransportation的实现部分,即在TRTransportation.m文件中,添加print方法的实现。
代码如下所示:
- #import "TRTransportation.h"
- @implementation TRTRansportation
- -(void)print{
- NSLog(@"显示交通工具信息");
- }
- @end
步骤二:定义TRTaxi类
首先,在Day04工程中新添加TRTaxi.h文件,用于定义新的类TRTaxi。
代码如下所示:
- #import "TRTransportation.h"
- @interface TRTaxi : TRTRansportation
- @end
上述代码可以看出,TRTaxi类继承自父类TRTRansportation。TRTaxi类并没有声明自己的属性和方法。
然后,在类TRTaxi的实现部分,即在TRTaxi.m文件中,重写由父类TRTRansportation继承的print方法。
代码如下所示:
- #import "TRTaxi.h"
- @implementation TRTaxi
- -(void)print{
- NSLog(@"交通工具为的士");
- }
- @end
步骤三:定义TRBus类
首先,在Day04工程中新添加TRBus.h文件,用于定义新的类TRBus。
代码如下所示:
- #import "TRTransportation.h"
- @interface TRBus : TRTRansportation
- @end
上述代码可以看出,TRBus类继承自父类TRTRansportation。TRBus类并没有声明自己的属性和方法。
然后,在类TRBus的实现部分,即在TRBus.m文件中,重写由父类TRTRansportation继承的print方法。
代码如下所示:
- #import "TRBus.h"
- @implementation TRBus
- -(void)print{
- NSLog(@"交通工具为巴士");
- }
- @end
步骤四:定义TRBike类
首先,在Day04工程中新添加TRBike.h文件,用于定义新的类TRBike。
代码如下所示:
- #import "TRTransportation.h"
- @interface TRBike : TRTRansportation
- @end
上述代码可以看出,TRBike类继承自父类TRTRansportation。TRBike类并没有声明自己的属性和方法。
然后,在类TRBike的实现部分,即在TRBike.m文件中,重写由父类TRTRansportation继承的print方法。
代码如下所示:
- #import "TRBike.h"
- @implementation TRBike
- -(void)print{
- NSLog(@"交通工具为单车");
- }
- @end
步骤五:在主程序中定义以上各类的对象
代码如下所示:
- #import <Foundation/Foundation.h>
- #import "TRTransportation.h"
- #import "TRTaxi.h"
- #import "TRBus.h"
- #import "TRBike.h"
- int main(int argc, const char * argv[])
- {
- @autoreleasepool {
- // insert code here...
- TRTRansportation* tran = [[TRTRansportation alloc] init];
- [tran print];
- TRTRansportation* tran1 = [[TRBus alloc] init];
- [tran1 print];
- TRTRansportation* tran2 = [[TRTaxi alloc] init];
- [tran2 print];
- TRTRansportation* tran3 = [[TRBike alloc] init];
- [tran3 print];
- }
- return 0;
- }
在上述代码中,以下代码:
- TRTRansportation* tran = [[TRTRansportation alloc] init];
- [tran print];
定义了一个TRTRansportation类的对象tran,并向对象tran发送print消息,以在控制台上输出“显示交通工具信息”。
在上述代码中,以下代码:
- TRTRansportation* tran1 = [[TRBus alloc] init];
- [tran1 print];
定义了一个TRTRansportation类类型的指针tran1,用它指向TRBus类的对象,这是允许的。在多态的概念中,允许用父类的指针指向派生类的对象。但此时只能向该指针指向的派生类对象发送父类中已有的消息,派生类中自定义的父类中没有的消息不能被发送,同时发送的这个消息将调用派生类中重写的函数。如上述代码中,向指向派生类对象的父类指针tran1发送父类中已经定义的print消息,调用的是派生类中重写的函数,所以在控制台上输出“交通工具为巴士”。
在上述代码中,以下代码:
- TRTRansportation* tran2 = [[TRTaxi alloc] init];
- [tran2 print];
定义了一个TRTRansportation类类型的指针tran2,用它指向TRTaxi类的对象,向该指针tran2发送父类中已经定义的print消息,调用的是派生类中重写的函数,所以在控制台上输出“交通工具为的士”。
在上述代码中,以下代码:
- TRTRansportation* tran3 = [[TRBike alloc] init];
- [tran3 print];
定义了一个TRTRansportation类类型的指针tran3,用它指向TRBike类的对象,向该指针tran3发送父类中已经定义的print消息,调用的是派生类中重写的函数,所以在控制台上输出“交通工具为单车”。
1.3 完整代码
本案例中,类TRTransportation声明,即TRTransportation.h文件,完整代码如下所示:
- #import <Foundation/Foundation.h>
- @interface TRTRansportation : NSObject
- -(void)print;
- @end
类TRTransportation实现,即TRTransportation.m文件,完整代码如下所示:
- #import "TRTransportation.h"
- @implementation TRTRansportation
- -(void)print{
- NSLog(@"显示交通工具信息");
- }
- @end
本案例中,类TRTaxi声明,即TRTaxi.h文件,完整代码如下所示:
- #import "TRTransportation.h"
- @interface TRTaxi : TRTRansportation
- @end
类TRTaxi实现,即TRTaxi.m文件,完整代码如下所示:
- #import "TRTaxi.h"
- @implementation TRTaxi
- -(void)print{
- NSLog(@"交通工具为的士");
- }
- @end
本案例中,类TRBus声明,即TRBus.h文件,完整代码如下所示:
- #import "TRTransportation.h"
- @interface TRBus : TRTRansportation
- @end
类TRBus实现,即TRBus.m文件,完整代码如下所示:
- #import "TRBus.h"
- @implementation TRBus
- -(void)print{
- NSLog(@"交通工具为巴士");
- }
- @end
本案例中,类TRBike声明,即TRBike.h文件,完整代码如下所示:
- #import "TRTransportation.h"
- @interface TRBike : TRTRansportation
- @end
类TRBike实现,即TRBike.m文件,完整代码如下所示:
- #import "TRBike.h"
- @implementation TRBike
- -(void)print{
- NSLog(@"交通工具为单车");
- }
- @end
主程序,即main.m,完整代码如下所示:
- #import <Foundation/Foundation.h>
- #import "TRTransportation.h"
- #import "TRTaxi.h"
- #import "TRBus.h"
- #import "TRBike.h"
- int main(int argc, const char * argv[])
- {
- @autoreleasepool {
- // insert code here...
- TRTRansportation* tran = [[TRTRansportation alloc] init];
- [tran print];
- TRTRansportation* tran1 = [[TRBus alloc] init];
- [tran1 print];
- TRTRansportation* tran2 = [[TRTaxi alloc] init];
- [tran2 print];
- TRTRansportation* tran3 = [[TRBike alloc] init];
- [tran3 print];
- }
- return 0;
- }
2 编写SuperMan类
2.1 问题
本案例需要定义两个协议,一个是TRPerson,它定义了一个方法job;另一个是TRFly,它定义了一个方法fly。创建一个SuperMan类,采用以上两个协议,使该类具有飞翔和工作的能力。
2.2 步骤
实现此案例需要按照如下步骤进行。
步骤一:定义协议TRPerson
首先,在Day04-2工程中新添加TRPerson.h文件,用于定义新的协议TRPerson。
代码如下所示:
- #import <Foundation/Foundation.h>
- @protocol TRPerson <NSObject>
- -(void)job;
- @end
在上述代码中,定义了一个协议TRPerson,在协议中有一个方法job的声明,表示采纳这个协议的类将具有工作的能力。该协议继承自NSObject。
步骤二:定义协议TRFly
首先,在Day04-2工程中新添加TRFly.h文件,用于定义新的协议TRFly。
代码如下所示:
- #import <Foundation/Foundation.h>
- @protocol TRFly <NSObject>
- -(void)fly;
- @end
在上述代码中,定义了一个协议TRFly,在协议中有一个方法fly的声明,表示采纳这个协议的类将具有飞翔的能力。该协议继承自NSObject。
步骤三:定义SuperMan类
首先,在Day04-2工程中新添加SuperMan.h文件,用于定义新的类SuperMan。
代码如下所示:
- #import <Foundation/Foundation.h>
- #import "TRFly.h"
- #import "TRPerson.h"
- @interface SuperMan : NSObject<TRFly,TRPerson>
- @end
上述代码可以看出,SuperMan类采纳了两个协议,一个是TRFly,另一个是TRPerson。SuperMan类并没有声明自己的属性和方法。
然后,在类SuperMan的实现部分,即在SuperMan.m文件中,对采纳的两个协议中的方法进行实现。当一个类采纳了某个协议后,在类的实现部分必须写出协议中声明的方法的实现。
代码如下所示:
- #import "SuperMan.h"
- @implementation SuperMan
- -(void)fly{
- NSLog(@"具有飞行的能力");
- }
- -(void)job{
- NSLog(@"具有了工作的能力");
- }
- @end
步骤四:在主程序中使用SuperMan类
代码如下所示:
- #import <Foundation/Foundation.h>
- #import "TRPerson.h"
- #import "TRFly.h"
- #import "SuperMan.h"
- int main(int argc, const char * argv[])
- {
- @autoreleasepool {
- // insert code here...
- id<TRFly,TRPerson> sMan = [[SuperMan alloc]init];
- [sMan fly];
- [sMan job];
- }
- return 0;
- }
在上述代码中,以下代码:
- id<TRFly,TRPerson> sMan = [[SuperMan alloc]init];
定义了一个id类型的指针sMan,并将指针sMan指向SuperMan类的对象。该id类型的指针由于进行了<TRFly,TRPerson>协议限定,所以指针只能调用SuperMan类中实现的TRFly协议和TRPerson协议声明的方法。
在上述代码中,以下代码:
- [sMan fly];
- [sMan job];
分别向id类型的指针发送fly消息和job消息,调用SuperMan类中的方法实现。
2.3 完整代码
本案例中,协议TRPerson声明,即TRPerson.h文件,完整代码如下所示:
- #import <Foundation/Foundation.h>
- @protocol TRPerson <NSObject>
- -(void)job;
- @end
本案例中,协议TRFly声明,即TRFly.h文件,完整代码如下所示:
- #import <Foundation/Foundation.h>
- @protocol TRFly <NSObject>
- -(void)fly;
- @end
本案例中,类SuperMan声明,即SuperMan.h文件,完整代码如下所示:
- #import <Foundation/Foundation.h>
- #import "TRFly.h"
- #import "TRPerson.h"
- @interface SuperMan : NSObject<TRFly,TRPerson>
- @end
类SuperMan实现,即SuperMan.m文件,完整代码如下所示:
- #import "SuperMan.h"
- @implementation SuperMan
- -(void)fly{
- NSLog(@"具有飞行的能力");
- }
- -(void)job{
- NSLog(@"具有了工作的能力");
- }
- @end
主程序,即main.m,完整代码如下所示:
- #import <Foundation/Foundation.h>
- #import "TRPerson.h"
- #import "TRFly.h"
- #import "SuperMan.h"
- int main(int argc, const char * argv[])
- {
- @autoreleasepool {
- // insert code here...
- id<TRFly,TRPerson> sMan = [[SuperMan alloc]init];
- [sMan fly];
- [sMan job];
- }
- return 0;
- }
Objective-C--- 多态 、 协议的更多相关文章
- Objective中的协议(Protocol)
Objective中的协议(Protocol) 作用: 专门用来声明一大堆方法. (不能声明属性,也不能实现方法,只能用来写方法的声明). 只要某个类遵守了这个协议.就相当于拥有这个协议中的所有的方法 ...
- Objective C多态
面向对象的封装的三个基本特征是.继承和多态. 包是一组简单的数据结构和定义相关的操作在上面的其合并成一个类,继承1种亲子关系,子类能够拥有父类定的成员变量.属性以及方法. 多态就是指父类中定义的成员变 ...
- Objective - C 之协议
一.创建方法: 二.实现过程: 1.遵循协议: @protocol NurseWorkingProtocol <NSObject> //<> 表示遵守协议,创建时就有(Nu ...
- Objective-C中的类目,延展,协议
Objective-C中的类目(Category),延展(Extension),协议(Protocol)这些名词看起来挺牛的,瞬间感觉OC好高大上.在其他OOP语言中就没见过这些名词,刚看到这三个名词 ...
- Xcode 常用编译选项设置
Xcode 常用编译选项设置 在xcconfig文件中指定即可. 用标准库连接 LINK_WITH_STANDARD_LIBRARIES = YES如果激活此设置,那么编译器在链接过程中会自动使用通过 ...
- 小程序多端框架全面测评:chameleon、Taro、uni-app、mpvue、WePY
摘要: 微信小程序开发技巧. 作者:coldsnap 原文:小程序多端框架全面测评 Fundebug经授权转载,版权归原作者所有. 最近前端届多端框架频出,相信很多有代码多端运行需求的开发者都会产生一 ...
- Clojure 哲学
简单性.专心编程不受打扰(freedom to focus).给力(empowerment).一致性和明确性:Closure编程语言中几乎每一个元素的设计思想都是为了促成这些目标的实现. 学习一门新的 ...
- Automake
Automake是用来根据Makefile.am生成Makefile.in的工具 标准Makefile目标 'make all' Build programs, libraries, document ...
- 五.OC基础--1.多态,2.类对象,3.点语法,4.@property&@synthesize,5.动态类型,内省(判断对象是否遵循特定的协议,以及是否可以响应特定的消息)
五.OC基础--1.多态, 1. 多态概念,定义:多态就是某一类事物的多种形态: 表现形式: Animal *ani = [Dog new]; 多态条件:1.有继承关系 2.有方法的重写 2.多态代码 ...
- Python:多态、协议和鸭子类型
多态 问起面向对象的三大特性,几乎每个人都能对答如流:封装.继承.多态.今天我们就要来说一说 Python 中的多态. 所谓多态:就是指一个类实例的相同方法在不同情形有不同表现形式.多态机制使具有不同 ...
随机推荐
- mvc+ef+oracle环境中报错:ORA-00001: 违反唯一约束条件
分析原因: 在oracle中,主健不能自动生成,不过可以通过“序列”来实现,如果是这样的话,问题很可能就出在“序列”上了: ORACLE表主键ID突然从已经存在的ID值开始自动生成,导致违反主键唯一性 ...
- Java面试题(1)- 高级特性
1. The diffrence between java.lang.StringBuffer and java.lang.StringBuilder? java.lang.StringBuffer: ...
- solr 4.3.0 配置
scheme.xml <?xml version="1.0" encoding="UTF-8" ?> <schema name="t ...
- Windows获取文件大小
Windows最初的设计允许我们处理非常大的文件,所以最初的设计者选用64位值来表示文件大小.但是我们在日常处理过程中文件大小一般不会超过4GB.故Windows提供了两个联合类型的数据结构表示文件大 ...
- mrg_myIsam分表引擎用法
CREATE TABLE `test`.`article_0` ( `id` BIGINT( 20 ) NOT NULL , `subject` VARCHAR( 200 ) NOT NULL , ` ...
- 转: Jsp9个内置对象详解
1.request对象 客户端的请求信息被封装在request对象中,通过它才能了解到客户的需求, 然后做出响应.它是HttpServletRequest类的实例. 序号方法说明 objectgetA ...
- POJ 3006 Dirichlet's Theorem on Arithmetic Progressions 素数 难度:0
http://poj.org/problem?id=3006 #include <cstdio> using namespace std; bool pm[1000002]; bool u ...
- ROS TF——learning tf
在机器人的控制中,坐标系统是非常重要的,在ROS使用tf软件库进行坐标转换. 相关链接:http://www.ros.org/wiki/tf/Tutorials#Learning_tf 一.tf简介 ...
- #数据结构-fib
/////////////////////////////////////////////////////////////////////////////// // // FileName : fic ...
- JSON初步
1.什么是JSON (1)JSON(Java Script Object Notation)是一种轻量级的数据交换语言, 以文本字符串为基础,且易于让人阅读 XML就是一个重量级的数据交换语言 (2) ...