`
evasiu
  • 浏览: 165387 次
  • 性别: Icon_minigender_2
  • 来自: 广州
博客专栏
Fa47b089-e026-399c-b770-017349f619d5
TCP/IP详解卷一>阅读...
浏览量:12256
社区版块
存档分类
最新评论

c++ premier -- 变量和基本类型

 
阅读更多

呵呵,要重新拾起C++啦。也在这个过程中深入了解一种语言的定义吧。希望能够通过这本书,把c++真正熟悉起来。这段日子对Steve Jobs的stay hungry, stay foolish有了一种更深的理解。现在我要从基础开始!

 

变量和基本类型中需要注意的点:

 

1. 反斜杠("\")的使用

处理长字符有一个更基本的方法(不常用),这个方法依赖于很少使用的程序格式化特性:在一行的末尾加一反斜杠可以将此行和下一行当作同一行处理。如:

std::cou\
t << "Hi” << st\
d::endl;

 等价于

std::cout<<"Hi"<<std::endl;

 但是必须注意的是,反斜杠符号必须是该行的尾字符——不允许后面有注释或空格。

 

2. 初始化和赋值是不同的操作。(有关这一点,我现在也没有看到更多的解释,但希望强化这个问题的概念,书的后面应该会解释的)。

 

3. 定义和声明的区别(这个在C专家编程里面也有提到)

定义用于为变量分配存储空间,还可以为变量指定初始值。在一个程序中,变量有且仅有一个定义。

声明用于向程序表明变量的类型和名字(所以定义也是一个声明),可以通过使用extern关键字声明变量而不定义它。

 

4. 引用(&)

引用只是它绑定的对象的另一个名字,作用在引用上的所有操作都是事实上作用在该引用绑定的对象上。

const引用是 指向const对象的引用。(这一点在C专家编程中,有过与指针相关的论述)

const int ival = 1023; 
const int& refval = ival; //valid
int& refval2 = ival; //error

 试想,如果最后一条语句可以使用的话,原先作为不可修改的变量ival现在岂不是可以通过refval2进行修改了?所以最后一条语句不合法。

另外,const引用可以初始化为不同类型的对象或初始化为右值,如:

int i=42;
const int &r = 42;
const int &r2 = i+2;

double dval = 3.14;
const int &ri = dval;

//编译器会把这些最后两句代码转换成:

int temp = dval; //先转型
const int& ri = temp;

 如果ri不是const,那么可以给ri赋一新值,这样做不会修改dval(因为实际上ri指向的是temp),而是修改了temp,期望对ri的赋值会修改dval的程序员会发现dval并没有被修改。

因此,非const引用只能绑定到与该引用同类型的对象,而const引用则可以绑定到不同但相关的类型的对象或绑定到右值。

 

5. c++中的struct和class

c++中,用class和struct关键字定义的唯一差别在于默认访问级别:默认情况下,struct的成员为public,而class的成员了private。

(现在我有个问题就是,struct中也可以定义方法吗?希望后续的阅读能够找到答案)

 

6. const对象默认为文件的局部变量

2.4中的第2点我不是很确定。按照书中的说法,非const变量默认为extern,要使const变量能够在其他的文件中访问,必须显式地指定它为extern。(先记在这里了)

 

7. 头文件用于声明而不是用于定义

//下面的语句都是定义,所以不应该放在头文件里
extern int ival = 10; //已经初始化,所以是定义
double fica_rate;     //没有extern,所以是一个定义

 

因为头文件包含在多个源文件中,所以不应该含有变量或函数的定义。

对于头文件不应该含有定义这一规则,有三个例外,头文件可以定义类、值在编译时就已经知道的const对象和inline函数,这是因为编译器需要它们的定义(不只是声明)来产生代码。例如:为了产生能够定义或使用类的对象的代码,编译器需要知道组成该类型的数据成员。同样还需要知道能够在这些对象上执行的操作。类定义提供所需要的信息。

 

8. 定义在头文件中的const对象

前面说过const变量默认时是定义该变量的文件的局部变量,“正如我们现在所看到的,这样设置默认情况的原因在于允许 const变量定义在头文件中”。(似乎正回答了我前面提到的问题。。。)

下面有一段推理,“一般来说,常量表达式是编译器在编译时就能够计算出结果的表达式。当const整型变量通过常量表达式自我初始化时,这个const整型变量就可能是常量表达式。而const变量要成为常量表达式,初始化式必须为编译器可见。为了能够让多个文件使用相同的常量值,const变量和它的初始化式必须是每个文件可见的。而要使初始化式可见,一般都把这样的const变量定义在头文件中。”

这种行为有一个很重要的含义:当我们在头文件中定义了const变量后,每个包含该头文件的源文件都有了自己的const变量,其名称和值都一样。

 

9. 头文件保护符(header guard)

头文件经常#include其他头文件,例如我们自定义的类self的头文件可能会包含string头文件,而使用该自定义的类的文件main.c也可能会包含string头文件。这种情况下,main.c不会也不应该知道self头文件中包含了string头文件。因此,设计头文件时,应使其可以多次包含在同一个源文件中,我们必须保证多次包含同一头文件不会引起该头文件定义的类和对象被多次定义。使得头文件安全的通知做法,是使用预处理器定义头文件保护符。

#ifndef SELF_H
#define SELF_H
//here goes the definition
...

#endif

 为了保证头文件在给定的资源中只处理过一次,我们首先检测#ifndef。第一次处理头文件时,测试会成功,因为SELF_H还未定义。下一条语句定义了SELF_H,那样的话,如果我们编译的文件恰好又一次包含了该头文件,#ifndef指示会发现SELF_H已经定义,并且忽略该头文件的剩余部分。

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics