2.2 继承9 i) v* W+ v- g
面向对象程序设计中最重要的一个概念是继承。继承允许我们依据另一个类来定义一个类,使得创建和维护一个应用程序变得更容易。也达到了重用代码功能和提高执行效率的效果。& |! A' M* q$ d, U
当创建一个类时,不需要重新编写新的数据成员和成员函数,只需指定新建的类继承了一个已有的类的成员即可。这个已有的类称为基类,新建的类称为派生类。一个类可以派生自多个类,这意味着,可以从多个基类继承数据和函数。 . |0 b! C$ ~5 H( ^) }+ c
定义一个派生类,我们使用一个类派生列表来指定基类。类派生列表以一个或多个基类命名,形式如下: - class derived-class: access-specifier base-class
) s: ~) k. K8 G/ }0 m' P$ a - //访问修饰符 access-specifier是 public/protected/private中的一个* X- w- h5 H! D$ E' e
- //base-class是之前定义过的某个类的名称# G4 Y( H( {/ ]: ~
- //若未使用访问修饰符access-specifier,则默认为private
复制代码继承的方式有如下三种: 公有继承(public):当一个类派生继承公有基类时,基类的公有成员也是派生类的公有成员,基类的保护成员也是派生类的保护成员,基类的私有成员不能直接被派生类访问,但是可以通过调用基类的公有和保护成员来访问 保护继承(protected):当一个类派生继承保护基类时,基类的公有和保护成员将成为派生类的保护成员 私有继承(private):当一个类派生继承私有基类时,基类的公有和保护成员将成为派生类的私有成员
0 D4 d' x: }! p- ^4 h. O) W. n
- /***** inherit_example.cpp *****// G1 C# T% ^. W
- #include <iostream>( C2 O# k- s3 R8 L7 z2 r& s/ |# O
- #include <string>+ v0 Z0 [8 l9 Y0 N; K) O1 P
- using namespace std;
# A- l2 y3 u6 }4 Z; ^# n6 Z - /* 动物类,抽象出颜色和体重,这两种动物都具有的属性 */* l$ y7 l4 [) }9 M2 C' H! ^
- class Animal { //基类
; q C6 b [4 T( P# `( q S - public:# b* m0 C* a( T( d! w, }/ Q! S
- string color; //颜色成员变量. X% \6 W) o C2 |. X
- int weight; //体重成员变量2 O q; w; q7 T+ s- D6 A3 W# o
- };) L9 B& O' ~+ {) A8 Y
- /* 狗类继承了动物基类,并在狗类里添加自己的属性 */
" L" `- x a' p' W3 s4 g% t - class Dog : public Animal { //派生类,公有继承了基类
* }4 D$ ?, O7 k - public:
* F$ b! y& c; o8 E - string name;4 _* e* a0 L" p
- int age; N2 Q6 E+ q' R/ `# O' d
- void run();& _ @% ]& _8 L5 T" Q
- };
0 o& ]1 {7 x8 x! s) f/ N - / @* S& K3 z3 x* g+ p% A; V) y6 O
- int main() {- a0 g3 p3 T! v7 a4 ]3 I( f
- Dog dog;1 x% E$ t5 W5 w9 F
- dog.name = "旺财";
4 ^/ Z% J& h+ h4 { - dog.age = 2;) d9 L# S7 K4 H( X$ R5 D
- dog.color = "黑色";" Y5 @6 k, x: x! O& P
- dog.weight = 120;8 ]6 {; o u6 Y1 s: j: Q# I1 s
- cout<<"狗的名字叫: "<<dog.name<<endl;
9 k8 I; X& x' L& N T4 m$ `0 a - cout<<"狗的年龄是: "<<dog.age<<endl;
6 I; V$ `. x/ y$ V/ @8 V6 ~ - cout<<"狗的毛发颜色是: "<<dog.color<<endl;
" _# b: u, W2 j) V5 \' n - cout<<"狗的体重是: "<<dog.weight<<endl;
6 r* _. @! F, x, r0 s - return 0;% M4 \7 k" a. \$ X& f# E2 b/ {% C
- }
复制代码执行下面的指令开始编译 ' ]/ ]$ H7 a% i! C% h
- g++ inherit_example.cpp -o inherit_example1 I: s9 ^1 S) \# C. N- T9 A
复制代码执行./inherit_example后结果如下 - 狗的名字叫:旺财
# U0 k6 Z: P& @% A - 狗的年龄是:2/ f, C: V9 S$ f F( i3 u
- 狗的毛发颜色是:黑色. @/ U A- W6 J/ T- ~$ l+ U
- 狗的体重是:120
复制代码 # I/ ]# W0 z$ y" D0 m7 |
2.3 重载
, V% `0 z* r1 I/ b HC++允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载。重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列表和定义(实现)不相同。 ! Q2 d$ `) u8 \) R8 M0 o. k" v
当调用一个重载函数或重载运算符时,编译器通过把您所使用的参数类型与定义中的参数类型进行比较,决定选用最合适的定义。选择最合适的重载函数或重载运算符的过程,称为重载决策 $ i6 t% `+ \4 O8 F" i6 Z0 P
函数重载:在同一个作用域内,可以声明几个功能类似的同名函数,但是这些同名函数的形式参数(指参数的个数、类型或者顺序)必须不同 /***** func_overloading.cpp *****/8 d- I h8 t. b8 B
#include <iostream>
; A9 u; Z$ a$ G& v& M#include <string>9 M5 L8 X' f0 S7 y
using namespace std;4 e! ~5 S$ u1 F, M& \% [( b( q
1 \2 A2 I D) I3 d ]! ?5 P0 x e
class Dog {& b: \* Q0 @* n( f _# ^. I
public: E% S$ Y4 ?/ ^. X* J- i* X- T
string name; X, G- y8 J" i4 M: M2 F4 a
void getWeight(int weight) { //写了一个方法,以int类型作为参数3 T" w T' x+ M6 S# }: D/ D, U" t1 J
cout<<name<<"的体重是: "<<weight<<"kG"<<endl;
5 G* s% F9 ?+ z4 a% X- j }* I* r+ F( E$ _$ i) E1 V' u3 Y
//以相同的函数名 getWeight,不同的参数类型 double weight,构成了函数重载
N0 e/ J# n" X; p, l' v! _ void getWeight(double weight) { 9 W6 f2 r$ I* b* A! D5 F2 D" \
cout<<name<<"的体重是: "<<weight<<"kG"<<endl;
/ U$ H2 V1 [6 }0 ], T }
0 b; F0 { w- G! V) R, \ u8 V};
- R0 ^: ?! ?% {% p$ W
' ~8 @6 E# h% r# Yint main() {5 f9 M; f1 v% v" ], ]* u
Dog dog;
/ g9 d( }( y2 [2 z: R4 d5 E dog.name = "旺财";
7 Z% _+ I6 Z3 h; _2 I dog.getWeight(10); //传进不同的参数,程序会匹配不同的重载函数
1 k( Z/ t! x1 H" A u dog.getWeight(10.5); //传进不同的参数,程序会匹配不同的重载函数+ J! U3 K9 g; k# C; _
return 0;8 _. p2 M) C$ i
} 0 v( e- ?* T8 f) ^% z9 l
) ?, k! i3 ]" W# W. U$ w& G3 i; p- Q. C4 T% t. R
执行下面的指令开始编译 - g++ func_overloading.cpp -o func_overloading2 o4 F7 b9 Y7 p2 r3 a7 s: v
复制代码执行./func_overloading后结果如下 - 旺财的体重是:10KG
. i$ `, Z7 W, ~' Q - 旺财的体重是:10.5KG
复制代码运算符重载:实质就是函数重载或多态,目的在于让人能够用同名的函数来完成不同的基本操作。重载运算符的格式如下: - <返回类型说明符> operator <运算符符号>(<参数表>)
4 F1 H6 d- \4 @ - {6 Y+ A- D5 j: d6 l* B
- <函数体>1 V* ~# ~3 U% n0 |3 d
- }
复制代码- /***** operator_example.cpp *****/
9 ?* s1 u; E( ~ - #include <iostream> d/ X7 n+ T# y; A1 W
- #include <string>5 h* L5 W; t% [- G f
- using namespace std;' |! Q5 l; ]8 G' k( ~4 ^+ a2 I7 F
- 0 i4 S7 o, K8 }
- class Dog {
$ R. p7 G" Q: Q2 G$ \, Y8 b - public:
/ d4 g! @. A- H2 g/ ?/ H6 } - int weight;8 u6 G. j# z4 z4 F/ |) \( K/ m
- //重载“+”运算符,注意函数必须与类名同名,把Dog对象作为传递0 V# V7 D; h4 p9 |* i" x9 |
- //使用this运算符进行访问,然后返回一个dog对象1 K3 S* V3 ?2 s, n/ z1 E
- Dog operator+(const Dog &d) {
- P+ e& A8 H) G- U/ a- ?# o1 ? - Dog dog;4 M$ C3 U/ d* r1 p3 B$ S7 R/ Q
- dog.weight = this->weight + d.weight;. V; B9 {' x. o8 R* A9 _
- return dog;; w' N3 j, j- K- z0 b
- }; P6 Z5 ^& _5 \+ N0 y
- };
$ p2 m2 W" T) F7 r0 g( ^$ L8 z - , b/ b1 A/ S, P. A( Q% u' U' k3 M
- int main() { C1 y3 z/ l: b" m
- Dog dog1;
, o2 W1 L7 ]) U% I( D, R7 ?' {' R - Dog dog2;
/ ~# k/ v: c' S1 @* d0 c/ b - Dog dog3;0 F4 U( ^* k' j( S
- 9 ^$ R" T( k$ v( H) o
- dog1.weight = 10;( P5 F3 ?( X0 [ w* ^, Z+ E8 {
- dog2.weight = 20;
4 c0 |+ j! M6 i T - dog3 = dog1 + dog2;2 s4 {& k1 d& a a
- cout<<"第三只狗的体重是: "<<dog3.weight<<endl;
1 z4 L; F p5 s4 S) O6 G% e# t - return 0;* Q) j" w2 l* r C8 i+ Y
- }
复制代码执行下面的指令开始编译 Y* U$ U& {, [/ v t+ p0 H' b
- g++ operator_example.cpp -o operator_example
复制代码
5 ] P* Q' `0 x* u- @0 t9 L8 U执行./operator_example后结果如下 - 第三只狗的体重是:30+ L! A1 n H0 ]# ~& C
复制代码2.4 多态 ! r/ {% Z. g* z# J i' a3 a' O& s
2 S N a$ |8 a% O7 cC++多态意味着调用成员函数时,会根据调用函数的对象的类型来执行不同的函数。形成多态必须具备三个条件:必须存在继承关系;继承关系必须有同名虚函数;存在基类类型的指针或者引用,通过该指针或引用调用虚函数 # l" [' A* ?4 x, u' [
虚函数:是在基类中使用关键字virtual声明的函数。在派生类中重新定义基类中定义的虚函数时,会告诉编译器不要静态链接到该函数。我们想要的是在程序中任意点可以根据所调用的对象类型来选择调用的函数,这种操作被称为动态链接,或后期绑定。虚函数必须实现,如果不实现,编译器将报错虚函数声明:virtual ReturnType FunctionName(Parameter) $ G# D6 w1 ?8 @8 f; Y
纯虚函数:若在基类中定义虚函数,以便在派生类中重新定义该函数更好地适用于对象,但是在基类中又不能对虚函数给出有意义的实现,这时就会用到纯虚函数。纯虚函数一定没有定义,用来规范派生类的行为,即接口。包含纯虚函数的类是抽象类,抽象类不能定义实例,但可以声明指向实现该抽象类的具体类的指针或引用 : V6 T' W1 D4 x6 T
- q4 l6 n ^ ~9 ~3 T
纯虚函数声明:virtual void funtion1()=0 - /***** polymorphism_example.cpp *****/
/ G& r' C) d9 h; H - #include <iostream>
& m: b2 r2 {* Z8 X. \) G - #include <string>
$ l: C& R, n4 I- w/ i8 `: o- A# s - using namespace std;
( l) D2 k6 z- n9 f( r% J0 t5 s5 n8 f - /* 定义一个动物类 */
1 r' t6 T3 F& t2 ? - class Animal {( [, G. d& h* @7 `- b( v
- public:" b8 l! n1 h6 X& v* w+ Y8 W% l. J
- virtual void run() { //虚函数
. i: y2 l, t5 f. H* a8 F5 ]- l0 f# D - cout<<"Animal 的 run()方法"<<endl;' g; h ^) Q G, g9 Z) p- e, Q6 Z
- }
! E/ }0 f8 T+ ^( }4 m - };9 p& \& X1 Z' E3 d0 w
- /* 定义一个狗类,并继承动物类 */
' B/ X& |) ^, u* Q7 S - class Dog : public Animal {# X) l: b# v6 f9 p& Z7 v6 r' [
- public:
- [) C% d" T! M6 o+ |1 H - void run() {
+ r5 c! h( t$ k - cout<<"Dog 的 run()方法"<<endl;
$ T8 a3 Q, z2 E* q! n) @# r. `; O - }8 J7 z. I0 R2 s/ }6 z
- };
, X" v( ~) a( @' e3 e - /* 定义一个猫类,并继承动物类 */! ~0 j& A! `0 k, }5 q
- class Cat : public Animal {
3 ~6 x- }. i6 e. a - public:
+ y$ H) V, y6 a2 n4 e2 H7 ` - void run() {" ^) f) J' |2 l
- cout<<"Cat 的 run()方法"<<endl;
6 x1 i& \+ ?. M k - }3 U- o! k& m% P7 L* [- }4 s* ~
- };
9 y6 n" ]% @$ Y" @( ^( G - + W* D3 U* G. ]) V+ |
- int main() {7 i: w+ N+ A- m2 y/ P- I
- /* 声明一个 Animal 的指针对象,注:并没有实例化 */. @$ K2 w8 d1 b( X
- Animal *animal; 1 b8 Z8 {/ |7 O. {' d* S" @% Q; y
- Dog dog; /* 实例化 dog 对象 */ 2 E6 V: Y6 g1 h, U3 a
- Cat cat; /* 实例化 cat 对象 */ 0 k$ ?9 h8 L9 d' [* y( w7 N
- animal = &dog; /* 存储 dog 对象的地址 */
7 F# b1 K% s8 ^; C7 q/ A9 W9 U, [ - animal->run(); /* 调用 run()方法 */ - ^* h6 z! j; J1 h
- animal = &cat; /* 存储 cat 对象的地址 */ 2 C$ j. E* U4 M) {! X
- animal->run(); /* 调用 run()方法 */, [- W% D: Z' Y0 J1 N P$ g# `
- return 0;
/ F! F* I" B% q, v - }
复制代码执行下面的指令开始编译 - g++ polymorphism_example.cpp -o polymorphism_example
. u8 {$ ~% _# t( t
复制代码 ) y6 Y: `' a) n. b" |
执行./polymorphism_example后结果如下 - Dog的run()方法" ?- F5 Z" U' {3 q3 ]- M
- Cat的run()方法
复制代码 & ^, T3 X3 p: G0 c" A2 @& u Q
转载自嵌入式攻城狮 y2 Q9 L. s5 D: N& T1 K# |
5 n. E) c$ g; N$ O! O V" k9 }! ~
& L) ~: D2 T o& o0 U2 v
& e6 X5 |) a7 K& z1 C9 b
6 e/ z. G' v6 U4 Q! G. F, w4 ~( n |