(转载)OC学习篇之---类的三大特性:封装,继承,多态

之前的一片文章介绍了OC中类的初始化方法和点语法的使用,今天来继续学习OC中的类的三大特性,我们在学习Java的时候都知道,类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今天就来看一下OC中类的三大特性:

一、封装

学习过Java中类的同学可能都知道了,封装就是对类中的一些字段,方法进行保护,不被外界所访问到,有一种权限的控制功能,Java中有四种访问权限修饰符:public,default,protected,private,访问权限一次递减的,这样我们在定义类的时候,哪些字段和方法不想暴露出去,哪些字段和方法可以暴露,可以通过修饰符来完成,这就是封装,下面来看一个例子吧:

Car.h

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

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

@public、@protected、@private、@package

其中默认的修饰符是@private

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

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

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

二、继承

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

首先来看一下父类:Car

Car.h

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

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

Car.m

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

方法的实现

在来看一下子类:

Taxi.h

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

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

Taxi.m

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

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

再看一下另外一个子类:

Truck.h

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

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

Truck.m

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

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

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

三、多态

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

打印机的例子,抽象的打印机类Printer

Printer.h

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

就是一个简单的方法print

Printer.m

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

实现也是很简单的

下面来看一下具体的子类

ColorPrinter.h

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

ColorPrinter.m

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

再看一下另外一个子类

BlackPrinter.h

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

BlackPrinter.m

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

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

Person.h

 1 //
 2 //  Person.h
 3 //  07_DynamicDemo
 4 //
 5 //  Created by jiangwei on 14-10-11.
 6 //  Copyright (c) 2014年 jiangwei. All rights reserved.
 7 //
 8
 9 #import <Foundation/Foundation.h>
10
11 #import "ColorPrinter.h"
12 #import "BlackPrinter.h"
13
14 //扩展性不高,当我们需要添加一个新的打印机的时候还要定义对应的一个方法
15 //所以这时候就可以使用多态技术了
16
17 @interface Person : NSObject{
18     NSString *_name;
19 }
20
21 //- (void) printWithColor:(ColorPrinter *)colorPrint;
22
23 //- (void) printWithBlack:(BlackPrinter *)blackPrint;
24
25 - (void) doPrint:(Printer *)printer;
26
27 @end  

Person.m

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

再来看一下测试代码:

main.m

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

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

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

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

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

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

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

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

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

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

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

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

总结

这篇文章主要介绍了类的三大特性:封装、继承、多态,这三个特性也是后面学习面向对象的重要基础。

时间: 2024-10-27 00:09:19

(转载)OC学习篇之---类的三大特性:封装,继承,多态的相关文章

OC学习篇之---类的三大特性(封装,继承,多态)

之前的一片文章介绍了OC中类的初始化方法和点语法的使用:http://blog.csdn.net/jiangwei0910410003/article/details/41683873,今天来继续学习OC中的类的三大特性,我们在学习Java的时候都知道,类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今天就来看一下OC中类的三大特性: 一.封装 学习过Java中类的同学可能都知道了,封装就是对类中的一些字段,方法进行保护,不被外界所访问到,有一种权限的控制功能,Java

(1) 深入理解Java面向对象三大特性 封装 继承 多态

转眼已经工作快6年了,最开始做了2年J2EE:然后整了2年的数据仓库,主要是Cognos的报表开发:现在从事4G LTE核心网的开发,用的语言任然是Java,但写代码的机会不多,基本都是看代码找BUG,偶尔做点new feature也是在比较成熟的框架上复制.粘贴.修改,大部分时间还是在理解业务,钱多.事少.离家近,当时来这家公司图的是后面2点,2年过去了,英文水平有所提升,对敏捷开放也有一定的了解,但技术方面明显退步了或者说没有进步吧,本来以前也不怎么好,因为工作上用不到,自己也没怎么学习,所

面向对象的三大特性---封装继承多态

面向对象的三大特性---封装继承多态 首先我们来创建一个控制台程序 新建一个父类Person 创建一个子类继承父类Person 子类: main函数: 输出结果: 根据结果显示创建子对象,首先调用父类的构造函数在调用子类的构造函数. 继承的传递性 例如: A:B,B:C,则A:C ⅰ.继承过来之后被隐藏,有意隐藏加一个关键字"new",如果加了关键字"new",就告诉子类,父类里面的sayHi只能被隐藏,不能被覆写  ⅱ.继承过来之后被覆写/重写,覆写需要在父类里面

面向对象三大特性(封装/继承/多态)

引入:在面向对象思想中,有三大特性,封装  继承  多态. 思考:我们在使用类和对象实现编程的时候,到底是遵循这什么样的逻辑呢?为什么要这样规范类呢? 引入:面向对象很重要的思想就是隐藏,即外部不知道类内部是如何实现业务逻辑的,只管调用和使用结果,这个就是面向对象三大特性之一:  封装 1.封装[掌握] 定义:封装,字面意思就是将内容装到某个容器中,并进行密封保存,在面向对象思想中,封装指数据和对数据的操作捆绑到一起,形成 对外界的隐藏,同时对外提供可以操作的接口. 1.数据:即要操作的数据,在

面向对象三大特性:封装,继承,多态(二、多态)

多态 一,多态的概念 多态(polymorphism), 是指一个名字可以有多种语义.在面向对象语言中,多态是指在一棵继承树中的类中可以有多个同名但不同方法体及不同形参的方法.通常有两种途径实现多态:方法的重载和覆盖. 多态性允许以统一的风格处理已存在的变量及相关的类.多态性使得向系统里增加新功能变得容易.继承性和多态性是降低软件复杂性有有效技术. 二,多态的实现 1.方法的重载 方法重载时: 参数必须不同.参数个数不同或类型不同. 返回值可以相同,也可以不同. 重载的价值在于它允许通过使用一个

面向对象三大特性:封装,继承,多态(一,封装和继承)

封装.继承和多态是面向对象程序设计的三个核心特性. 封装是面向对象技术中的一个重要概念,是一种信息隐藏技术,它体现了面向对象程序设计的思想. 继承是面向对象程序设计方法的一个重要手段,通过继承可以更有效地组织程序结构,明确类间的关系,育雏利用已有的类来完成更复杂.更深入的程序开发. 多态允许以一种统一的风格处理已存在的变量和相关的类,多态性使得向系统增加功能变的容易. 一.封装 封装的含义是:类的设计者把类设计成一个黑匣子,使用者只能看见类中定义的公共变量和公共方法,而看不见方法的具体实现细节,

Java面向对象三大特性 封装 继承 多态

1.封装 封装的定义: 首先是抽象,把事物抽象成一个类,其次才是封装,将事物拥有的属性和动作隐藏起来,只保留特定的方法与外界联系 为什么需要封装: 封装符合面向对象设计原则的第一条:单一性原则,一个类把自己该做的事情封装起来,而不是暴露给其他类去处理,当内部的逻辑发生变化时,外部调用不用因此而修改,他们只调用开放的接口,而不用去关心内部的实现 举例: public class Human { private int age; private String name; public int get

面向对象三大特性:封装,继承,多态(三、多态的进一步讲解)

多态(polymoph)指的是“执行期间(而非编译期间)”判断所引用对象的实际类型.,根据其实际的类型调用其相应的方法. 多态也叫动态绑定,也叫迟绑定. 多态的存在有三个必要条件: 要有继承 要有重写 父类引用指向子类对象

转 OC温故:类的三大特性(封装,继承,多态)

原文标题:OC学习篇之---类的三大特性(封装,继承,多态) 我们都知道,面向对象程序设计中的类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今天就来看一下OC中类的三大特性: 一.封装 封装就是对类中的一些字段,方法进行保护,不被外界所访问到,有一种权限的控制功能,Java中有四种访问权限修饰符: 1 public,default,protected,private 访问权限依次递减,这样我们在定义类的时候,哪些字段和方法不想暴露出去,哪些字段和方法可以暴露,可以通过