复合类型是指基于其他类型而定义的类型。 而这里介绍的是引用和指针。并且指针和引用都提供了对其他对象的间接访问。

  

引用

  引用还是很好理解的,就是为对象起了另外一个名字,引用类型引用另外一种类型。 通常将声明符协程 &d 的形式来定义引用类型,其中d是声明的变量名。   

  注意:引用不是对象。 

  一般,在初始化变量时,如int a = 3; int b = a; 这时,这里b的初始化就会将a的值拷贝到新建的对象中。即int b = 3这里新建了一个对象来存储,对a和b的操作是独立的。 而定义引用时,程序会把引用和他的初始值绑定在一起,而不是将初始值拷贝给引用。如下所示:

#include <iostream>
using namespace std;
int main()
{
cout << "beginning!" << endl;
int a = ;
int &b = a;
cout << "b的值为:" << b << endl; // 2
b = ;
cout << "a的值为:" << a << "\t b的值为:" << b << endl; // a 和 b都是5
return ;
}

  如上,我们可以看到:b就是a的引用,这里b的值也是2,所以引用就是给a了一个别名b,并且并没有给b一个新的对象,而是将b和a绑定在了一起,我们修改b的值,发现a和b的值同时修改成了5,这就是引用。

  那么引用的目的是怎样的呢? 一般来说,引用比指针更容易操作、也更不容易出错。可以代替指针。

  需要注意的:

  • 引用必须赋初值。
  • 引用和之前的类型要严格匹配。
  • 如上的int &b 是指引用类型b是指向int类型的引用。

指针

  指针是指向另外一种类型的复合类型。

  指针如果要指向某个对象,是存放着某个对象的地址。 如下:

    int a = ;
int *p = &a;
cout << &a << endl;

  其中&a就是获取a的地址。  int *p是指p是一个指向int类型对象的指针。

  以上,我们已经定义了指针,那么如何利用指针访问它所指的对象呢? 使用 * 解引用符即可。 

    int a = ;
int *p = &a;
cout << *p << endl; //

  如上所示: 我们可以得到 *p 的值为10。

  

  注意: 某些符号具有多重含义!这里讲解 * 和 &。

  int i = 42;

  int &r = i;  // &紧随类型名出现,这里的&为引用。

  int *p; // *紧随类型名出现,这里的 * 为定义指针类型的p。  

  p = &i; // 这里的&出现在表达式中,是一个取地址符。

  *p = i; // 这里的 * 出现在表达式中,是一个解引用符。

  int &r2 = *p; // 这里的 * 出现在表达式中,是一个解引用符。

  

  

空指针

  即Null pointer,它是不指向任何对象的,一般我们建议:初始化所有的指针,并且在可能的情况下,尽量等定义了对象之后在定义指向他的指针,如果不清楚它指向何处,就初始化为空指针。

  定义空指针一般有三种方法,如下所示:

#include <iostream>
#include <cstdlib>
using namespace std;
void IsNull(int *p);
int main()
{
cout << "beginning!" << endl;
int *p = ;
IsNull(p);
int *p1 = nullptr;
IsNull(p1);
int *p2 = NULL;
IsNull(p2);
int a = , *p3 = &a;
IsNull(p3);
return ;
}
void IsNull(int *p)
{
if (p == nullptr)
{
cout << "空指针" << endl;
}
else
{
cout << "非空指针" << endl;
}
}

  三种方法为赋值为0、赋值为NULL、赋值为 nullptr,最后一种方法是c++11所新定义的,也是我们推荐的,在判断一个指针是否为空指针时,我们只需要判断其是否与nullptr相等即可。 注意:NULL的值就是0,是在 cstdlib 库中定义的。

  注意:如果指针没有初始化为空指针或者指向明确的对象,那么很有可能会导致整个程序的崩溃,并且难以入手解决问题。

    int i = ;
int *pi = ; // 定义pi为空指针
int *pi2 = &i; // pi2指向i
cout << *pi2 << endl;
int *pi3; pi3 = pi2; // 注意:这里并不是Pi3指向pi2,而是pi3指向了和pi2相同的对象,即这时pi3同样指向i。
pi2 = ;
cout << *pi3; // 42
cout << *pi2; // 报错

另外,如果指针指向为空,则if判断时的结果为false,且指针之间可以使用 == 来判断是否相等,比如指向同一个对象、都为空等等。

注意: void*指针可以存放任何类型的对象,不受限制,但利用也比较有限。

  

总结

  指针和引用都提供了访问对象的其他方法,但为什么c++中同时有指针和引用呢? 如下:

可以看到,在第一个例子中,显然使用引用更加舒服一些。 另外,我们怎么理解这种参数的使用方式呢,实际,f1要求的两个指针,比如f1在调用时是f1(p1, p2),即const complex * x = p1。 因为p1是指针,可以直接赋值。

指向指针的指针  

  

        int i = 42;
int *p1 = &i;
int *p2 = nullptr;
p2 = p1;
cout << *p2; //42

如上所示,p2 = p1的意思是p2指针指向p1指针所指向的对象,而不是p2指向p1,那么如何定义p2指向p1,即指向指针的指针呢?

    int i = ;
int *p1 = &i;
int **p2 = &p1;
cout << **p2; //

如上所示: p2就是一个指向指针的指针。

指向指针的引用

  指针是对象,所以可以有指向指针的引用,如下:

    int i = ;
int *p1 = &i;
int *&r = p1; // 这里的r就是指向指针的引用
*r = ;
cout << *r; //

  如上所示:p1是指向int型变量i的指针,而 r 是指向指针型变量的引用,即我们对r的操作就是对p1的操作,因此,我们可以得到最后的结果为365。

参考链接:http://www.stroustrup.com/bs_faq2.html#pointers-and-references

最新文章

  1. JAVA中遇到 UTF-八 序列的字节 1 无效
  2. 【bzoj2333】 [SCOI2011]棘手的操作 可并堆+lazy标记
  3. Node.js执行存储过程
  4. 开源一个基于nio的java网络程序
  5. GCD之dispatch queue深入浅出
  6. [Windows驱动]流媒体驱动开发
  7. 回调函数的应用误区4(c/s OK版本回调小程序)
  8. 有感,懂市场比懂产品重要,懂产品比懂技术重要——想起凡客诚品和YY语音了
  9. C#与.Net Framework的各种版本和联系
  10. fiddler--手机https
  11. 如何在同一系统中启动多个 TOMCAT
  12. hadoop学习;大数据集在HDFS中存为单个文件;安装linux下eclipse出错解决;查看.class文件插件
  13. 【转载】关于api-ms-win-crt-runtimel1-1-0.dll缺失的解决方案
  14. Synergy简单使用小记
  15. c# System.Console
  16. Spark Streaming之窗口函数和状态转换函数
  17. Centos7下部署两套python版本并存环境的操作记录
  18. 关于gzip zgrep zcat 的使用
  19. golang常用模块介绍
  20. 解决win10休眠后无法唤醒

热门文章

  1. 更改Ubuntu默认python版本的两种方法python-&gt; Anaconda
  2. windows10; ERROR 1010 (HY000): Error dropping database (can&#39;t rmdir &#39;./test/&#39;, errno: 17);默认数据库位置查找
  3. ubuntu下非交互式安装MySQL
  4. visual studio的试用版评估期已结束 解决办法
  5. CentOS 利用Yum安装mysql后无法启动(MySQL Daemon failed to start.)
  6. day36(动态代理)
  7. linux初学terminal命令(1)ls、cd、su、man、pwd、useradd、passwd、cat、Ctrl+C、Ctrl+Z、Ctrl+L
  8. linux 修改密码
  9. Android记录10--android.os.NetworkOnMainThreadException异常解决办法
  10. Linked List