版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、第六章 C+程序的结构本章主要内容作用域与可见性对象的生存期数据与函数静态成员共享数据的保护友元编译预处理命令多文件结构和工程作用域:函数原型的作用域块作用域类作用域1. 文件作用域一、作用域与可见性1、函数原型的作用域函数原型中的参数,其作用域始于(,结束于)。例如,设有下列原型声明:double Area(double radius);radius 的作用域仅在于此,不能用于程序正文其它地方,因而可有可无。double Area(double);2、块作用域在块中声明的标识符,其作用域自声明处起,在块中声明的标识符,其作用域自声明处起,限于块中,例如:限于块中,例如:void fun(in
2、t a) int b(a); cinb; if (b0) int c; . c的作用域b的作用域3、类作用域类作用域作用于特定的成员名。类作用域作用于特定的成员名。类类C的成员的成员M具有类作用域具有类作用域,对对M的访问方式如下:的访问方式如下: 如果在如果在C的成员函数中没有声明同名的局部作用域标识符,那么在该函数内可以的成员函数中没有声明同名的局部作用域标识符,那么在该函数内可以访问成员访问成员M。通过表达式通过表达式c(对象)(对象).M或者或者C:M访问。访问。通过表达式通过表达式prt-Mclass CBox public: double Volume() const return
3、 m_Length*m_Width*m_Height; private: double m_Length; / Length of a box in inches double m_Width; / Width of a box in inches double m_Height; / Height of a box in inches;m_length是成员,具有类作用域是成员,具有类作用域4、文件作用域 不在前述各个作用域中出现的声明,具有文件作用域,这样声明不在前述各个作用域中出现的声明,具有文件作用域,这样声明的标识符的作用域开始于声明点,结束于文件尾。的标识符的作用域开始于声明点,结
4、束于文件尾。#include using namespace std;int i; /i是全局变量,文件作用域是全局变量,文件作用域int main() i= 5; int i= 7; /局部变量,块作用域局部变量,块作用域 coutiendl;return 0;可见性: 可见性是从对标识符的引用的角度来谈的概念 可见性表示从内层作用域向外层作用域“看”时能看见什么。 如果标识在某处可见,则就可以在该处引用此标识符。块作用域类作用域文件作用域可见性的一般规则:标识符应声明在先,引用在后。标识符应声明在先,引用在后。如果某个标识符在外层中声明,且在内层中没有如果某个标识符在外层中声明,且在内层中
5、没有同一标识符的声明,则该标识符在内层可见。同一标识符的声明,则该标识符在内层可见。对于两个嵌套的作用域,如果在内层作用域内声对于两个嵌套的作用域,如果在内层作用域内声明了与外层作用域中同名的标识符,则外层作用明了与外层作用域中同名的标识符,则外层作用域的标识符在内层不可见。域的标识符在内层不可见。在同一作用域内的对象名、函数名、枚举常量名在同一作用域内的对象名、函数名、枚举常量名会隐藏同名的类名或枚举类型名。会隐藏同名的类名或枚举类型名。重载的函数可以有相同的函数名。重载的函数可以有相同的函数名。#includeusing namespace std;int i; /文件作用域int mai
6、n() i=5; int i; /块作用域 /全局的i被隐藏 i=7; couti=iendl; /输出7 couti=i; /输出5 return 0;例 5.1二、对象的生存期对象(包括简单变量)从产生到结束的对象(包括简单变量)从产生到结束的这段时间就是它的生存期。这段时间就是它的生存期。在对象生存期内,在对象生存期内,对象将保持它的状态(即数据成员的值),对象将保持它的状态(即数据成员的值),变量也将保持它的值,直到被更新为止。对变量也将保持它的值,直到被更新为止。对象象(包括简单变量)的生存期可以分为静态(包括简单变量)的生存期可以分为静态生存期和动态生存期两种。生存期和动态生存期两
7、种。静态生存期 静态生存期与程序的运行期相同。 在文件作用域中声明的对象具有这种生存期。(全局变量比较危险,在程序中任何位置都可以访问全局变量,使得他们非常容易被意外修改) 在函数内部声明静态生存期对象,要冠以关键字static 。 例如:声明函数中的静态变量 static int count = 0;#includeusing namespace std;int i=5; /文件作用域文件作用域/i具有静态生存期具有静态生存期void record();();int main() couti=iendl; record(); return 0;void record(void) static
8、 int count = 0; /静态生存期静态生存期 count +;例动态生存期 块作用域中声明的,没有用static修饰的对象是动态生存期的对象(习惯称局部生存期对象)。 开始于程序执行到声明点时,结束于命名该标识符的作用域结束处。#includeusing namespace std;void fun();int main() fun(); fun();void fun() static int a=1; int i=5; a+; i+; couti=i,a=aendl;运行结果:运行结果:i=6, a=2i=6, a=3i是动态生存期是动态生存期a是静态生存期是静态生存期例例5-2
9、变量的生存期与可见性#includeusing namespace std;int i=1; / i 为全局变量,具有静态生存期。为全局变量,具有静态生存期。int main() static int a; / 静态局部变量,有全局寿命,局部可见。静态局部变量,有全局寿命,局部可见。默认初值始终是默认初值始终是0 int b=-10; / b, c为局部变量,具有动态生存期。为局部变量,具有动态生存期。 int c=0; void other(void); /函数原型声明函数原型声明 cout-MAIN-n; cout i: i a: a b: b c: cendl; /1 0 -10 0 c
10、=c+8; other(); /33 4 0 15 cout-MAIN-n; cout i: i a: a b: b c: cendl; /33 0 -10 8 i=i+10; /43 other(); /75 6 4 15 return 0;运行结果:运行结果:-MAIN- i: 1 a: 0 b: -10 c: 0-OTHER- i: 33 a: 4 b: 0 c: 15-MAIN- i: 33 a: 0 b: -10 c: 8-OTHER- i: 75 a: 6 b: 4 c: 15void other(void) static int a=2; static int b; / a,b为
11、静态局部变量,具有全局寿命,局部可见。 /只第一次进入函数时被初始化。 int c=10; / C为局部变量,具有动态生存期, /每次进入函数时都初始化。 a=a+2; /第一次调用,a = 4 i=i+32; /第一次调用,i = 33 c=c+5; /第一次调用,c = 15 cout-OTHER-n; cout i: i a: a b: b c: cendl; /第一次调用,i: 33 a:4 b: 0 c: 15 b=a; /第一次调用b =417例5-3具有静态、动态生存期对象的时钟程序#includeusing namespace std;class Clock/时钟类声明publ
12、ic:/外部接口Clock();void SetTime(int NewH, int NewM, int NewS); /三个形参均具有函数原型作用域void ShowTime();Clock()private:/私有数据成员int m_Hour, m_Minute, m_Second;/时钟类成员函数实现Clock:Clock()/构造函数 m_Hour=0;m_Minute=0;m_Second=0;void Clock:SetTime(int NewH, int NewM, int NewS) m_Hour=NewH;m_Minute=NewM;m_Second=NewS;void Cl
13、ock:ShowTime() coutm_Hour:m_Minute: m_Secondendl;20Clock globClock;/声明对象globClock, /具有静态生存期,文件作用域int main()/主函数coutFirst time output:endl;/引用具有文件作用域的对象:globClock.ShowTime(); /对象的成员函数具有类作用域globClock.SetTime(8,30,30);Clock myClock(globClock); /声明具有块作用域的对象myClockcoutSecond time output:endl;myClock.Show
14、Time();/引用具有块作用域的对象21程序的运行结果为:First time output:0:0:0Second time output:8:30:3022三、类的静态成员 类的数据成员和函数成员都可以被声明为static。 类的静态数据成员1. 类的静态函数成员静态数据成员用关键字static声明该类的所有对象维护该成员的一个实例Class CBox public: static int objectCount; private: double m_length; double m_width; double m_height;m_length m_width m_heightm_le
15、ngth m_width m_heightm_length m_width m_height对象3对象2对象1objectCount每个静态数据成员的副本在所有的类每个静态数据成员的副本在所有的类 类型对象间共享类型对象间共享- 必须在类外定义和初始化,用(:)来指明所属的类。初始化必须在类定义外部进行。 int CBox:objectCount = 0;(在声明任何对象前初始化static成员objectCount 的)例给CBox类中添加一个static成员,以计算生成对象的数目)这个成员是public型的#include using std:cout;using std:endl;cla
16、ss CBox / Class definition at global scope public: static int objectCount; / Count of objects in existence / Constructor definition CBox(double lv, double bv = 1.0, double hv = 1.0) cout endl Constructor called.; m_Length = lv; / Set values of m_Width = bv; / data members m_Height = hv; objectCount+
17、; CBox() / Default constructor cout endl Default constructor called.; m_Length = m_Width = m_Height = 1.0; objectCount+; / Function to calculate the volume of a box double Volume() const return m_Length*m_Width*m_Height; private: double m_Length; / Length of a box in inches double m_Width; / Width o
18、f a box in inches double m_Height; / Height of a box in inches;int CBox:objectCount = 0; / Initialize static member of CBox classint main() CBox boxes5; / Array of CBox objects declared CBox cigar(8.0, 5.0, 1.0); / Declare cigar box cout endl endl Number of objects (through class) = CBox:objectCount
19、; cout endl Number of objects (through object) = boxes2.objectCount; cout endl; return 0;Default constructor calledDefault constructor calledDefault constructor calledDefault constructor calledDefault constructor calledConstructor calledNumber of objects (through class)=6boxes2.objectCount=6例5-4 具有静
20、态数据成员的 Point类#include /private访问类型的静态数据成员using namespace std;class Pointpublic:Point(int xx=0, int yy=0) X=xx; Y=yy; countP+; Point(Point &p);int GetX() return X;int GetY() return Y; void GetC();private:int X,Y;static int countP;Void Point :GetC() cout Object id=countPendl;静态成员非静态函数访问静态数据成员Point:Poi
21、nt(Point &p) X=p.X;Y=p.Y;countP+;int Point:countP=0; int main() Point A(4,5);coutPoint A,A.GetX(),A.GetY();A.GetC(); /Point B(A);coutPoint B,B.GetX(),B.GetY();B.GetC();28静态成员函数类外代码可以使用类名和作用域操作符来调用静态成员函数。静态成员函数只能引用属于该类的静态数据成员或静态成员函数。引用普通的非静态类成员,必须通过参数传递得到对象名,通过对象名来访问。即使本类的任何对象都不存在,它们也能存在并被调用。这种情况下,只能
22、使用静态数据成员。Static void Afunction(int n)aBox .Afunction(10);CBox:Afunction(10);具有静态数据、函数成员的 Point类#include using namespace std;class Point/Point类声明public:/外部接口Point(int xx=0, int yy=0) X=xx;Y=yy;countP+;Point(Point &p);/拷贝构造函数int GetX() return X;int GetY() return Y;static void GetC() cout Object id=cou
23、ntPendl;private:/私有数据成员int X,Y;static int countP;静态函数访问静态数据成员Point:Point(Point &p) X=p.X;Y=p.Y;countP+;int Point:countP=0;int main()/主函数实现主函数实现 Point A(4,5);/声明对象声明对象AcoutPoint A,A.GetX(),A.GetY();A.GetC(); /输出对象号,对象名引用输出对象号,对象名引用Point B(A);/声明对象声明对象BcoutPoint B,B.GetX(),B.GetY();Point:GetC();/输出对象号
24、,类名引用输出对象号,类名引用32class A public: static void f(A a); private: int x;void A:f(A a) couta.x;静态函数访问非静态数据成员类中成员之间的访问静态数据成员非静态数据成员非静态函数成员静态函数成员OkOk通过对象名作为参数访问非内联,且函数体的定义应该与静态成员的初始化在同一个源程序文件中 四、类的友元 友元是C+提供的一种破坏数据封装和数据隐藏的机制。 通过将一个模块声明为另一个模块的友元,一个模块能够引用到另一个模块中本是被隐藏的信息。 可以使用友元函数和友元类。 为了确保数据的完整性,及数据封装与隐藏的原则,
25、建议尽量不使用或少使用友元。 友 元1、友元函数 有些情况下,由于这样或那样的原因,我们希有些情况下,由于这样或那样的原因,我们希望某些虽然不是类成员的函数能够访问类的所有成望某些虽然不是类成员的函数能够访问类的所有成员员他们拥有特殊权限。友元函数是在类声明中他们拥有特殊权限。友元函数是在类声明中由关键字由关键字friend修饰说明的非成员函数,在它的函修饰说明的非成员函数,在它的函数体中能够通过对象名访问数体中能够通过对象名访问 private 和和 protected成员。成员。 作用:增加灵活性,使程序员可以在封装和作用:增加灵活性,使程序员可以在封装和快速性方面做合理选择。快速性方面做
26、合理选择。 访问对象中的成员必须通过对象名。访问对象中的成员必须通过对象名。 注意:友元函数不是类的成员,因此访问特性不适用于他们,这些注意:友元函数不是类的成员,因此访问特性不适用于他们,这些函数只是拥有特殊权限的普通全局函数函数只是拥有特殊权限的普通全局函数。例5-6 使用友元函数计算两点距离#include #include using namespace std;class Point/Point类声明 public:/外部接口Point(int xx=0, int yy=0) X=xx;Y=yy;int GetX() return X;int GetY() return Y;frie
27、nd float Distance(Point &a, Point &b); private:/私有数据成员int X,Y; 友 元double Distance( Point& a, Point& b) double dx=a.X-b.X; double dy=a.Y-b.Y; return sqrt(dx*dx+dy*dy);int main() Point p1(3.0, 5.0), p2(4.0, 6.0); double d=Distance(p1, p2); coutThe distance is dendl; return 0;36例:生成友元函数计算表面积例:生成友元函数计算表
28、面积/ Creating a friend function of a class计算表面积计算表面积#include using std:cout;using std:endl;class CBox / Class definition at global scope public: / Constructor definition CBox(double lv = 1.0, double bv = 1.0, double hv = 1.0) cout endl Constructor called.; m_Length = lv; / Set values of m_Width = bv;
29、 / data members m_Height = hv; / Function to calculate the volume of a box double Volume() return m_Length*m_Width*m_Height; private: double m_Length; / Length of a box in inches double m_Width; / Width of a box in inches double m_Height; / Height of a box in inches / Friend function friend double B
30、oxSurface(CBox aBox);/ friend function to calculate the surface area of a Box objectdouble BoxSurface(CBox aBox) return 2.0*(aBox.m_Length*aBox.m_Width + aBox.m_Length*aBox.m_Height + aBox.m_Height*aBox.m_Width);int main() CBox match(2.2, 1.1, 0.5); / Declare match box CBox box2; / Declare box2 - no
31、 initial values cout endl Volume of match = match.Volume(); cout endl Surface area of match = BoxSurface(match); cout endl Volume of box2 = box2.Volume(); cout endl Surface area of box2 = BoxSurface(box2); cout endl; return 0;2、友元类 若一个类若一个类A为另一个类为另一个类B的友元,则此类的友元,则此类A的所有成员函数都能访的所有成员函数都能访问对方类的私有和保护成员。
32、问对方类的私有和保护成员。 声明语法:将友元类名在另一个类中使用声明语法:将友元类名在另一个类中使用friend修饰说明。修饰说明。class B friend class A; /声明A为B的友元类 假如已经定义了一个表示瓶子的假如已经定义了一个表示瓶子的CBottle类:类:class CBottle 、 public: CBottle(double height,double diameter) m_ height = height; m_ diameter = diameter; private: double m_height; double m_ diameter; friend
33、CCarton:CCarton(const CBottle& aBottle); /使用友元函数使用友元函数 firend CCarton;/使用友元类,使得使用友元类,使得CCatton中所有函数成员都有权访问中所有函数成员都有权访问CBottle类的所有数据成员。类的所有数据成员。 现在需要一个类来表示一打瓶子的包装箱,该类应该自动具有可以容纳特定种类瓶子的定制尺寸:现在需要一个类来表示一打瓶子的包装箱,该类应该自动具有可以容纳特定种类瓶子的定制尺寸:class CCaton 、 public: / Constructor definition CCaton (const CBottle&
34、 aBottle) m_Height =aBottle m_height; m_Length= 4.0*aBottle. m_ diameter; m_Width =3.0*aBottle. m_ diameter; private: double m_Length; / Length of a box in inches double m_Width; / Width of a box in inches double m_Height; / Height of a box in inches出错,出错,CBottle类的数据成员是类的数据成员是private外部不能访问!外部不能访问!友元
35、关系是单向的友元关系是单向的友元关系不是互惠的。如果声明B类是A类的友元,B类的成员函数就可以访问A类的私有和保护数据,但A类的成员函数却不能访问B类的私有、保护数据。 友元关系也是不可继承的。五、共享数据的保护 对于既需要共享,又需要防止改变的数据应该声明为常量。 const引用 类的const对象1. 类的const 成员函数常类型 常类型的对象必须进行初始化,而且不能被更新。常引用:被引用的对象不能被更新。const 类型说明符 &引用名常对象:必须进行初始化,不能被更新。const 类名 对象名常数组:数组元素不能被更新。类型说明符 const 数组名大小.常指针:指向常量的指针。例5
36、-7常引用做形参#includeusing namespace std;void display(const double& r);int main() double d(9.5); display(d); return 0;void display(const double& r)/常引用做形参,在函数中不能更新 r所引用的对象。 coutrendl; 共享数据的保护常对象举例class A public: A(int i,int j) x=i; y=j; . private: int x,y;const A a(3,4); /a是常对象,不能被更新#include using std:co
37、ut;using std:endl;class CBox / Class definition at global scope public: / Constructor definition CBox(double lv = 1.0, double bv = 1.0, double hv = 1.0) cout endl Volume() xBox.Volume(); private: double m_Length; / Length of a box in inches double m_Width; / Width of a box in inches double m_Height;
38、 / Height of a box in inches;int main() CBox match(2.2, 1.1, 0.5); / Declare match box CBox cigar(8.0, 5.0,1.0); / Declare cigar box const CBox cigar(8.0, 5.0,1.0); if(cigar.Compare(match) cout endl match is smaller than cigar; else cout endl match is equal to or larger than cigar; cout Volume() xBo
39、x.Volume(); 如果如果const成员函数的定义出现在类的外部成员函数的定义出现在类的外部,那么:那么:double Volume() const;doubleCBox: Volume() const应该总是将所有应该总是将所有不修改当前类对不修改当前类对象的成员函数声象的成员函数声明为明为const六、多文件结构和编译预处理命令1、一个源程序可以划分为多个源文件:、一个源程序可以划分为多个源文件: 类声明文件(类声明文件(.h文件)文件) 类实现文件(类实现文件(.cpp文件)文件) 类的使用文件(类的使用文件(main()所在的所在的.cpp文件文件)利用工程来组合各个文件。利用工
40、程来组合各个文件。 例如:建立例如:建立CBox类,用类,用Box.h和和Box.cpp实实现,在现,在Box.cpp开头放入开头放入#include,在主在主函数文件开头放入函数文件开头放入#include,这样这样Box.cpp和和main.cpp文件分别编译生成各自的目文件分别编译生成各自的目标文件标文件.obj。再连接生成可执行文件。再连接生成可执行文件.exe。2、外部变量和外部函数 外部变量是可以为多个源文件所共享的全局变量。用extern关键字来说明。 /1.cpp int i=3; int main() i+; /2.Cpp extern int i;/声明一个在其他文件中定义
41、的外部变量。 如果在全局作用域中定义一个static变量,那么它的作用范围仅限于定义它的编译单元。外部函数 在类之外定义的函数,都是全局的,可以在不同的文件中使用,但调用前要声明函数原型。或者用extern修饰。3、编译预处理命令#include 文件包含指令将一个源文件嵌入到当前源文件中该点处。#include 按标准方式搜索,文件位于C+系统目录的include子目录下#include文件名首先在当前目录中搜索,若没有,再按标准方式搜索。#define 宏定义指令定义符号常量,很多情况下已被const定义语句取代。定义带参数宏,已被内联函数取代。#undef删除由#define定义的宏,使
42、之不再起作用。如何使用多文件结构文件-新建-工程-windows console Application在class view中,选择工程,右键选择Add class.在弹出的对话框中写入类的名字。系统自动生成.h(头文件)及.cpp文件。在.h文件中有一条预处理指令:#pragma once该指令的作用是防止编译器在编译过程中多次将该文件打开并嵌入到源代码中。这样不会出现某个类的多次定义。这条指令时Microsoft特有的指令。在其他开发环境中不支持。如果需要在其他环境中编译,可以在头文件中使用:#ifndef BOX_H#define BOX_Hcode#endif#includeusin
43、g namespace std;enum ErrorTypeinvalidArraySize,memoryAllocationError,indexOutOfRange;char* errorMsg = invalid Array Size,memory Allocation Error,indexOutOfRange;class Point public: Point() X=Y=0; coutDefault Constructor called.endl; Point(int xx,int yy) X=xx; Y=yy; cout Constructor called.endl; Poin
44、t() coutDestructor called.endl; int GetX() return X; int GetY() return Y; void Move(int x,int y) X=x; Y=y; private: int X,Y;class ArrayOfPoints public: void Error(ErrorType erroe,int badIndex = 0) const; ArrayOfPoints(int n) numberOfPoints=n; points=new Pointn; ArrayOfPoints(ArrayOfPoints& arraya) n
45、umberOfPoints=arraya.numberOfPoints; points=new PointnumberOfPoints; for(int i = 0;inumberOfPoints;i+) pointsi=arraya.pointsi; ArrayOfPoints() coutDeleting.endl; numberOfPoints=0; delete points; Point& operator(int n) if(n numberOfPoints-1) Error(indexOutOfRange,n); return pointsn; Point& Element(in
46、t n) if(n numberOfPoints-1) Error(indexOutOfRange,n); return pointsn; private: Point *points; int numberOfPoints;void ArrayOfPoints: Error(ErrorType error,int badIndex ) const couterrorMsgerror; if(error =indexOutOfRange) coutbadIndex; coutendl; exit(1); int main()int number;coutnumber; ArrayOfPoints poi
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 精密压铸技术改进与升级方案
- 隧道交通疏导与应急预案方案
- 人防工程环境影响评估方案
- 桥梁施工人员培训与管理方案
- 草莓棚转租合同协议书
- 产品付费推广协议书
- 建筑工程认证机构数字化转型路径研究
- 与医院共建位协议书
- 买断土地口头协议书
- 个人资金监管协议书
- 日本经营管理讲解课件
- (正式版)DB36∕T 2169-2025 《公路工程固化土应用技术规范》
- 脊柱后凸矫形课件
- 2025年公安机关人民警察高级执法资格考试真题及答案解析
- 抖音品牌授权协议书范本
- 耕地建房排查培训课件
- 国际展览会策划与执行指南
- 手术部保洁员培训课件
- 心脑血管疾病科普课件
- 基于大数据分析的智能采购技术方案设计与优化路径
- 2025上海对外经贸大学金融管理学院教学秘书招聘笔试参考题库附答案解析
评论
0/150
提交评论