Way on c & c++ 小记 [五] – 初始化列表与成员函参作用域

简介:



初始化列表与成员函参作用域

作者:Jason Lee @http://blog.csdn.net/jasonblog

日期:2010-04-11

 

[1]初始化列表

所谓初始化列表,从语法角度讲,就是构造函数圆括号后的冒号与左花括号之间的紧跟初始值的变量序列,如下是一个示例:

Demo(): b(2), a(b){

作为初始化列表,它以它的作用和初始化顺序而出名。

 

初始化列表有什么作用呢?我觉得可以一言以蔽之:初始化列表最重要的作用就是用来初始化不能被赋值的成员。比如引用必须在第一次出现的时候进行初始化,之后就不能再被赋值了。另外两种需要使用初始化列表的是被 const 修饰的类成员以及对象,而后者使用初始化列表进行初始化的主要原因方便和性能上的提高。

 

虽然初始化列表是一个有序的待初始化的变量序列,但真正对变量进行初始化的顺序却并不是如同初始化列表中所标明的一样,比如在如下代码中,是先使用 b (未知数值)初始化 a ,然后再用数值 2 初始化 b

class Demo { public: Demo(): b(2), a(b){ cout << a << endl << b << endl; }; private: int a,b; };

这是由于在类成员的声明次序中 a 先于 b 声明。

进一步反问: a 先于 b 声明会导致什么呢?

在词法分析的过程中,当分析出一个新的变量时,需要向符号表进行该变量的信息注册。在上述代码中, a 先于 b 声明,所以 a 先于 b 被注册到符号表中。以 Demo 类为例,该类仅有两个数据成员,并且在该类的符号表中成员 a 是在成员 b 之前。那么当我们要实例化一个对象(比如 demo )的时候,需要分配一定的内存空间,这时应该思考一下,如果是自己设计会如何进行内存的分配呢?我想,一种合理的方案是根据符号表中各项记录的信息由上及下进行空间的申请与分配,于是 a 先于 b 获得了内存空间,那么自然 a 要先被初始化。

以上只是一种个人猜想的解释。另外一种权威的解释是说为了保证析构函数进行对象消亡处理的效率,必须保证有序性,因为无序性会带来昂贵的开销。所以构造函数与析构函数对成员的调用是相反的。

但是如何保证有序呢?我个人觉得对符号表(或者类似机制)的依赖是不可少的,或许,有点殊途同归?

 

[2] 成员函参作用域

在类定义体外定义函数,必须指明相关作用域,比如 void Demo::func(){}

看下面一段代码:

 

#include <iostream> class Demo { public: typedef int index; void func(index); index func2(); private: }; void Demo::func(index arg){ std::cout << arg << std::endl; } Demo::index Demo::func2(){return 0;}; int main(){ Demo demo; demo.func(1); return 0; }

值得注意的是 func 函数后面的形参 arg index 类型,该类型在 Demo 类中定义,这里无需特指该类型的定义域。而对于 func2 函数的返回值,该返回值是一个 index 类型,如果没有指定作用域的话则会出错。

这是为什么呢?

以下是个人的理解:

函数具有函数作用域,类也具有类作用域,这种局部作用域是如何体现出来的呢?我想一种可能的方案是采用局部的符号表,而全局作用域则对应着全局符号表。我们都知道,当进入某个函数,相应地就进入到该函数的局部作用域,一个比较常见的例子就是当前的局部变量会覆盖掉同名的全局变量。

假设当前采用的是符号表的机制,从全局进入某个局部作用域后,当前符号表从全局切换到某个局部符号表。所以当我们在类定义体外定义的成员函数时,首先指明了类作用域,再进入类的成员函数,这时候符号表就切换成与 Demo 类的符号表,从而即可访问 index 类型,因为该类型在声明的时候已经注册到当前的符号表中。

而对于返回类型为 index 的函数 func2 来讲,由于返回类型是出现在类成员函数之前,所以如果在类定义体外定义,当前处于符号表,并无该类型的注册信息,无法查找到,于是就需要指明相关类作用域。

再稍微涉及一点,如果把原语句改动一下:

Demo::index func2(){return 0;}

则该函数是在全局中定义的一个函数,而非类的成员函数。这是因为指明 index 所属域的时候并没有进行符号表的切换,所以接下来是将 func2 函数注册到当前符号表,也就是全局符号表中。这又是为什么呢?换位思考一下,只是访问某个符号表中某个变量的注册信息值得切换当前符号表吗?

 

目录
相关文章
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
522 12
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
924 6
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
426 5
|
编译器 C语言 C++
C++入门3——类与对象2-2(类的6个默认成员函数)
C++入门3——类与对象2-2(类的6个默认成员函数)
267 3
|
存储 编译器 数据安全/隐私保护
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解2
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解
234 3
|
存储 编译器 C++
C++入门3——类与对象2-1(类的6个默认成员函数)
C++入门3——类与对象2-1(类的6个默认成员函数)
277 2
|
C++
C++构造函数初始化类对象
C++构造函数初始化类对象
220 0
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
11月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
275 0
|
11月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
443 0