C语言学习笔记—P23(<C语言高阶>+指针的进阶<1>+题例+图解)

指针的主题,我们在初级阶段的《指针》章节已经接触过了,我们知道了指针的概念:
1. 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
2. 指针的大小是固定的 4/8 个字节( 32 位平台 /64 位平台)。
3. 指针是有类型,指针的类型决定了指针的 +- 整数的步长,指针解引用操作的时候的权限。
4. 指针的运算。

这个章节,我们继续探讨指针的高级主题。

首先分析以下代码:

#include<stdio.h>
int main()
{
	int a = 10;
	printf("%p\n",  &a);
	return 0;
}

 注:上图中打印地址只供分析使用,因为在不同机器中打印的地址是不相同的!

1. 字符指针

在指针的类型中我们知道有一种指针类型为字符指针 char* ;

一般使用:

int main()
{
    char ch = 'w';
    char *pc = &ch;
    *pc = 'w';
    return 0;
}

 还有一种使用方式如下:

int main ()
{
    const char* pstr = "hello bit" // 这里是把一个字符串放到 pstr 指针变量里了吗?
    printf ( "%s\n" , pstr );
    return 0 ;
}
代码 const char* pstr = "hello bit.";  特别容易让同学以为是把字符串 hello bit 放到字符指针 pstr 里了,但是 / 本质是把字符串 hello bit. 首字符的地址放到了 pstr 中。

注:上图中打印地址只供分析使用,因为在不同机器中打印的地址是不相同的! 

上面代码的意思是把一个常量字符串的首字符 h 的地址存放到指针变量 pstr 中。
再来分析以下代码:
int main()
{
	//int a = 10;
	//printf("%p\n",  &a);
	char ch = 'w';
	char* p = &ch;

    char* p = "abcdef";
	return 0;
}

 

分析以下代码:

#include <stdio.h> 	​
int main()
{
	char* p = "abcdef";
	*p = 'w';
	return 0;
}​

  结论:以上代码逻辑错误!此处无法对p进行更改,因为abcdef是存储在内存的只读数据区!

因此,代码严谨写法如下:(添加const表示不可更改!)

#include<stdio.h>
int main()
{
	const char* p = "abcdef";
	*p = 'w';

	return 0;
}

注: 这样编写代码更加严谨! 

那就有可这样的面试题:

#include <stdio.h>
int main()
{
	char str1[] = "hello bit.";
	char str2[] = "hello bit.";
	const char* str3 = "hello bit.";
	const char* str4 = "hello bit.";
	if (str1 == str2)
		printf("str1 and str2 are same\n");
	else
		printf("str1 and str2 are not same\n");

	if (str3 == str4)
		printf("str3 and str4 are same\n");
	else
		printf("str3 and str4 are not same\n");

	return 0;
}

这里最终输出的是:  

 这里str3str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当 几个指针。指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化 不同的数组的时候就会开辟出不同的内存块。所以str1str2不同,str3str4不同。

分析以下代码:

int main()
{
	char arr1[] = "abcdef";
	char arr2[] = "abcdef";

	const char* str1 = "abcdef";
	const char* str2 = "abcdef";

	if (arr1 == arr2)
		printf("arr1==arr2\n");
	else
		printf("arr1!=arr2\n");

	if (str1 == str2)
		printf("str1==str2\n");
	else
		printf("str1!=str2\n");

	return 0;
}

 注:

1.arr1和arr2是两个不同的数组,而数组名又代表首元素的地址,这是内存中两块不同的空间,其起始地址不相同,故arr1!=arr2。

2.str1和str2为常量字符串,不能被更改,并且其存储的内容相同,因此没有必要存储两份!故str1和str2指向相同!

3.图中打印地址为假设,讲解使用,不必深究此地址内容!

注意:

 

 

2. 指针数组

在《指针》章节我们也学了指针数组,指针数组是一个存放指针的数组。
这里我们再复习一下,下面指针数组是什么意思?
int* arr1 [ 10 ];             // 整形指针的数组
char * arr2 [ 4 ];           // 一级字符指针的数组
char ** arr3 [ 5 ];         // 二级字符指针的数组

 

 

int main()
{
	char* arr[] = { "abcdef", "qwer", "zhangsan" };
	int i = 0;
	int sz = sizeof(arr) / sizeof(arr[0]);

	for (i = 0; i < sz; i++)
	{
		printf("%s\n", arr[i]);
	}
	return 0;
}

 

 分析以下代码: 

int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 2,3,4,5,6 };
	int arr3[] = { 3,4,5,6,7 };

	int* arr[] = {arr1, arr2, arr3};

	int i = 0;
	for (i = 0; i < 3; i++)
	{
		int j = 0;
		for (j = 0; j < 5; j++)
		{
			printf("%d ", arr[i][j]);//*(*(arr+i)+j)
		}
		printf("\n");
	}

	return 0;
}

 此代码模拟了二维数组。

后记:
●由于作者水平有限,文章难免存在谬误之处,敬请读者斧正,俚语成篇,恳望指教!

                                                               ——By 作者:新晓·故知

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值