一 、&

c的&被称为“寻址运算符”,作用是指向某变量的指针;

请看以下代码:

int main(void){

	int int_1 = 16;
printf("%d\n", int_1);
printf("%p\n", &int_1);
return 0;
}

输出的第一行会输出变量int_1的值,即16.

第二行会输出变量int_1的内存地址,即0x7ffd2fb152dc;

二 、*

*被称为“间接访问操作符”,作用是访问指针的变量指向的内容(因为指针的变量保存的是一个内存地址);

请看以下代码:

int main(void){

	int *int_2;
*int_2 = 32;
printf("%p\n", int_2);
printf("%d\n", *int_2);
return 0;
}

代码的第一行声明了一个int类型的指针(如果你不明白什么是指针,请先参考一下c语言指针的相关介绍文章).

第二行实际上是,运用*访问了这个指针指向的内存空间,给这个这个指针指向的内存空间赋值了32,理论上这两个操作同等于

int int_2 = 32;

此时你回头看看&,可能会发觉这两个操作符颇有一对反义词的味道;

再来看看输出了什么,

输出的第一行输出了int_2的值,因为int_2是一个指针,它的值就是一个内存地址,所以这里输出了一个内存地址

第二行运用了*操作符,输出了这个指针指向的变量,即32

三 、 进阶

我们来创建一个多维数组,叫做int_3

int int_3[3][3] = {
{-2 ,-1, 0},
{1 ,2, 3},
{4 ,5, 6}
};

如果你对c语言的数组有所了解,你肯定知道int_3,或者int_3[x],x是一个指针偏移量,它的意思是,基于原指针指向内存区域的长度n,内存地址偏移x*n个byte.(如果你不了解这个,不要紧,先看看『c语言入门经典』这本书,我也是看这本学的c).

要进一步了解,请看看以下代码:

int int_4 = {-2 ,-1, 0};
printf("%d\n", *(int_4+1));

它的意思类似于,指针int_4偏移1个量(实际上内存地址偏移了4个byte,因为它是一个int类型的数组)并输出这个指针指向的结果,这里是-1;

如果我们现在尝试把这个语法运用于多维数组int_3,看看会有什么有趣的事情发生:

printf("%p\n", *(int_3+1));

没错,这里输出了一个内存地址,它实际上是多维数组里面的sub数组的指针,

*(int_3)或者说是*int_3,是数组{-2 ,-1, 0};
*(int_3+1),是数组{1 ,2, 3};
*(int_3+2),是数组{4 ,5, 6};

但是这个指针跟指针变量会有一点点区别

printf("%p\n", *(int_3+2));
printf("%p\n", &(*(int_3+2)));

上面两条语句,会输出同一个地址,因为*(int_3+2)不是指针变量,它本身就是一个内存地址,而之指针变量实际上也是变量,只是这个变量保存了一个指针(内存地址)

int *Pint_3 = (int*)int_3;
printf("%p\n", Pint_3); //输出0x7fffd68ca700
printf("%p\n", int_3); //输出0x7fffd68ca700
printf("%p\n", &(Pint_3)); //输出0x7fffd68ca6f8

这里我们新建一个指针变量,并把它指向了int_3,用&地址访问符可以直接输出这个指针变量的地址

如何访问sub数组里面的item?我想你已经猜到了:

printf("%d\n", *(*(int_3+2)+2) );

*(*(int_3+2)+2)实际上跟int_3[2][2]是等价操作,这里自然就输出了6;

自然地,我们利用寻址运算符&就可以轻易的输出6所在的内存地址

printf("%p\n", &(*(*(int_3+2)+2)));

当然,这里的等价操作是&int_3[2][2].

我们尝试强行解析一番这个操作都经历了什么:

1.&(*(*(int_3+2)+2))

  int_3指针偏移了2个量,由于int_3的指针变量长度是12(3*4byte),实际上就是偏移了24(12*2)byte,此时指针指向了{4 ,5, 6}这个数组,实际上这个指针指向的地址不仅指向了这个数组,而且还指向了这个数组里面的4这个元素,但是int_3+2跟*(int_3+2)不是同一个指针!!!,因为他们的内存区块长度不一样,这导致了当你使用指针偏移量语法时(int_3+2)+2和*(int_3+2)+2明显不会一样(int_3+2)+2同等于int_3+4,而*(int_3+2)+2则是取int_3+2指向的内存地址保存的那个指针再偏移2哦

//输出数组的地址
printf("%p\n", int_3+2);
//输出数组的第一个元素
printf("%d\n", *(*(int_3+2)));

2.&(*(*(int_3+2)+2))

  输出int_3+2实际的值,因为int_3+2是一个指针,所以这里输出了一个指针,这个指针的类型是int数组,所以变量长度是4(1*4byte)。

3.&(*(*(int_3+2)+2))

  *(int_3+2)便宜了2个量,实际上就是偏移了8(2*4)byte,此时指针指向了6这个元素。

4.&(*(*(int_3+2)+2))

  输出*(int_3+2)+2实际的值,也就是6。

5.&(*(*(int_3+2)+2))

  输出6的内存引用地址,实际上同等于*(int_3+2)+2。

『完』

随机推荐

  1. 解决浏览器跨域限制方案之JSONP

    一.什么是JSONP JSONP即:JSON with Padding,是一种解决因浏览器跨域限制不允许访问跨域资源的方法. JSONP是一个非官方的协议,它允许在服务器端返回javascript标签 ...

  2. Java - 网络编程完全总结

    本文主要是自己在网络编程方面的学习总结,先主要介绍计算机网络方面的相关内容,包括计算机网络基础,OSI参考模型,TCP/IP协议簇,常见的网络协议等等,在此基础上,介绍Java中的网络编程. 一.概述 ...

  3. 【二】Spring Cloud 入门

    官网 版本号: SpringCloud中文网:https://springcloud.cc SpringCloud中文社区:http://springcloud.cn 以下代码就是Maven父子工程, ...

  4. Javaweb学习笔记——(十五)—————— sql复习

    sql复习 数据库管理系统(DBMS)的概述 1.什么是DBMS:数据的仓库 *方便查询 *可存储的数据量大 *保证数据的完整.一致 *安全可靠 2.DBMS的发展:今天主流数据库为关系型数据库管理系 ...

  5. luogu 3045 优先队列反悔/bzoj 2590

    N头奶牛,价格Pi,K张优惠券,优惠券购买降为Ci,不超过M的钱最多可买多少奶牛 先将c值k小的加入,将它们省下的钱加入优先队列(省下的钱由少到多),在将k+1-n用p排序,再逐个与优先队列中弹出的比 ...

  6. hasnMap的基本操作 源码(三)

    一.初始化: hashMap有四种初始化方式: public HashMap(int initialCapacity, float loadFactor) { if (initialCapacity ...

  7. java实现数据缓存

    摘抄自java并发实战 有时候需要对数据缓存.用Map缓存数据比较合适.但是由于对吞吐量,一致性,计算性能的要求,对数据进行缓存的设计还是需要慎重考虑的. 一.利用HashMap加同步 (1)说明 把 ...

  8. 梯度下降算法对比(批量下降/随机下降/mini-batch)

    大规模机器学习: 线性回归的梯度下降算法:Batch gradient descent(每次更新使用全部的训练样本) 批量梯度下降算法(Batch gradient descent): 每计算一次梯度 ...

  9. Swiper.js使用教程

    官网地址:(http://www.swiper.com.cn/). 一.Swiper.js简介: Swiper(前称Swiper master) 是一款免费以及轻量级的移动设备触控滑块的js框架,使用 ...

  10. python之字典的增删改查

    Python字典是另一种可变容器模型,且可存储任意类型对象,如字符串.数字.元组等其他容器模型.字典都是无序的,但查询速度快. 字典是一个key/value的集合,key可以是任意可被哈希(内部key ...