C++标识符的作用域与可见性

news/2024/6/26 22:14:05

标识符是一个由程序员定义的名称,为标识区别变量、函数和数据类型等,代表程序的某些元素,变量名就是标识符的一个展现。
作用域讨论的是标识符的有效范围可见性讨论的是标识符是否可以被引用。在一个函数中声明的变量就只能在这个函数中起作用,这就是受变量的作用域和可见性的限制。作用域与可见性既相互联系又存在很大差异。

1.作用域

作用域是一个标识符在程序正文中有效的区域。C++中标识符的作用域有函数原型作用域、局部作用域(块作用域)、类作用域和命名空间作用域。

(1)函数原型作用域

函数原型作用域是C++程序中最小的作用域。在函数原型中一定要包含形参的类型说明。在函数原型声明时形式参数的作用范围就是函数原型作用域。
例如,有以下函数声明:

double area(double radius);

标识符radius的作用(或称有效)范围就在函数area形参列表的左右括号之间,在程序的其他地方不能引用这个标识符。因此标识符radius的作用域称为函数原型作用域。
【注意】由于在函数原型的形参列表中起作用的只是形参类型,标识符并不起作用,因此是允许省去的。但考虑到程序可读性,通常还是要在函数原型声明时给出形参标识符。

(2)局部作用域

【例】
在这里插入图片描述
这里函数fun的形参列表中声明了形参a,在函数体内声明了变量b,并且用a的值初始化b。接下来,在if语句中又声明了变量c。a,b,c都具有局部作用域,只是它们分别属于不同的局部作用域。
函数形参列表中形参的作用域,从形参列表中的声明处开始,到整个函数体结束是之处为止。因此,形参a的作用域从a的声明处开始,直到fun函数的结束处为止。
函数体内声明的变量,其作用域从声明处开始,一直到声明所在的块结束为止。 所谓块,就是一对大括号括起来的一段程序。在这个例子中,函数体是一个块,if语句后的分支体又是一个较小的块,二者是包含关系。因此,变量b的作用域从声明处开始,到它所在的块(即整个函数体)结束处为止,而变量c的作用域从声明处开始,到它所在的块,即分支体结束为止。
具有局部作用域的变量也称为局部变量。

(3)类作用域

类可以被看成是一组有名成员的集合,类X的成员m具有类作用域,对m的访问方式有如下三种:
①如果X的成员函数中没有声明同名的局部作用域标识符,那么在该函数内可以直接访问成员m。也就是说类的数据成员m在这样的函数中都起作用。
X的成员函数中没有声明与类中数据成员同名的局部作用域标识符的情况:

class X
{
public:
	X(int m = 0):m(m){}
	int getM()
	{
		//int m=7;
		return m;
	}
private:
	int m;
};
int main()
{
	X b(5);
	b.getM();
	cout << b.getM() << endl;
}

运行结果:
在这里插入图片描述
X的成员函数中声明了与类中数据成员同名的局部作用域标识符的情况:

class X
{
public:
	X(int m = 0):m(m){}
	int getM()
	{
		int m=7;
		return m;
	}
private:
	int m;
};
int main()
{
	X b(5);
	b.getM();
	cout << b.getM() << endl;
}

运行结果:
在这里插入图片描述
②通过表达式x.m或者X::m。这正是程序中访问对象的最基本方法。X::m的方式用于访问类的静态成员。
③通过ptr->m这样的表达式,其中ptr为指向X类的一个对象指针。
C++中,类及其对象还有其他特殊的访问和作用域规则。

(4)命名空间作用域

命名空间的概念:
在C/C++中,变量、函数和类都是大量存在的,这些变量、函数和类的名称将都存在于全局作用域中,可能会导致很多冲突。使用命名空间的目的是对标识符的名称进行本地化,以避免命名冲突或名字污染,namespace关键字的出现就是针对命名空间的。
命名空间的定义:
定义命名空间,需要使用到namespace 关键字,这个关键字的作用就是定义一个命名空间。
命名空间的语法形式如下:

namespace 命名空间名 
{
	命名空间内的各种声明(函数声明、类声明...)
}

一个命名空间确定了一个命名空间的作用域,凡是在该命名空间之内声明的、不属于前面所述各个作用域的标识符,都属于该命名空间的作用域。在命名空间你内部可以直接引用当前命名空间中声明的标识符,如果需要引用其他命名空间的标识符,需要使用下面的语法:

命名空间名::标识符名

例如:
定义一个命名空间

namespace SomeNs
{
	class SomeClass
	{
    	...
    };
    void somemeFunc()
    {
    	...
    }
}

如果需要引用类名SomeClass或者函数名somemeFunc,需要使用下面的方式:

SomeNs::SomeClass obj1;//声明一个SomeNs::SomeClass型的对象obj1

有时,在标识符前总使用这样的命名空间限定会显得过于冗长,为了解决这一问题,C++又提供了using语句,using语句有两种形式:

using 命名空间::标识符名;//1
using namespace 命名空间名;//2

第一种形式将指定的命名空间中的标识符暴露在当前的作用域中,使得在当前作用域中可以之间引用该标识符;第二种形式将指定命名空间内的所有标识符暴露在当前的作用域中,使得在当前作用域中可以直接引用该命名空间内的任何标识符。
事实上,C++标准程序库的所有标识符都被声明在std命名空间内,平时所用到的cincoutendl等标识符都是如此,因此,前面的程序中都使用了using namespace std;。如果去掉这条语句,则引用相应的标识符需要使用std::cinstd::coutstd::endl这样的语法。
命名空间也允许嵌套,例如:

namespace OuterNs
{
	namespace InnerNs
	{
		class SomeClass
		{
			...
		};
	}
}

引用其中的SomeClass类,需要使用OuterNs::InnerNs::SomeClass的语法形式。
此外,还有两类特殊的命名空间——全局命名空间和匿名命名空间。全局命名空间就是默认的命名空间,在显式声明的命名空间之外的标识符都在一个全局命名空间中。匿名命名空间是一个需要显式声明的没有名字的命名空间,声明方式如下:

namespace 
{
	匿名命名空间内的各种声明(函数声明、类声明、......)
}

在包含多个源文件的工程中,匿名命名空间常常被用来屏蔽不希望暴露给其他源文件的标识符,这是因为每个源文件的命名空间是彼此不同的,在一个源文件中没有办法访问其他源文件的匿名命名空间。
【例】作用域实例,以下声明的全局变量就具有命名空间作用域,它在整个文件中都有效。

int i;//在全局命名空间中的全局变量
namespace Ns
{
	int j;//在Ns命名空间中的全局变量
}
int main()
{
	i = 5;//为全局变量i赋值
	Ns::j = 6;//为Ns命名空间中的全局变量j赋值
	{   //子块1
		using namespace Ns;//使得在当前块中可以直接引用Ns命名空间中的标识符
		int a;    //局部变量,局部作用域
		a = 7;    
		cout << "a=" << a<< endl;//输出7
		int i = 8; //局部变量,局部作用域
		cout << "j=" << j << endl;//输出6
		cout << "i=" << i << endl;//输出8
	}
	cout << "i=" << i << endl;//输出5
	return 0;
}

运行结果:
在这里插入图片描述
在上面的例子中,变量i具有命名空间作用域,它属于全局命名空间,其有效作用范围到文件尾才结束。在主函数开始处给这个具有命名空间作用域的变量赋初值5,接下来在子块1中又声明了同名变量并赋初值8。所以第一次输出i的值为8,这是因为具有局部作用域的变量i把具有命名空间作用域的变量i隐藏了,具有命名空间作用域的变量i变得不可见。在程序运行到子块1结束后,变量i进行第二次输出时,输出的就是具有命名空间作用域的变量i的值5。变量j也具有命名空间作用域,它被声明在命名空间Ns当中,在主函数中通过Ns::j的方式引用并为其赋初值,接下来在块1中,通过using namespace Ns使得命名空间的标识符可以在该块中被直接引用,因此输出j的值时可以直接使用标识符j。
具有命名空间作用域的变量也称为全局变量。

2.可见性

从标识符引用的角度,来看标识符的有效范围,即标识符的可见性。程序运行到某一点,能够引用到的标识符,就是该处可见的标识符。

不同作用域之间的关系:

命名空间作用域最大,接下来依次是类作用域和局部作用域。如下图所示:
在这里插入图片描述
可见性表示从内层作用域向外层作用域“看”时能看到什么,可见性和作用域之间有着密切的关系。

作用域的可见性的一般规则如下:

(1)标识符要声明在前,引用在后。
(2)在同一作用域中,不能声明同名的标识符。
(3)在没有互相包含关系的不同的作用域中声明的同名标识符互不影响。
(4)如果两个或多个具有包含关系的作用域中声明了同名标识符,则外层的标识符在内层不可见。
【例】

int i;//在全局命名空间中的全局变量
namespace Ns
{
	int j;//在Ns命名空间中的全局变量
}
int main()
{
	i = 5;//为全局变量i赋值
	Ns::j = 6;//为Ns命名空间中的全局变量j赋值
	{   //子块1
		using namespace Ns;//使得在当前块中可以直接引用Ns命名空间中的标识符
		int i = 8; //局部变量,局部作用域
		cout << "i=" << i << endl;//输出8
	}
	cout << "i=" << i << endl;//输出5
	return 0;
}

运行结果:
在这里插入图片描述

上例是命名空间作用域与局部作用域互相包含的实例,在主函数内块1之外,可以引用具有该命名空间作用域的变量,也就是说它是可见的。当程序进入块1后,就只能引用具有局部作用域的同名变量,具有命名空间作用域的同名变量被隐藏了。这也是为什么两次输出的变量i的值不同。
【注意】作用域和可见性的原则不只适用于变量名,也适用于其他各种标识符,包括常量名、用户定义的类型名、函数名、枚举类型的取值等。


http://lihuaxi.xjx100.cn/news/1396489.html

相关文章

没有使用IaC的DevOps系统都是耍流氓 |IDCF

作者&#xff1a;徐磊 文章首发地址&#xff1a;https://smartide.cn/zh/blog/2022-1010-iac/ 作为现代软件工程的基础实践&#xff0c;基础设施即代码&#xff08;Infrastructure as Code, IaC&#xff09;是云原生、容器、微服务以及DevOps背后的底层逻辑。应该说&#xff…

vue : 无法加载文件 C:\Users\86182\AppData\Roaming\npm\vue.ps1,因为在此系统上禁止运行脚本。

windows11&#xff1a; PS E:\VueProjects> vue vue : 无法加载文件 C:\Users\86182\AppData\Roaming\npm\vue.ps1&#xff0c;因为在此系统上禁止运行脚本。有关详细信息&#xff0c;请参阅 https:/ go.microsoft.com/fwlink/?LinkID135170 中的 about_Execution_Policie…

【143. 重排链表】

来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 描述&#xff1a; 给定一个单链表 L 的头节点 head &#xff0c;单链表 L 表示为&#xff1a; L0 → L1 → … → Ln - 1 → Ln 请将其重新排列后变为&#xff1a; L0 → Ln → L1 → Ln - 1 → L2 → Ln - 2 → … 不…

GDAL OGR C++ API 学习之路 (5)OGRLayer篇 代码示例

GetStyleTable virtual OGRStyleTable *GetStyleTable () 返回图层样式表 返回: 指向不应由调用方修改或释放的样式表的指针 // 假设图层对象为 poLayer OGRStyleTable* poStyleTable poLayer->GetStyleTable(); if (poStyleTable ! nullptr) {// 处理样式表信息// ..…

Spring | Bean 作用域和生命周期

一、通过一个案例来看 Bean 作用域的问题 Spring 是用来读取和存储 Bean&#xff0c;因此在 Spring 中 Bean 是最核心的操作资源&#xff0c;所以接下来我们深入学习⼀下 Bean 对象 假设现在有⼀个公共的 Bean&#xff0c;提供给 A 用户和 B 用户使用&#xff0c;然而在使用的…

HCIE Security——防火墙互联技术

目录 一、防火墙接口互联接口 1.防火墙支持的接口及板卡 2.物理链接线缆 3.支持接口种类 &#xff08;1&#xff09;物理接口 &#xff08;2&#xff09;逻辑接口 二、相关配置命令 1.配置三层接口IP地址 2.配置PPPOE拨号接口 3.配置VLANIF接口、子接口、回环接口 4…

81%的消费者预算增加,今年返校季会有哪些机会?

对于跨境卖家来说&#xff0c;7-9月的重头戏莫过于“返校季”。多项调研表明&#xff0c;今年“返校季”消费者预计投入更多预算&#xff0c;并瞄准线上渠道。目前&#xff0c;亚马逊“返校季”促销活动正在火热进行中&#xff0c;许多产品甚至卖到脱销。 那么&#xff0c;今年…

XGBoost的基础思想与实现

目录 1. XGBoost VS 梯度提升树 1.1 XGBoost实现精确性与复杂度之间的平衡 1.2 XGBoost极大程度地降低模型复杂度、提升模型运行效率 1.3 保留了部分与梯度提升树类似的属性 2. XGBoost回归的sklearnAPI实现 2.1 sklearn API 实现回归 2.2 sklearn API 实现分类 3. XGBo…