C++的数据共享与保护你了解吗

 更新时间:2022年03月13日 14:51:09   作者:风云诀4  
这篇文章主要为大家详细介绍了C语言零基础入门的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

1.作用域

作用域是一个标识符在程序正文中有效的区域

作用域关系从大到小为:

命名空间作用域 > 类作用域 > 局部作用域

标识符的有效范围就是标识符的可见性,可见性的一般规则为:

  • 标识符声明在前,使用在后
  • 在同一作用域中,不能声明同名的标识符
  • 在没有包含关系的不同的作用域中声明的同名标识符,互不影响
  • 如果在多个作用域中定义了同名标识符,则外层标识符在内层不可见

2.对象生存期

如果对象生存期与程序的运行期相同,则它具有静态生存期

局部生存期对象诞生于声明点,结束于声明所在块执行完毕之时

  • 命名空间作用域中声明的对象都具有静态生存期
  • 定义时未指定初值的基本类型静态生存期变量,会被赋予0值初始化
  • 如果要在函数内部的局部作用域中声明具有静态生存期的对象,则要使用关键字static
void f()
{
	static int m=0;	//m仅在f中能访问 
	m++;
	cout<<m<<endl;
}

静态数据成员

如果某个属性为整个类所共有,可以采用static关键字来声明为静态成员。

  • 静态成员在每个类中只有一个副本,由该类的所有对象共同维护和使用
  • 在类中不能对static静态数据成员进行初始化,要初始化必须在类外进行定义
class A{
public:
	static int m; 
};
int A::m=20;			//在类外进行初始化 
int main(){
	A a,b;
	a.m=10;			    //可以赋值 
	cout<<a.m<<endl;
	cout<<b.m<<endl;	//两者的值相同
}

静态成员函数

  • 静态成员函数可以通过类名或对象名来进行调用,非静态成员函数只能通过对象名来调用
  • 静态成员函数可以直接访问该类的静态数据和函数成员,而访问非静态成员,必须通过对象名
class A{
public:
	static void f(A a);
	static void g(){
		cout<<"hello"<<endl;
	}
	void fun(){
		cout<<"world"<<endl;
	}
private:
	int x;
	static int y;
};
int A::y=10;
void A::f(A a){
	cout<<a.x<<endl;    //必须通过对象名访问
	cout<<y<<endl; 		//直接访问该类的静态数据成员 
	g();				//直接访问该类的静态成员函数 
}
int main(){
	A a;
	A::f(a);            //直接通过类名调用
	return 0; 
}

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<<x<<endl;
	} 
};
class B{
public:
	void print(){
		a.x=5;
		a.y=10;
		cout<<a.y<<endl;//可以访问A类对象的私有成员 
		a.f();			//可以访问A类对象的私有函数 
	}
private:
	A a;
}; 

注意(!!!)

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<<x<<" "<<y<<endl;
	}
	void print() const{
		//常成员函数
		cout<<x<<" const "<<y<<endl;
	}
private:
	int x,y;
};
int main(){
	A a(1,2);
	a.print();  //调用 void print() 
	const A b(3,4);
	b.print();  //调用 void print() const 
}

使用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<<dist(m1,m2)<<endl;    //两点间距离 
}

Tips

  • include书写方式
    • include <文件名>,按照标准方式搜索嵌入文件,文件位于编译环境include子目录下。示例:include <iostream>
    • 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语言实现可保存的动态通讯录的示例代码

    C语言实现可保存的动态通讯录的示例代码

    这篇文章主要为大家详细介绍了如何利用C语言实现一个简单的可保存的动态通讯录,文中的示例代码讲解详细,对我们学习C语言有一定帮助,需要的可以参考一下
    2022-07-07
  • 基于c语言知识点的补遗介绍

    基于c语言知识点的补遗介绍

    本篇文章是对c语言知识点的一些补遗进行详细的分析介绍,需要的朋友参考下
    2013-05-05
  • 深入解析int(*p)[]和int(**p)[]

    深入解析int(*p)[]和int(**p)[]

    以下是对int(*p)[]和int(**p)[]的使用进行了详细的分析介绍,需要的朋友可以参考下
    2013-07-07
  • C语言打印杨辉三角形的示例代码

    C语言打印杨辉三角形的示例代码

    杨辉三角形,又称帕斯卡三角形、贾宪三角形、海亚姆三角形,它的排列形如三角形。本文将为大家介绍通过C语言实现打印杨辉三角形的示例代码,需要的可以参考一下
    2022-02-02
  • C语言实现socket简单通信实例

    C语言实现socket简单通信实例

    这篇文章主要介绍了C语言实现socket简单通信的方法,是学习C语言网络编程非常基础而又实用的实例,需要的朋友可以参考下
    2014-09-09
  • 查找算法之二分查找的C++实现

    查找算法之二分查找的C++实现

    今天小编就为大家分享一篇关于查找算法之二分查找的C++实现,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • C语言全局变量和局部变量的示例代码

    C语言全局变量和局部变量的示例代码

    本文主要介绍了C语言全局变量和局部变量的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-05-05
  • 深入VC回调函数的使用详解

    深入VC回调函数的使用详解

    本篇文章是对VC回调函数的使用进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++中hashmap的一些使用建议

    C++中hashmap的一些使用建议

    由于hashmap不是c++ stl中标准实现,这样在跨平台使用时就可能会出现问题,下面这篇文章主要给大家介绍了关于C++中hashmap的一些使用建议,需要的朋友可以参考下
    2023-03-03
  • C语言实现文件内容的加密与解密

    C语言实现文件内容的加密与解密

    文件内容需要加密与解密功能的原因主要有两个方面:保护数据安全和确保数据完整性,所以接下来小编就给大家介绍一下如何通过C语言实现文件内容加密与解密,需要的朋友可以参考下
    2023-08-08

最新评论