[C++ Primer] 第2章: 变量
初始化与赋值
当对象在创建时获得了一个特定的值, 我们就说这个对象被初始化了. 在使用=号时, 如果创建了新的对象, 则为初始化而非赋值.
初始化不是赋值, 初始化的含义是创建变量时赋予其一个初始值, 而赋值的含义是吧对象的当前值擦除, 而以一个新值代替.
在使用列表初始化(C++11支持)时, 如果存在丢失信息的风险, 则编译器会报错.
默认初始化: 如果变量定义时没有指定初值, 则被默认初始化. 默认值由变量类型和变量定义的位置决定. **定义在任何函数之外的内置类型变量被初始化为0, 定义在函数体内部的内置类型变量将不被初始化. **
定义在函数内的内置类型的对象如果没有初始化, 其值是未定义的.类对象如果没有显示初始化, 其值由类决定.
变量声明与定义
分离式编译允许将程序分割为若干个文件, 每个文件可被独立编译. 为了支持分离式编译, C++语言将声明和定义区分开来, 声明使得名字为程序所知, 如果一个文件想要使用别处定义的名字则必须包含对那个名字的声明, 而定义负责创建与名字关联的实体.
如果想声明一个变量而非定义它, 就在变量名前添加关键字extern且不要显示地初始化变量:
extern int i; //声明i而非定义i
int i; //声明并定义i
任何包含显式初始化的声明即成为定义. extern语句如果包含了初始值就不再是声明, 而变成定义了:
extern int i = 10; //定义
在函数体内部, 如果试图初始化一个由extern关键字标记的变量, 将引发错误. 变量能且只能被定义一次, 但是却可以多次声明.
变量能且只能被定义一次, 但是可以被声明多次.
要在多个文件中使用同一个变量, 就必须将声明和定义分离, 变量的定义必须出现在且只能出现在一个文件中, 其他用到该变量的文件必须对其进行声明, 却绝对不能重复定义.
引用和指针
引用就是一个对象的别名, 引用必须初始化, 一旦初始化完成, 引用将和它的初始值对象一直绑定在一起, 无法令引用重新绑定到另外一个新的对象上.
引用并非对象, 他只是为一个已经存在的对象所起的另外一个名字而已.
引用本身不是对象, 因此无法定义引用的引用.
两种方法为指针置空
int *p = nullptr; // C++11标准推荐, 字面值类型
int *q = NULL // 需要#include <cstdlib>, 预处理器变量
int *r = 0; // OK, 初始化字面值常量0
现代C++程序最好使用nullptr, 避免使用NULL.
const限定符
初始化和const: 如果利用一个对象去初始化另一个对象, 则它们是不是const都无关紧要:
int i = 42;
const int ci = i;
int j = ci;
const常量特征只在构造完成时才生效, 因此初始化时可以给它一个初始值.
默认状态下, const对象仅在文件内有效, 如果想要在一个文件中定义它, 而在多个文件中声明并使用它, 解决的办法是对于const变量不管是声明还是定义都添加extern关键字.
extern const int bufSize = fcn(); // file1.cc定义并初始化一个常量, 该常量能被其他文件访问
extern const int bufSize;// file1.h中, 与file1.cc中是同一个bufSize
如果想在多个文件之间共享const对象, 必须在比变量的定义之前添加extern关键字.
可以把引用绑定到const对象上, 称之为对常量的引用, 对常量的引用不能被用作修改它所绑定的对象.
const int ci = 1024;
const int &r = ci; // 引用以及对应的对象都是常量
int i = 42;
const int &r2 = i; // 允许将const int &绑定到普通int对象上, 但是不允许通过r2改变i的值
如果const关键字出现在星号左侧, 表示被指物是常量, 如果出现在星号右侧, 表示指针自身是常量.
顶层const表示指针本身是个常量, 底层const表示指针所指的对象是一个常量.
constexpr和常量表达式
常量表达式(const expression)指值不会改变并且在编译过程中就能得到计算结果的表达式
类型别名
typedef int ElementType;
using ElementType = int; // C++11标准, 与typedef等价的定义
类型别名的错误理解
const char *pstring;
const pstring cstr = 0; // cstr是指向char的常量指针, 即cstr本身是const的不可改变, 但所指对象可以改变
const char *cstr2; // cstr2是指向const char的指针, 即cstr2本身可以改变, 所指对象不可改变.
auto类型说明符:
auto让编译器通过初始值来推算变量的类型, 显然auto定义的变量必须有初始值.
使用引用类型推断结果是引用所绑定的对象的类型, 如果希望推断出引用类型必须明确指出.
int i = 0, &r = i;
auto a = r; // a的类型为int 而非int &
auto &b = r; // b的类型为int &
auto一般会忽略掉顶层const, 同时底层const则会被保留下来
const int ci = i, &cr = ci;
auto b = cr; // b是int类型 而非const int
const auto c = ci; // ci推断出是int, c是const int类型
decltype类型指示符
如果decltype使用的表达式是一个变量, 则decltype返回该变量的类型, 包括顶层const和引用在内.
const int ci = 0, &cj = ci;
decltype(ci) x = 0; // x类型为const int
decltype(cj) y = x; // y类型为const int &, 因此必须初始化
// 以下是decltype几个特殊用法
decltype(f()) sum = x; // x类型为f()返回值类型, f()并没有被调用
int i = 42, *p = &i, &r = i;
decltype(r + 0) b; // 加法结果是int, 因此b是未初始化的int
decltype(*p) c; // 若表达式内容是解引用操作, 则得到引用类型, 因此c是int &, 必须初始化
decltype((i)) d; // 双层括号的结果永远是引用类型, 而单层括号只有当表达式是引用类型其结果才是引用类型