目录
1.作用域2.对象生存期静态数据成员静态成员函数3.类的友元友元函数友元类4.共享数据的保护常对象常成员函数常引用Tips总结1.作用域
作用域是一个标识符在程序正文中有效的区域
作用域关系从大到小为:
命名空间作用域 > 类作用域 > 局部作用域
标识符的有效范围就是标识符的可见性,可见性的一般规则为:
标识符声明在前,使用在后在同一作用域中,不能声明同名的标识符在没有包含关系的不同的作用域中声明的同名标识符,互不影响如果在多个作用域中定义了同名标识符,则外层标识符在内层不可见2.对象生存期
如果对象生存期与程序的运行期相同,则它具有静态生存期
局部生存期对象诞生于声明点,结束于声明所在块执行完毕之时
命名空间作用域中声明的对象都具有静态生存期定义时未指定初值的基本类型静态生存期变量,会被赋予0值初始化如果要在函数内部的局部作用域中声明具有静态生存期的对象,则要使用关键字staticvoid f() { static int m=0; //m仅在f中能访问 m++; cout<静态数据成员
如果某个属性为整个类所共有,可以采用static关键字来声明为静态成员。
静态成员在每个类中只有一个副本,由该类的所有对象共同维护和使用在类中不能对static静态数据成员进行初始化,要初始化必须在类外进行定义class A{ public: static int m; }; int A::m=20; //在类外进行初始化 int main(){ A a,b; a.m=10; //可以赋值 cout<静态成员函数
静态成员函数可以通过类名或对象名来进行调用,非静态成员函数只能通过对象名来调用静态成员函数可以直接访问该类的静态数据和函数成员,而访问非静态成员,必须通过对象名class A{ public: static void f(A a); static void g(){ cout<<"hello"<3.类的友元
友元关系是一种数据共享机制,通过友元关系,一个普通函数或者类中的成员函数可以访问封装于另外一个类中的数据
为了确保数据的完整性和封闭性,建议尽量少地使用友元
友元函数
友元函数是类中用关键字friend修饰的非成员函数,在它的函数体中可以通过对象名访问类的private和protected成员
class A{ public: friend int main(); private: int x; }; int main(){ A a; a.x=5; //访问 private对象 }友元类
若B类为A类的友元类,则B类的所有成员函数都是A类的友元函数,都可以访问A类的private和protected成员
class A{ public: friend class B; private: int x,y; void f(){ cout<注意(!!!)
1.友元关系是单向的。
B类是A类的友元,B能访问A的私有数据,但A不能访问B的私有数据
2.友元关系不能传递。B是A的友元,C是B的友元,C和A之间如果没有声明就不存在友元关系
3.友元关系不能被继承。
4.共享数据的保护
常对象
数据成员值在对象的整个生存期间不能被修改的对象叫做常对象,一般用const进行修饰
常对象必须进行初始化,而且不能被更新
声明常对象的语法形式为:
const 类型说明符 对象名;
class A{ public: A(int i,int j):x(i),y(j){···} private: int x,y; }; int main(){ const A a(3,4); //a为常对象,不能被更新 }常成员函数
使用const关键字修饰的函数为常成员函数
语法:
类型说明符 函数名(参数表)const;
如果将一个对象说明为常对象,通过该常对象只能调用它的常成员函数,不能调用其他函数(常对象唯一的对外接口方式)
class A{ public: A(int i,int j):x(i),y(j){···} void print(){ cout<使用const说明的数据成员为常数据成员
类中说明了常数据成员,任何函数不能对它赋值,构造函数对它进行初始化只能通过初始化列表
class A{ public: //常数据成员只能通过构造函数的初始化列表来获得初值 A(int i):a(i){···} private: const int a; //常数据成员 static const int b;//静态常数据成员 }; const int A::b=10; //静态常数据成员在类外初始化常引用
如果在声明引用时用const修饰,被声明的引用就是常引用,常引用所引用的对象不能被更新
常引用的声明形式:
const 类型说明符 & 引用名;
一个常引用,无论是绑定到一个普通对象,还是一个常对象,通过该引用访问该对象时,都只能把该对象当做常对象
class Point{ public: Point(int x,int y):x(x),y(y){} friend float dist(const Point &p1,const Point &p2); private: int x,y; }; float dist(const Point &p1,const Point &p2){ //常引用作形参 double x=p1.x-p2.x; double y=p1.y-p2.y; return sqrt(x*x+y*y); } int main(){ const Point m1(1,1),m2(4,5); cout<Tips
include书写方式include <文件名>,按照标准方式搜索嵌入文件,文件位于编译环境include子目录下。示例:include
include “文件名”,在当前目录下搜索嵌入文件,如果搜不到则转为标准搜索。示例:include "point.h"
外部变量如果一个变量可以在本文件和其他文件中使用,称为外部变量,用 extern 关键字说明extern int n;//声明一个在其他文件定义的外部变量n
对于外部函数,加不加 extern 效果都是一样的
如果在定义变量和函数时使用static关键字,可以让该变量和函数无法被其他编译单元引用
动态内存分配运算符new的功能是动态分配内存,语法形式为
new 数据类型(初始化参数列表)
对于基本数据类型,如果不希望在分配内存后设立初值,可以把括号省去
int* point = new int;
如果保留括号,但括号中不写任何数值,则表示用0来进行初始化
int* point = new int( );
运算符delete用来删除由new建立的对象,释放指针所指向的内存空间,格式为:
delete 指针名;
用new分配的内存,必须用delete加以释放,否则会造成“内存泄漏”,而且只用delete进行一次删除,对同一内存空间多次使用delete进行删除会导致运行错误
用new创建一维数组时,在方括号后加或者不加小括号的区别和“new T( )”一样,用delete删除时在指针名前面要加“[ ]”
深复制与浅复制浅复制只是对指针的复制,复制后两个指针指向同一个内存空间;深复制不但对指针进行复制,而且对指针指向的内容进行复制,经深复制后的指针是指向两个不同地址的指针int* p = new int[100]; //不设置初值
int *p = new int[100] ( ); //用0进行初始化
delete[ ] p;默认复制构造函数,进行的是浅复制,对指针复制后会出现两个指针指向同一个内存空间的情况,内存空间会被析构函数释放两次,导致运行错误
this指针this指针是一个隐含于每一个类的非静态成员函数中的特殊指针,它用于指向正在被成员函数操作的对象当局部作用域中声明了与类成员同名的标识符时,对该标识符的直接引用代表的是局部作用域中所声明的标识符,这时为了访问该类成员,可以通过this指针解决这一问题必须要自己定义复制构造函数,使复制后的对象指针成员有自己的内存空间,即进行深复制,这样就避免了内存泄漏发生。
class A{ public: void display(int x){ this->x=x; //前一个x为数据成员,后一个x为形参 } private: int x; };总结
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!
下一篇:C++中类的转换函数你了解吗
X 关闭
X 关闭
- 15G资费不大降!三大运营商谁提供的5G网速最快?中国信通院给出答案
- 2联想拯救者Y70发布最新预告:售价2970元起 迄今最便宜的骁龙8+旗舰
- 3亚马逊开始大规模推广掌纹支付技术 顾客可使用“挥手付”结账
- 4现代和起亚上半年出口20万辆新能源汽车同比增长30.6%
- 5如何让居民5分钟使用到各种设施?沙特“线性城市”来了
- 6AMD实现连续8个季度的增长 季度营收首次突破60亿美元利润更是翻倍
- 7转转集团发布2022年二季度手机行情报告:二手市场“飘香”
- 8充电宝100Wh等于多少毫安?铁路旅客禁止、限制携带和托运物品目录
- 9好消息!京东与腾讯续签三年战略合作协议 加强技术创新与供应链服务
- 10名创优品拟通过香港IPO全球发售4100万股 全球发售所得款项有什么用处?