指针就该这样学

深入了解指针

本片文章以代码形式来讲解不同情况下不同指针(即地址)的含义,通过认识不同指针的应用进一步对指针有更深一步认识。
对于以下代码,首先我们得先知道两个信息
1.sizeof(数组名) - 数组名表示整个数组 ,sizeof(数组名)计算的是整个数组的大小。(数组名单独放在sizeof内部的时候)
2.&数组名 - 数组名表示整个数组,取出的是整个数组的地址。
3.除此之外,所有的的数组名都是数组首元素的地址。另外当a+1的时候就不能表示2的结论,sizeof(a+1)不能表示1的结论。

int main ()
{
int a[] ={1,2,3,4,};
return 0;
}
printf("%d\n",sizeof(a));  //16

数组名单独的放在sizeof里面,计算的是整个数组的大小。即4*4=16

printf("%d\n",sizeof(a+0)); // 4/8

数组名不是单独的放在sizeof里面,那么数组名表示数组名首元素的 地址,a+0表示第一个元素的地址指针即地址,指针的大小 4/8

printf("%d\n",sizeof(*a)); // 4

数组名不是单独的,a表示数组名,数组名即首元素的地址,那么*解引用得到首元素,sizeof(*a)计算的是首元素的大小。

printf("%d\n",sizeof(a+1)); // 4/8

数组名不是单独的放在sizeof里面,那么数组名表示数组名首元素的地址,a+1表示第二个元素的地址,指针即地址,指针的大小 4/8

printf("%d\n",sizeof(a[1]));//4

a[1]表示数组第二个元素,sizeof(a[1])计算的是第二个元素的大小

printf("%d\n",sizeof(&a));//4/8

&a单独在sizeof里,取出的是整个数组的大小。那么sizeof(&a)计算的是整个数组地址的大小,指针即地址,指针的大小 4/8

printf("%d\n",sizeof(*&a));//16

&a表示整个数组的地址,用数组指针表示 int (*p)[4] =&a;
p表示指针,指向带有4个元素的,每个元素是整形的数组,解引用后得到的是整个数组a的大小即:16

printf("%d\n",sizeof(&a+1));//4/8

sizeof里&a得到整个数组的地址,加1后跳过一个数组,即跳过a数组到下一块空间的起始地址,指针即地址,指针大小为 4/8

printf("%d\n",sizeof(&a[0]));//4/8

a[0]是首元素,那么&a[0]得到的是首元素的地址,指针即地址,指针的大小 4/8

printf("%d\n",sizeof(&a[0]+1));//4/8

首先取到首元素的地址再加1得到第二个元素的地址,指针即地址,指针的大小 4/8

return 0;

}

int main()
{
    //六个元素
	char arr[] = { 'a','b','c','d','e','f' };
	return 0;
}

在这里提一下只有字符串才自带 ‘\0’

printf("%d\n", sizeof(arr)); //6

arr是数组名,并且单独存在的。在sizeof(arr)计算的是整个数组的大小!char为1byte,六个元素:1*6 = 6

printf("%d\n", sizeof(arr + 0));//4/8

arr并不是单独的放在sizeof里,故arr表示数组首元素的地址,加上0地址不变,指针即地址,指针的大小为4/8

printf("%d\n", sizeof(*arr)); //1

arr并不是单独的放在sizeof里,故arr表示数组首元素的地址,解引用后得到首元素的值.。故大小为1

printf("%d\n", sizeof(arr[1]));//1

这里arr[1]访问数组的第二个元素,故大小为1

printf("%d\n", sizeof(&arr));//4/8

&arr得到的是整个数组的地址,指针即地址,指针的大小为4/8

printf("%d\n", sizeof(&arr+1));//4/8

&arr得到的是整个数组的地址,加1后跳过一个数组,即跳过arr数组到下一块空间的起始地址,指针即地址,指针大小为 4/8

printf("%d\n", sizeof(&arr[0] + 1));//4/8

&arr[0]得到首元素的地址,再加1得到第二个元素的地址,指针即地址,指针大小为 4/8

int main()
{
	char arr[] = { 'a','b','c','d','e','f' };
	return 0;
}
printf("%d\n", strlen(arr)); //随机值
arr是数组名为数组首元素地址,strlen函数从首元素位置数字符直到遇见'\0'返回字符个数。而数组中并没有'\0',strlen函数一周往后找直到遇见'\0',而这个'\0'并不知道它的位置,故返回的是随机值。
printf("%d\n", strlen(arr+0));//随机值

arr数组名为首元素地址, 同样arr+0表示的是首元素地址,数组中没有‘\0’,故返回值为随机值

printf("%d\n", strlen(*arr));//err

arr数组名为首元素地址,解引用后得到首元素,而strlen()函数计算a字符的话把他的Ascll码值97作为地址,以这个地址去找字符并不知道找到了哪里故报错

printf("%d\n", strlen(arr[1]));//err

同样的道理arr[1]为数组第二个元素,即字符b strlen()函数计算a字符的话把他的Ascll码值97作为地址,以这个地址去找字符并不知道找到了哪里,故报错

printf("%d\n", strlen(&arr));

这里&arr取出的数组地址,数组的地址同样指向的是数组首元素的地址,传给strlen函数后就变成了字符的首地址,发生类型的转换。同样没有’\0’,故为随机值

printf("%d\n", strlen(&arr+1));

&arr为数组的地址,加1指向另一块数组空间的地址,从这个地址出发数字符并不知道什么时候遇见’\0’,故同样为随机值

printf("%d\n", strlen(&arr[0] + 1));

&arr[0]得到数组首元素的地址,首元素地址加1得到第二个元素的地址,strlen函数从第二个元素开始数字符,直到遇见’\0’,这里并没有’\0’,故为随机值

int main()
{
	char arr[] = "abcdef";
	return 0;
}

在这里提一下,sizeof计算字符串时,字符串末尾包含结束标志’\0’,也要算进字符个数中

printf("%d\n", sizeof(arr));//7

sizeof函数计算字符和字符串时返回的是字符个数,sizeof(arr)计算的是真个字符串的大小而计算字符串时字符串的结束标志’\0’也会被计入其中,字符串中字符6个加上结束标志故为7个字符

printf("%d\n", sizeof(arr+0));// 4/8

arr+0在sizeof里面数组名没有单独的放在sizeof里面故arr表示的是首元素的地址,首元素地址加0,仍然是首元素的地址,指针即地址,故为 4/8

printf("%d\n", sizeof(*arr));//1

arr同样没有单独的放在sizeof里面故表示数组首元素地址,解引用后得到首元素,一个字符的大小为1

printf("%d\n", sizeof(arr[1]));// 1

同样的道理arr[1]表示第二个元素,字符的大小为1

printf("%d\n", sizeof(&arr));// 4/8

&arr表示的是整个数组的地址,指针即地址,故大小为4/8

printf("%d\n", sizeof(&arr+1));// 4/8

&arr表示的是整个数组的地址,加1跳过一个数组的地址,仍然是地址,故大小为4/8

printf("%d\n", sizeof(&arr[0] + 1)); // 4/8

&arr[0]得到首元素的地址,加1后得到第二个元素的地址,故大小为 4/8

int main()
{
	char arr[] = "abcdef";
	return 0;
}

在这里提一下strlen函数在计算字符串时,字符串的结束标志即’\0’,不会计入字符数里

printf("%d\n", strlen(arr));// 6

在这里arr表示的是首元素地址,而strlen函数从首元素开始数字符个数,字符串包含的结束标志不计入,故返回6

printf("%d\n", strlen(arr + 0));//6

同样在这里arr表示的是首元素地址,加0仍然是首元素的地址,从首元素开始计入字符返回6

printf("%d\n", strlen(*arr));//err

arr表示首元素地址,解引用后得到首元素即字符a,strlen()函数计算a字符的话把他的Ascll码值97作为地址,以这个地址去找字符并不知道找到了哪里,故报错

printf("%d\n", strlen(arr[1]));//err

同样arr[1]表示数组第二个元素,即字符b,strlen()函数计算a字符的话把他的Ascll码值97作为地址,以这个地址去找字符并不知道找到了哪里,故报错

printf("%d\n", strlen(&arr));

这里&arr取出的数组地址,数组的地址同样指向的是数组首元素的地址,传给strlen函数后就变成了字符的首地址,发生类型的转换。同样没有’\0’,故为随机值

printf("%d\n", strlen(&arr + 1));

&arr为数组的地址,加1指向另一块数组空间的地址,从这个地址出发数字符并不知道什么时候遇见’\0’,故同样为随机值

printf("%d\n", strlen(&arr[0] + 1));//5 

&arr[0]得到首元素的地址,加1得到第二个元素的地址,传给strlen函数后,strlen函数从第二个元素开始计入字符,不包括结束标志,故返回5

int main()
{
	char* p = "abcdef";
	return 0;
}
printf("%d\n" ,sizeof(p));// 4/8

首先p是指针,p里存放的是字符串首元素的地址,故字节大小为4/8

printf("%d\n" ,sizeof(p+1));//4/8

p本来指向的是a的地址加1后指向b的地址,p是指针,指针的大小为4/8

printf("%d\n" ,sizeof(*p));// 1

p是指针指向的是a的地址,解引用后得到字符a字符的大小为1

printf("%d\n" ,sizeof(p[0]));//1

p[0] == *(p+0) == *p,故p[0]表示的是首字符a,它的大小同样是1

printf("%d\n" ,sizeof(&p));//4/8

p本是指针,&p取得是p的地址,指针即地址,大小为4/8

printf("%d\n" ,sizeof(&p+1));//4/8

&p是指针p的地址,加1呢是另外一个地址,同样大小为4/8

printf("%d\n" ,sizeof(&p[0] + 1));//4/8

p[0] == *(p+0) == *p
&p[0]取到的就是第一个元素的地址,加1后得到第二个元素的地址,指针即地址,指针的大小为4/8

int main()
{
	char* p = "abcdef";
	return 0;
}
printf("%d\n", strlen(p));//6

p是a的地址,而strlen函数从a的位置开始计算字符个数,返回字符个数为6

printf("%d\n", strlen(p + 1));

p是a的地址,加1为b的地址,strlen函数从b的位置开始计算字符个数,返回字符个数为5

printf("%d\n", strlen(*p));//err

p是a的地址,解引用后strlen()函数计算a字符的话把他的Ascll码值97作为地址,以这个地址去找字符并不知道找到了哪里,故报错

printf("%d\n", strlen(p[0]));

p[0] == *(p+0) ==
*p,得到字符a,strlen()函数计算a字符的话把他的Ascll码值97作为地址,以这个地址去找字符并不知道找到了哪里,故报错

printf("%d\n", strlen(&p));

这里&p取出的是字符串的地址,地址同样指向的是数组首元素的地址,传给strlen函数后就变成了字符的首地址,发生类型的转换。同样没有’\0’,故为随机值

printf("%d\n", strlen(&p + 1));

&p取出的是字符串的地址,加1跳过这块字符串地址,在另外一块地址处并不知道结束标志在哪里,故为随机值

printf("%d\n", strlen(&p[0] + 1)); //5

p[0]表示首元素字符a,&p[0]得到a的地址,加1得到字符串第二个字符的地址,然后strlen函数从第二个元素开始计算字符个数,故为5

int main()
{
	int a[3][4] = { 0 };
	return 0;
}
printf("%d\n", sizeof(a)); // 48

a是一个二维数组,单独的放在sizeof函数里,计算的是整个数组的大小 34sizeof(int) = 48

printf("%d\n", sizeof(a[0][0]));//4

a[0][0] 表示的是0行0列的一个元素,大小为4

printf("%d\n", sizeof(a[0]));//16

在这里插入图片描述
我们看一维数组 int a[4] 它的结构是 数组名+[i]. 而我们二维数组int a[3][4]
如上图,可以把他看成三个一维数组的组成,a[0]是第一行的数组名,a[1]是第二行的数组名,a[2]是第三行的数组名。那么a[0]、a[1]、a[2]则为数组名,而后面的i决定了一维数组中的第几个元素。
第一行的数组结构为a[0]+[i] 第二行的数组结构为a[1]+[i] 第三行的数组结构为a[2]+[i]
a[0]为数组名单独放在sizeof内部,计算的是整个一维数组的大小 16

printf("%d\n", sizeof(a[0] + 1));//4/8

a[0]并没有单独放在sizeof内部,那么它表示第一行数组的首元素地址。加1后得到第一行数组第二个元素的地址,大小为 4/8

printf("%d\n", sizeof(*(a[0] + 1)));// 4

a[0]并没有单独放在sizeof内部,那么它表示第一行数组的首元素地址。加1后得到第一行数组第二个元素的地址,再解引用后得到第二个元素。大小为
4

printf("%d\n", sizeof(a + 1));//4/8

a是二维数组的数组名,并没有单独的放在sizeof内部,故a表示数组的首元素的地址,对于二维数组的首元素他是第一行的数组,即a[0]的地址。再加1得到第二行的地址即a[1],地址的大小为4/8

printf("%d\n", sizeof(*(a + 1)));//16

a是二维数组的数组名,并没有单独的放在sizeof内部,故a表示数组的首元素的地址,对于二维数组的首元素他是第一行的数组,即a[0]的地址。再加1得到第二行的地址即a[1],再解引用得到a[1],即第二行的数组,大小为16

printf("%d\n", sizeof(&a[0] + 1)); //4/8

a[0]是第一行的数组名,&a[0]取出的是整个第一行数组的地址,再加1得到第二行数组的地址。地址的大小为4/8

printf("%d\n", sizeof(*(&a[0] + 1)));//16

a[0]是第一行的数组名,&a[0]取出的是整个第一行数组的地址,再加1得到第二行数组的地址。再解引用后得到二维数组的第二行的数组,大小为16

printf("%d\n", sizeof(*a));

a并没有单独放在sizeof内部,a就表示的是首元素的地址,即二维数组第一行的地址,再解引用得到二维数组的第一行数组,大小为16

printf("%d\n", sizeof(a[3]));
int main()
{
	short s = 5;
	int a = 4;
	printf("%d\n", sizeof(s = a + 6));
	printf("%d\n", s);
	return 0;
}

在这里插入图片描述

sizeof内部的表达式是不会执行的

  • 11
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 7
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值