More Effective C++ 条款一

原创 2001年09月21日 23:26:00

条款一:指针与引用的区别

指针与引用看上去完全不同(指针用操作符’*’’->’,引用使用操作符’.’),但是它们似乎有相同的功能。指针与引用都是让你间接引用其他对象。你如何决定在什么时候使用指针,在什么时候使用引用呢?

 <?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

首先,要认识到在任何情况下都不能用指向空值的引用。一个引用必须总是指向某些对象。因此如果你使用一个变量并让它指向一个对象,但是该变量在某些时候也可能不指向任何对象,这时你应该把变量声明为指针,因为这样你可以赋空值给该变量。相反,如果变量肯定指向一个对象,例如你的设计不允许变量为空,这时你就可以把变量声明为引用。

“但是,请等一下”,你怀疑地问,“这样的代码会产生什么样的后果?”

char *pc = 0;          // 设置指针为空值

 

char& rc = *pc;        // 让引用指向空值

                      

这是非常有害的,毫无疑问。结果将是不确定的(编译器能产生一些输出,导致任何事情都有可能发生),应该躲开写出这样代码的人除非他们同意改正错误。如果你担心这样的代码会出现在你的软件里,那么你最好完全避免使用引用,要不然就去让更优秀的程序员去做。我们以后将忽略一个引用指向空值的可能性。

 

因为引用肯定会指向一个对象,在C里,引用应被初始化。

string& rs;             // 错误,引用必须被初始化

string s("xyzzy");

string& rs = s;         // 正确,rs指向s

 

指针没有这样的限制。

string *ps;             // 未初始化的指针

                        // 合法但危险

不存在指向空值的引用这个事实意味着使用引用的代码效率比使用指针的要高。因为在使用引用之前不需要测试它的合法性。

void printDouble(const double& rd)

{

    cout << rd;         // 不需要测试rd,

}                       // 肯定指向一个double

 

相反,指针则应该总是被测试,防止其为空:

void printDouble(const double *pd)

{

  if (pd) {             // 检查是否为NULL

    cout << *pd;

 }

}

 

指针与引用的另一个重要的不同是指针可以被重新赋值以指向另一个不同的对象。但是引用则总是指向在初始化时被指定的对象,以后不能改变。

 

string s1("Nancy");

string s2("Clancy");

 

string& rs = s1;         // rs 引用 s1

 

string *ps = &s1;        // ps 指向 s1

 

rs = s2;                 // rs 仍旧引用s1,

                         // 但是 s1的值现在是

                         // "Clancy"

 

ps = &s2;                // ps 现在指向 s2;

                         // s1 没有改变

 

总的来说,在以下情况下你应该使用指针,一是你考虑到存在不指向任何对象的可能(在这种情况下,你能够设置指针为空),二是你需要能够在不同的时刻指向不同的对象(在这种情况下,你能改变指针的指向)。如果总是指向一个对象并且一旦指向一个对象后就不会改变指向,那么你应该使用引用。

 

还有一种情况,就是当你重载某个操作符时,你应该使用引用。最普通的例子是操作符[]。这个操作符典型的用法是返回一个目标对象,其能被赋值。

 

vector<int> v(10);       // 建立整形向量(vector),大小为10;

                         // 向量是一个在标准C库中的一个模板(见条款35)

v[5] = 10;               // 这个被赋值的目标对象就是操作符[]返回的值

 

如果操作符[]返回一个指针,那么后一个语句就得这样写:

*v[5] = 10;

 

但是这样会使得v看上去象是一个向量指针。因此你会选择让操作符返回一个引用。(这有一个有趣的例外,参见条款30

 

当你知道你必须指向一个对象并且不想改变其指向时,或者在重载操作符并为防止不必要的语义误解时,你不应该使用指针。而在除此之外的其他情况下,则应使用指针。

《More Effective C++》条款27:如何让类对象只在栈(堆)上分配空间?

昨天一个同学去网易面试C++研发,问到了这么一个问题:如何限制一个类对象只在栈(堆)上分配空间? 一般情况下,编写一个类,是可以在栈或者堆分配空间。但有些时候,你想编写一个只能在栈或者只能在堆上面分...
  • hxz_qlh
  • hxz_qlh
  • 2013年10月26日 21:27
  • 6096

《Effective C++》:条款28-条款29

条款28避免返回handles指向对象内部成分:指的是不能返回对象内部数据/函数的引用、指针等。 条款29为异常安全而努力是值得的:指的是要有异常处理机制,避免发生异常时造成资源泄露等问题。...
  • KangRoger
  • KangRoger
  • 2015年02月19日 19:47
  • 1399

Effective C++ 条款2

尽量以const、enum、inline替换#define首先,大家要明白一个道理。#define是什么,有什么作用。很简单,大家都知道#define实现宏定义,如下代码:#define Flag 1...
  • u011058765
  • u011058765
  • 2015年06月19日 12:06
  • 502

《Effective C++》:条款41-条款42

条款41了解隐式接口和编译期多态 条款42了解typename的双重意义条款
  • KangRoger
  • KangRoger
  • 2015年03月10日 22:13
  • 1250

《Effective C++》:条款44-条款45

条款44将与参数无关的代码抽离templates 条款45运用成员函数模板接受所有兼容类型...
  • KangRoger
  • KangRoger
  • 2015年03月12日 22:01
  • 1512

Effective Modern C++ 条款23 理解std::move和std::forward

Effective Modern C++ 条款23
  • big_yellow_duck
  • big_yellow_duck
  • 2016年08月30日 17:11
  • 1242

《Effective C++》资源管理:条款13-条款15

在系统中,资源是有限的,一旦用完必须归还给系统,否则可能会造成资源耗尽或其他问题。例如,动态分配的内存如果用完不释放会造成内存泄漏。 这里说的资源不仅仅是指内存,还包括其他,例如文件描述符、网络连接、...
  • KangRoger
  • KangRoger
  • 2015年01月14日 21:46
  • 1336

Effective C++——条款10条,条款11和条款12(第2章)

条款10:    令operator=返回一个reference to *this Have assignment operators return a reference to *this ...
  • yiranant
  • yiranant
  • 2015年08月29日 23:35
  • 607

《Effective C++》:条款38-条款39

条款38通过复合塑模树has-a 或根据某物实现出 条款39明智而审慎的使用private继承...
  • KangRoger
  • KangRoger
  • 2015年03月08日 21:32
  • 1163

《Effective C++》:条款46-条款47

条款46需要类型转换时请为模板定义非成员函数 条款47请使用traits class表现类型信息条款...
  • KangRoger
  • KangRoger
  • 2015年03月13日 22:10
  • 1376
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:More Effective C++ 条款一
举报原因:
原因补充:

(最多只允许输入30个字)