1 编写交通工具程序

1.1 问题

本案例需要创建一个TRTransportation类,类中有一个方法叫print的方法,该方法默认输出 “显示交通工具信息”,这个类作为父类,派生出三个子类TRTaxi的士类、TRBus巴士类和TRBike 自行车类。TRTaxi的士类覆盖了父类的print方法,改成自己的输出,"交通工具为的士";TRBus巴士类覆盖了父类的print方法,改成自己的输出,"交通工具为巴士";TRBike 自行车类覆盖了父类的print方法,改成自己的输出,"交通工具为单车"。

在主程序中,创建三个交通工具,使用多态输出交通工具信息。

1.2 步骤

实现此案例需要按照如下步骤进行。

步骤一:定义TRTransportation类

首先,在Day04工程中新添加TRTransportation.h文件,用于定义新的类TRTransportation。

代码如下所示:

 
  1. #import <Foundation/Foundation.h>
  2. @interface TRTRansportation : NSObject
  3. -(void)print;
  4. @end

在上述代码中,为TRTransportation类添加一个方法print,用于输出信息“显示交通工具信息”到控制台。

然后,在类TRTransportation的实现部分,即在TRTransportation.m文件中,添加print方法的实现。

代码如下所示:

 
  1. #import "TRTransportation.h"
  2. @implementation TRTRansportation
  3. -(void)print{
  4. NSLog(@"显示交通工具信息");
  5. }
  6. @end

步骤二:定义TRTaxi类

首先,在Day04工程中新添加TRTaxi.h文件,用于定义新的类TRTaxi。

代码如下所示:

  1. #import "TRTransportation.h"
  2. @interface TRTaxi : TRTRansportation
  3. @end

上述代码可以看出,TRTaxi类继承自父类TRTRansportation。TRTaxi类并没有声明自己的属性和方法。

然后,在类TRTaxi的实现部分,即在TRTaxi.m文件中,重写由父类TRTRansportation继承的print方法。

代码如下所示:

  1. #import "TRTaxi.h"
  2. @implementation TRTaxi
  3. -(void)print{
  4. NSLog(@"交通工具为的士");
  5. }
  6. @end

步骤三:定义TRBus类

首先,在Day04工程中新添加TRBus.h文件,用于定义新的类TRBus。

代码如下所示:

  1. #import "TRTransportation.h"
  2. @interface TRBus : TRTRansportation
  3. @end

上述代码可以看出,TRBus类继承自父类TRTRansportation。TRBus类并没有声明自己的属性和方法。

然后,在类TRBus的实现部分,即在TRBus.m文件中,重写由父类TRTRansportation继承的print方法。

代码如下所示:

  1. #import "TRBus.h"
  2. @implementation TRBus
  3. -(void)print{
  4. NSLog(@"交通工具为巴士");
  5. }
  6. @end

步骤四:定义TRBike类

首先,在Day04工程中新添加TRBike.h文件,用于定义新的类TRBike。

代码如下所示:

  1. #import "TRTransportation.h"
  2. @interface TRBike : TRTRansportation
  3. @end

上述代码可以看出,TRBike类继承自父类TRTRansportation。TRBike类并没有声明自己的属性和方法。

然后,在类TRBike的实现部分,即在TRBike.m文件中,重写由父类TRTRansportation继承的print方法。

代码如下所示:

  1. #import "TRBike.h"
  2. @implementation TRBike
  3. -(void)print{
  4. NSLog(@"交通工具为单车");
  5. }
  6. @end

步骤五:在主程序中定义以上各类的对象

代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. #import "TRTransportation.h"
  3. #import "TRTaxi.h"
  4. #import "TRBus.h"
  5. #import "TRBike.h"
  6. int main(int argc, const char * argv[])
  7. {
  8. @autoreleasepool {
  9. // insert code here...
  10. TRTRansportation* tran = [[TRTRansportation alloc] init];
  11. [tran print];
  12. TRTRansportation* tran1 = [[TRBus alloc] init];
  13. [tran1 print];
  14. TRTRansportation* tran2 = [[TRTaxi alloc] init];
  15. [tran2 print];
  16. TRTRansportation* tran3 = [[TRBike alloc] init];
  17. [tran3 print];
  18. }
  19. return 0;
  20. }

在上述代码中,以下代码:

  1. TRTRansportation* tran = [[TRTRansportation alloc] init];
  2. [tran print];

定义了一个TRTRansportation类的对象tran,并向对象tran发送print消息,以在控制台上输出“显示交通工具信息”。

在上述代码中,以下代码:

  1. TRTRansportation* tran1 = [[TRBus alloc] init];
  2. [tran1 print];

定义了一个TRTRansportation类类型的指针tran1,用它指向TRBus类的对象,这是允许的。在多态的概念中,允许用父类的指针指向派生类的对象。但此时只能向该指针指向的派生类对象发送父类中已有的消息,派生类中自定义的父类中没有的消息不能被发送,同时发送的这个消息将调用派生类中重写的函数。如上述代码中,向指向派生类对象的父类指针tran1发送父类中已经定义的print消息,调用的是派生类中重写的函数,所以在控制台上输出“交通工具为巴士”。

在上述代码中,以下代码:

  1. TRTRansportation* tran2 = [[TRTaxi alloc] init];
  2. [tran2 print];

定义了一个TRTRansportation类类型的指针tran2,用它指向TRTaxi类的对象,向该指针tran2发送父类中已经定义的print消息,调用的是派生类中重写的函数,所以在控制台上输出“交通工具为的士”。

在上述代码中,以下代码:

  1. TRTRansportation* tran3 = [[TRBike alloc] init];
  2. [tran3 print];

定义了一个TRTRansportation类类型的指针tran3,用它指向TRBike类的对象,向该指针tran3发送父类中已经定义的print消息,调用的是派生类中重写的函数,所以在控制台上输出“交通工具为单车”。

1.3 完整代码

本案例中,类TRTransportation声明,即TRTransportation.h文件,完整代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. @interface TRTRansportation : NSObject
  3. -(void)print;
  4. @end

类TRTransportation实现,即TRTransportation.m文件,完整代码如下所示:

  1. #import "TRTransportation.h"
  2. @implementation TRTRansportation
  3. -(void)print{
  4. NSLog(@"显示交通工具信息");
  5. }
  6. @end

本案例中,类TRTaxi声明,即TRTaxi.h文件,完整代码如下所示:

  1. #import "TRTransportation.h"
  2. @interface TRTaxi : TRTRansportation
  3. @end

类TRTaxi实现,即TRTaxi.m文件,完整代码如下所示:

  1. #import "TRTaxi.h"
  2. @implementation TRTaxi
  3. -(void)print{
  4. NSLog(@"交通工具为的士");
  5. }
  6. @end

本案例中,类TRBus声明,即TRBus.h文件,完整代码如下所示:

  1. #import "TRTransportation.h"
  2. @interface TRBus : TRTRansportation
  3. @end

类TRBus实现,即TRBus.m文件,完整代码如下所示:

  1. #import "TRBus.h"
  2. @implementation TRBus
  3. -(void)print{
  4. NSLog(@"交通工具为巴士");
  5. }
  6. @end

本案例中,类TRBike声明,即TRBike.h文件,完整代码如下所示:

  1. #import "TRTransportation.h"
  2. @interface TRBike : TRTRansportation
  3. @end

类TRBike实现,即TRBike.m文件,完整代码如下所示:

  1. #import "TRBike.h"
  2. @implementation TRBike
  3. -(void)print{
  4. NSLog(@"交通工具为单车");
  5. }
  6. @end

主程序,即main.m,完整代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. #import "TRTransportation.h"
  3. #import "TRTaxi.h"
  4. #import "TRBus.h"
  5. #import "TRBike.h"
  6. int main(int argc, const char * argv[])
  7. {
  8. @autoreleasepool {
  9. // insert code here...
  10. TRTRansportation* tran = [[TRTRansportation alloc] init];
  11. [tran print];
  12. TRTRansportation* tran1 = [[TRBus alloc] init];
  13. [tran1 print];
  14. TRTRansportation* tran2 = [[TRTaxi alloc] init];
  15. [tran2 print];
  16. TRTRansportation* tran3 = [[TRBike alloc] init];
  17. [tran3 print];
  18. }
  19. return 0;
  20. }

2 编写SuperMan类

2.1 问题

本案例需要定义两个协议,一个是TRPerson,它定义了一个方法job;另一个是TRFly,它定义了一个方法fly。创建一个SuperMan类,采用以上两个协议,使该类具有飞翔和工作的能力。

2.2 步骤

实现此案例需要按照如下步骤进行。

步骤一:定义协议TRPerson

首先,在Day04-2工程中新添加TRPerson.h文件,用于定义新的协议TRPerson。

代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. @protocol TRPerson <NSObject>
  3. -(void)job;
  4. @end

在上述代码中,定义了一个协议TRPerson,在协议中有一个方法job的声明,表示采纳这个协议的类将具有工作的能力。该协议继承自NSObject。

步骤二:定义协议TRFly

首先,在Day04-2工程中新添加TRFly.h文件,用于定义新的协议TRFly。

代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. @protocol TRFly <NSObject>
  3. -(void)fly;
  4. @end

在上述代码中,定义了一个协议TRFly,在协议中有一个方法fly的声明,表示采纳这个协议的类将具有飞翔的能力。该协议继承自NSObject。

步骤三:定义SuperMan类

首先,在Day04-2工程中新添加SuperMan.h文件,用于定义新的类SuperMan。

代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. #import "TRFly.h"
  3. #import "TRPerson.h"
  4. @interface SuperMan : NSObject<TRFly,TRPerson>
  5. @end

上述代码可以看出,SuperMan类采纳了两个协议,一个是TRFly,另一个是TRPerson。SuperMan类并没有声明自己的属性和方法。

然后,在类SuperMan的实现部分,即在SuperMan.m文件中,对采纳的两个协议中的方法进行实现。当一个类采纳了某个协议后,在类的实现部分必须写出协议中声明的方法的实现。

代码如下所示:

  1. #import "SuperMan.h"
  2. @implementation SuperMan
  3. -(void)fly{
  4. NSLog(@"具有飞行的能力");
  5. }
  6. -(void)job{
  7. NSLog(@"具有了工作的能力");
  8. }
  9. @end

步骤四:在主程序中使用SuperMan类

代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. #import "TRPerson.h"
  3. #import "TRFly.h"
  4. #import "SuperMan.h"
  5. int main(int argc, const char * argv[])
  6. {
  7. @autoreleasepool {
  8. // insert code here...
  9. id<TRFly,TRPerson> sMan = [[SuperMan alloc]init];
  10. [sMan fly];
  11. [sMan job];
  12. }
  13. return 0;
  14. }

在上述代码中,以下代码:

  1. id<TRFly,TRPerson> sMan = [[SuperMan alloc]init];

定义了一个id类型的指针sMan,并将指针sMan指向SuperMan类的对象。该id类型的指针由于进行了<TRFly,TRPerson>协议限定,所以指针只能调用SuperMan类中实现的TRFly协议和TRPerson协议声明的方法。

在上述代码中,以下代码:

  1. [sMan fly];
  2. [sMan job];

分别向id类型的指针发送fly消息和job消息,调用SuperMan类中的方法实现。

2.3 完整代码

本案例中,协议TRPerson声明,即TRPerson.h文件,完整代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. @protocol TRPerson <NSObject>
  3. -(void)job;
  4. @end

本案例中,协议TRFly声明,即TRFly.h文件,完整代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. @protocol TRFly <NSObject>
  3. -(void)fly;
  4. @end

本案例中,类SuperMan声明,即SuperMan.h文件,完整代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. #import "TRFly.h"
  3. #import "TRPerson.h"
  4. @interface SuperMan : NSObject<TRFly,TRPerson>
  5. @end

类SuperMan实现,即SuperMan.m文件,完整代码如下所示:

  1. #import "SuperMan.h"
  2. @implementation SuperMan
  3. -(void)fly{
  4. NSLog(@"具有飞行的能力");
  5. }
  6. -(void)job{
  7. NSLog(@"具有了工作的能力");
  8. }
  9. @end

主程序,即main.m,完整代码如下所示:

  1. #import <Foundation/Foundation.h>
  2. #import "TRPerson.h"
  3. #import "TRFly.h"
  4. #import "SuperMan.h"
  5. int main(int argc, const char * argv[])
  6. {
  7. @autoreleasepool {
  8. // insert code here...
  9. id<TRFly,TRPerson> sMan = [[SuperMan alloc]init];
  10. [sMan fly];
  11. [sMan job];
  12. }
  13. return 0;
  14. }

Objective-C--- 多态 、 协议的更多相关文章

  1. Objective中的协议(Protocol)

    Objective中的协议(Protocol) 作用: 专门用来声明一大堆方法. (不能声明属性,也不能实现方法,只能用来写方法的声明). 只要某个类遵守了这个协议.就相当于拥有这个协议中的所有的方法 ...

  2. Objective C多态

    面向对象的封装的三个基本特征是.继承和多态. 包是一组简单的数据结构和定义相关的操作在上面的其合并成一个类,继承1种亲子关系,子类能够拥有父类定的成员变量.属性以及方法. 多态就是指父类中定义的成员变 ...

  3. Objective - C 之协议

    一.创建方法: 二.实现过程: 1.遵循协议: @protocol NurseWorkingProtocol <NSObject>   //<> 表示遵守协议,创建时就有(Nu ...

  4. Objective-C中的类目,延展,协议

    Objective-C中的类目(Category),延展(Extension),协议(Protocol)这些名词看起来挺牛的,瞬间感觉OC好高大上.在其他OOP语言中就没见过这些名词,刚看到这三个名词 ...

  5. Xcode 常用编译选项设置

    Xcode 常用编译选项设置 在xcconfig文件中指定即可. 用标准库连接 LINK_WITH_STANDARD_LIBRARIES = YES如果激活此设置,那么编译器在链接过程中会自动使用通过 ...

  6. 小程序多端框架全面测评:chameleon、Taro、uni-app、mpvue、WePY

    摘要: 微信小程序开发技巧. 作者:coldsnap 原文:小程序多端框架全面测评 Fundebug经授权转载,版权归原作者所有. 最近前端届多端框架频出,相信很多有代码多端运行需求的开发者都会产生一 ...

  7. Clojure 哲学

    简单性.专心编程不受打扰(freedom to focus).给力(empowerment).一致性和明确性:Closure编程语言中几乎每一个元素的设计思想都是为了促成这些目标的实现. 学习一门新的 ...

  8. Automake

    Automake是用来根据Makefile.am生成Makefile.in的工具 标准Makefile目标 'make all' Build programs, libraries, document ...

  9. 五.OC基础--1.多态,2.类对象,3.点语法,4.@property&@synthesize,5.动态类型,内省(判断对象是否遵循特定的协议,以及是否可以响应特定的消息)

    五.OC基础--1.多态, 1. 多态概念,定义:多态就是某一类事物的多种形态: 表现形式: Animal *ani = [Dog new]; 多态条件:1.有继承关系 2.有方法的重写 2.多态代码 ...

  10. Python:多态、协议和鸭子类型

    多态 问起面向对象的三大特性,几乎每个人都能对答如流:封装.继承.多态.今天我们就要来说一说 Python 中的多态. 所谓多态:就是指一个类实例的相同方法在不同情形有不同表现形式.多态机制使具有不同 ...

随机推荐

  1. for循环的嵌套——7月24日

      练习一:输入一个正整数,用for循环嵌套求阶乘的和 //输入一个正整数,求1!+2!+....+n! 用for循环嵌套 Console.Write("请输入一个正整数:"); ...

  2. Servlet容器如何同时来处理多个请求

    工作者线程Work Thread:执行代码的一组线程调度线程Dispatcher Thread:每个线程都具有分配给它的线程优先级,线程是根据优先级调度执行的Servlet采用多线程来处理多个请求同时 ...

  3. ANGULAR JS PROMISE使用

    Promise是一种模式,以同步操作的流程形式来操作异步事件,避免了层层嵌套,可以链式操作异步事件. 我们知道,在编写javascript异步代码时,callback是最最简单的机制,可是用这种机制的 ...

  4. 一步一步配置NLB

    废话不说,配置NLB需要准备以下环境: 1. 至少两个服务器,我的是windows server 2008 R2; 我的两个服务器名分别为NLB3和NLB2,其中NLB3是主,为什么呢?后面会谈到,在 ...

  5. node.js 对接公众平台

    http://www.tfan.org/wp-content/uploads/使用-Nodejs-和-MongoDB-开发高性能微信公众平台应用.pdf

  6. webApi跨域

    <system.webServer> <httpProtocol> <customHeaders> <add name="Access-Contro ...

  7. SQL语句的用法

    1.增加字段     alter table docdsp     add dspcodechar(200)2.删除字段     ALTER TABLE table_NAME DROP COLUMNc ...

  8. ZOJ 3329 One Person Game 概率DP 期望 难度:2

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=3754 本题分数为0的概率不确定,所以不能从0这端出发. 设E[i]为到达成功所 ...

  9. cpio的简单使用

    有如下文件 # file boot.kylin boot.kylin: ASCII cpio archive (SVR4 with no CRC) extract: # cpio -i <boo ...

  10. JS页面打印,预览,设置,分页

    一)在HTML页中加载打印对象 <object id="WebBrowser" width="0" height="0" classi ...