服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - 老生常谈C语言中指针的使用

老生常谈C语言中指针的使用

2022-09-27 15:23纸墨青鸢 C/C++

这篇文章主要为大家详细介绍了C语言中指针的使用,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

前提

指针,是C语言中的一个重要概念及其特点,也是掌握C语言比较困难的部分。指针也就是内存地址,指针变量是用来存放内存地址的变量,在同一CPU构架下,不同类型的指针变量所占用的存储单元长度是相同的,而存放数据的变量因数据的类型不同,所占用的存储空间长度也不同。有了指针以后,不仅可以对数据本身,也可以对存储数据的变量地址进行操作

指针描述了数据在内存中的位置,标示了一个占据存储空间的实体,在这一段空间起始位置的相对距离值。在 C/C++语言中,指针一般被认为是指针变量,指针变量的内容存储的是其指向的对象的首地址,指向的对象可以是变量(指针变量也是变量),数组,函数等占据存储空间的实体。

 

一.指针基础

 

1.1 变量指针

//首先我们声明一个变量
int a = 10;	//声明一个指针并指向该变量的地址	
int* b = &a;	printf("%d
", *b);

运行结果:10

 

1.2 数据指针

//首先我们声明一个数组变量
int c[10] = { 0,1,2,3,4,5,6,7,8,9 };		
printf("c的地址:%p
", c);	
printf("c[0]的地址:%p
", &c[0]);

运行结果:两个一样

Why? 数组本质为内存空间上连续的空间,而作为数组的地址,其实为首元素的地址,而对于数组,数组名称其实就是个指针(重点)然后我们创建数组指针

int* d=c
printf("d[9]的值为:%d
",*(d+9));
//另一种写法printf("d[9]的值为:%d
",d[9]);

运行结果:9

 

1.3 指针的本质

通过1.0和1.1的指示我们突然发现,整数的指针和整数数组的指针居然是同个类型!

事实上确实是一个东西,指针为指向变量地址的类型,所以对于指针来说是不需要类型的,但是为了程序规范性,增加了类型说法

下面使用无类型指针输出a的值(10)

void* p = &a;
printf("void指针的值:%d
", *(int*)p);
//对比b指针变量
printf("这是b的值:%p
", b);
printf("这是p的值:%p
", p);

老生常谈C语言中指针的使用

上述结果可以完全表明,指针变量储存的是地址,而且是单一变量的地址

接下来可以解释数组和数为啥指针就是同一个类型了

回到1.2所述 单个数占了1个空间(这里不提占用内存),而数组占了n个空间,而指针指向的都是首地址,而对于单个数来说

首地址就是数所在的地址,而对于数组来说,首地址就是头元素所在的地址

1.2解释了数组地址就是数组名本身,所以数组指针就是数组本身了,所以有下面这种写法

printf("直接使用数组名获取索引3的元素:%d
", c[3]);
printf("使用指向c的指针获取索引3的元素:%d
", d[3]);

老生常谈C语言中指针的使用

但是数组和指针数组还是有一定的区别

 

1.4 指针数组

/如果我们把指针看作一个变量,那么类比整数类型,我们一定可以声明一个指向指针的指针

int g = 50;
int* h = &g;
int** i = &h;
//输出一下
printf("h的值:%p
", h);
printf("i的值:%p
", *i);
//实验证明这是可行的,那我们尝试做一个数组
int j[5] = { 9,8,7,6,5 };
int* k[5] = { &j[0],&j[1],&j[2],&j[3],&j[4] };
int** l = k;
printf("k[1]的值:%p,对应的整数的值:%d
", k[1], *k[1]);
printf("l获取k1的值:%p,对应的整数的值:%d
", *(l + 1), **(l + 1));

老生常谈C语言中指针的使用

这就是指针数组,把指针当作一个变量看,指针也可以做数组

 

1.5 指针的移动

对于数组指针获取元素,上面展示了可以通过"变量名[索引]"的方式获取,当然也可以通过移动指针位置来获取

int* m = (int*)malloc(10 * sizeof(int));
for (int i = 0; i < 10; ++i)
	*(m++) = i;

指针移动到了分配内存的最后一块

“变量名[索引]“的方式表明了是以当前指针为数组头,偏移索引个单位为方法获取元素的方式,索引获取第8个元素

错误写法 printf(”%d”,m[8]);

正确写法

printf("%d
", *(m - 2));

老生常谈C语言中指针的使用

所以释放的时候要释放头指针,所以要回到头指针

for (int i = 0; i < 10; ++i)
	(--m);
free(m);

所以实际操作的情况下尽量不要去移动指针

int* n = (int*)malloc(10 * sizeof(int));
for (int i = 0; i < 10; ++i)
	*(n + i) = i;
printf("%d
", *(n + 8));
free(n);

老生常谈C语言中指针的使用

 

1.5 Scanf函数的解释

scanf函数为获取变量地址函数

首先我们看一下scanf函数的定义

scanf函数有一个格式符参数和一个可变参数

可变参数类型为指针!!!!

char o;

第一种直接写法

scanf("%c", &o);

printf(“输出的字符:%c ”, o);

同理

用指针获取

char* q = &o;
scanf("%c", q);
printf("通过指针获取输出的字符:%c
", *q);

老生常谈C语言中指针的使用

上面我们说了,指针可以代表数组,而c语言对字符串的定义就是字符数组

char r[] = { "Hello" };
scanf("%s", r);
printf("字符串的值:%s
", r);

这样我们就可以解释为什么对于获取字符串的时候,不用写&的原因了,因为字符数组本来就是指针!!

 

二.指针的进阶玩法

 

2.1 二维指针

由名字可得,二维指针就是指针的指针,指针可以代表数组,自然二维指针就可以代表二维数组

int u[4][4] = { {1,2,3,4},{2,3,4,5},{3,4,5,6},{4,5,6,7} };
int** v = u;

二维数组在内存空间上也是线性排列的,就如下顺序

1 2 3 4 2 3 4 5 3 4 5 6 4 5 6 7

所以二维数组获取元素也可以靠指针移动

printf("u[1][2]的值:%d
", *(v + 6));

也可以动态内存分配实现

int** w = (int**)malloc(4 * sizeof(int)); //分配行内存
for (int i = 0; i < 4; ++i)
	w[i] = (int*)malloc(4 * sizeof(int)); //分配列内存
for (int i = 0; i < 4; ++i)
	for (int j = 0; j < 4; ++j)
		w[i][j] = (j + 1) + i;
printf("w[1][2]的值:%d
", w[1][2]);
free(w); //记得释放

 

2.2 结构体指针

先声明一个结构体

struct people
{
	char* name;
	int age;
}SPeople;
//方便表示替换掉标识符
typedef struct people People;
People dr;
dr.name = "aaa";
dr.age = 18;
printf("dr的名字是:%s,年龄是:%d
", wusihao.name, wusihao.age);
People* lp = &wusihao; //注意属性获取的符号"->"
lp->name = "sss";
lp->age = 19;
printf("lp的名字是:%s,年龄是:%d
", lp->name, lp->age);

当然也可以动态内存分配使用

People* lps = (People*)malloc(1 * sizeof(People));
lps->name = "xxx";
lps->age = 20;
printf("lps的名字是:%s,年龄是:%d
", lps->name, lps->age);

 

结语

指针定义后,在不用时最好指向NULL,比如

int* s = &a;
printf("%p
", s);
s = NULL;

因为就算释放掉内存,但是指针指向的内存地址依旧可用(获取处于沉睡状态),所以最好不要去动它,不然很容易导致内存泄露

动态声明的指针一定要释放,free函数释放

动态分配内存最好要检查是否成功分配到

int* t = (int*)malloc(10 * sizeof(int));
if (t == NULL)
{
	//分配错误
	system("pause");
	return 0;
}
else
{
	//你的代码
	system("pause");
	free(t);
}

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注服务器之家的更多内容!     

原文链接:https://blog.csdn.net/qq_39935495/article/details/123031315

延伸 · 阅读

精彩推荐
  • C/C++一篇文章弄懂C++左值引用和右值引用

    一篇文章弄懂C++左值引用和右值引用

    左值(lvalue)和右值(rvalue)是 c/c++ 中一个比较晦涩基础的概念,这篇文章主要给大家介绍了关于如何通过一篇文章弄懂C++左值引用和右值引用的相关资料,需要的...

    爱吃菠萝不吃萝卜6732021-12-03
  • C/C++C++中extern "C"的用法

    C++中extern "C"的用法

    这篇文章主要介绍了C++中extern "C"的用法,是深入理解C++所应该掌握的概念,需要的朋友可以参考下...

    C++教程网10682021-01-28
  • C/C++详解C++实现线程安全的单例模式

    详解C++实现线程安全的单例模式

    这篇文章主要介绍了C++实现线程安全的单例模式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面...

    麻子来了5972021-07-25
  • C/C++一篇文章教你如何用C语言实现strcpy和strlen

    一篇文章教你如何用C语言实现strcpy和strlen

    这篇文章主要为大家介绍了C语言实现strcpy和strlen的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助...

    黎丶辰6122022-08-11
  • C/C++C语言中lseek()函数和fseek()函数的使用详解

    C语言中lseek()函数和fseek()函数的使用详解

    这篇文章主要介绍了C语言中lseek()函数和fseek()函数的使用详解,是C语言入门学习中的基础知识,需要的朋友可以参考下...

    C语言教程网8122021-03-09
  • C/C++浅谈c++中的输入输出方法

    浅谈c++中的输入输出方法

    下面小编就为大家带来一篇浅谈c++中的输入输出方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    C++教程网5952021-04-06
  • C/C++C++模板template用法小结(推荐)

    C++模板template用法小结(推荐)

    这篇文章主要介绍了C++模板template用法总结 ,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考...

    杨小平6642021-08-26
  • C/C++C语言自定义类型详解(结构体、枚举、联合体和位段)

    C语言自定义类型详解(结构体、枚举、联合体和位段)

    这篇文章主要给大家介绍了关于C语言中结构体、枚举、联合体和位段自定义类型的相关资料,分别介绍了结构体、枚举、联合体和位段等四种自定义类型,文...

    富春山居_ZYY6942021-12-21