C语言基础篇5:指针(一)

        指针是C语言的核心、精髓所在,用好了指针可以在C语言编程中起到事半功倍的效果。指针一方面可以提高程序的编译效率和执行速度,而且还可以通过指针实现动态的存储分配,另一方面使用指针可使程序更灵活,便于表示各种数据结构,编写高质量的程序。同时,其抽象概念,学习过程中要多看多练,使用时应多注意,否则操作不当会导致整个程序收到破坏。

1 指针相关概念

        指针是C语言最显著的优点之一,指针使用起来十分灵活而且能提高某些程序的效率,但是指针使用不当,会很容易造成系统错误,往往许多程序“挂死”的大部分原因都是错误地使用指针所造成的。

1.1 地址与指针

        系统的内存就像带有编号的房间,如果想使用内存就需要得到房间编号。例如,定义一个整型变量i,整型变量需要4个字节,所以编译器为变量i分配编号从1000~1003。

        什么是地址?地址就是内存区中对每个字节的编号,例如上面的1000~1003就是地址。

        什么是指针?这里仅把指针看作是内存的一个地址,多数情况下,这个地址是内存中另一个变量的位置,如下图:

        上图中定义了一个变量,在进行编译时就会给这个变量在内存中分配一个地址,通过访问这个地址就可以找到所需的变量,该变量的地址称为该变量的指针。上图中的地址1000就是变量i的指针。

【说明】在C语言中,存取变量值的方法有两种。按变量地址存取变量的方式称为“直接访问”;将变量地址存放在另一个变量中,先找到存放“变量地址”的另一个变量,通过另一个变量找到变量的地址,这种方法称为“间接访问”。

1.2 指针变量

1.2.1 变量与指针

        变量的地址是变量和指针两者之间连接的纽带,如果一个变量包含了另一个变量的地址,那么,第一个变量可以说成是指向第二个变量。所谓“指向”就是通过地址来体现的,在程序中用“*”表示指向。因为指针变量是指向一个变量的地址,所以将一个变量的地址值赋给这个指针变量后,这个指针变量就“指向”了该变量。例如,将变量i地址存放到指针变量p中,p就指向i。在程序代码中是通过变量名来对内存单元进行存取操作的,但是代码经过编译后已经将变量名转换成该变量在内存的存放地址,对变量值的存取都是通过地址进行的。例如,对变量i和j进行如下操作:

    int i,j;i + j;

        其含义根据变量名与地址的对应关系,找到变量i的地址1000,然后从1000开始读取4个字节数据放到CPU寄存器中,在找到变量j的地址1004,从1004开始读取4个字节的数据放到CPU的另一个寄存器中,通过CPU计算出结果。     

1.2.2 使用指针变量

        由于通过地址能访问指定的内存存储单元,可以说是地址“指向”该内存单元。地址可以称之为指针,意思是通过指针就能找到内存单元。一个变量的地址称为该变量的指针。如果有一个变量专门用来存放另一个变量的地址,它就是指针变量。在C语言中有专门用来存放内存单元地址的变量类型,就是指针类型。

        指针变量的一般形式:类型说明 * 变量名。其中,*表示这是一个指针变量,变量名即为定义的指针变量名,类型说明本指针变量所指向的变量的数据类型。

        指针变量的赋值,指针变量同普通变量一样,使用之前不仅要定义,而且必须赋予具体的值。未经赋值的指针变量不能使用。给指针变量所赋的值与给其他变量所赋的值不同,给指针变量的赋值只能赋予地址,而不能赋予任何其他数据,否则将引起错误。C语言提供了地址运算符“&”来表示变量的地址。一般形式为:& 变量名。例如,&a表示变量a的地址,&b表示变量b的地址。给一个指针变量赋值可以有两种方法。

        1、定义指针变量的同时进行赋值:int a;int *p = &a;

        2、先定义指针变量,之后在赋值:int a;int *p ;p = &a;

        【注意】两者的区别,如果在定义完指针变量之后再赋值,注意不要加*。

        【示例1】利用指针输出数据。

        

#include <stdio.h>
int main(){int a,b;//声明两个指针变量int *ip1,*ip2;printf("请输入苹果和香蕉的价格 \n");scanf("%d,%d",&a,&b);ip1 = &a;ip2 = &b;printf("苹果的价格是: %d/一斤 \n",*ip1);printf("香蕉的价格是:%d/一斤 \n",*ip2);return 0;
}

    

        指针变量的引用

        引用指针变量是对变量进行间接访问的一种形式。对指针变量的引用形式如下:

        * 指针变量。其含义是引用指针变量所指向的值。

        【示例2】利用指针变量实现数据的输入、输出。

#include <stdio.h>
int main(){int *p,q;printf("请输入香蕉的价格: \n");scanf("%d",&q);p = &q;printf("香蕉的价格是:%d 元一斤\n",*p);return 0;
}

 

1.3 & 和  * 运算符

        上面介绍指针变量的过程中用到了两个运算符& 和 * ,& 是一个返回操作数地址的单目运算符,叫做取地址运算符,例如:p = &a。就是把变量a的内存地址赋给p,这个地址是该变量在计算机内部存储的地址。

        * 是单目运算符,叫做指针运算符,作用是返回指定地址内变量的值,如上面提到的p中装有变量a的内存地址,则 q = *p。就是把变量a的值赋给q,假如a = 5,那么q = 5.

        指针运算符和取地址运算符可以组合使用,那么&*和*&有什么区别呢?有如下语句:

int a;
p = &a;

        通过以上两个语句来发分析 &* 和 *& 之间的区别,&和*的运算符优先级别相同,按自右而左的方向结合。因此&*先计算*运算,*p相当于变量a;再进行&运算,&*p就相当于取变量a的地址。*&a先计算&运算,&a就是取变量a的地址,再进行*计算,*&a相当于取变量a所在地址的值,实际就是a变量。

【示例1】 &* 应用。

#include <stdio.h>int main(int argc, const char * argv[]) {long i;long *p;printf("请输入一个数值:\n");scanf("%ld",&i);p = &i;printf("输出&*p结果是:%ld\n",&*p);printf("输出&i结果是:%ld\n",&i);return 0;
}

   

 【示例2】*& 应用

#include <stdio.h>int main(int argc, const char * argv[]) {long i, *p;printf("请输入一个数值:\n");scanf("%ld",&i);p = &i;printf("输出*&i的结果是:%ld\n",*&i);printf("输出i的结果是:%ld\n",i);printf("输出*p的结果是:%ld\n",*p);return 0;
}

        其中,示例1:&*p和&i的作用相同,都是取变量i的地址 。

                示例2:*&p和*p相同,取变量的值。

1.4 指针的算术运算

        指针有加减两种运算,即指针的自加和自减,这不同于普通变量的自加自减运算,并不是简单的加1或减1。

【示例1】整型变量地址输出

#include <stdio.h>int main(int argc, const char * argv[]) {int i;int *p;printf("请输入一个数值:\n");scanf("%d",&i);p = &i;printf("p的结果是:%d\n",p);p++;printf("p的结果是:%d\n",p);return 0;
}

        把上面代码的变量修改为short变量,再次执行。

#include <stdio.h>int main(int argc, const char * argv[]) {short i;short *p;printf("请输入一个数值:\n");scanf("%d",&i);p = &i;printf("p的结果是:%d\n",p);p++;printf("p的结果是:%d\n",p);return 0;
}

 

        从两个示例可以看出,这里的指针自加不是简单的在地址上加1,而是指向下一个存放基本整型数的地址,第一个示例的变量是基本整型,所以p++后,p的值增加4(基本整型占4个字节);第二个示例的变量定义为短整型,所以p++后p的值加2(短整型占2个字节)。

        【结论】 指针都是按照它所指向的数据类型的直接长度进行加减的。

【范例1】转向的指针:通过交换两个指针变量的值取改变指针的方向。

#include <stdio.h>int main(void) {int *p1,*p2,a,b,*t;printf("请输入a,b的值\n");scanf("%d,%d",&a,&b);p1 = &a;p2 = &b;if(*p1 < *p2){t = p1;p1 = p2;p2 = t;}printf("%d > %d\n",*p1,*p2);return 0;
}

2 一维数组与指针

        数组在内存中存放也同样具有地址。 对于数组来说,数据名就是数组在内存中存储的首地址。指针变量用于存放变量的地址,自然也可以存放数组的首地址或数组元素的地址,这样就给数组和指针之间建立了一个联系。

2.1 指向数组元素的指针

        当定义了一个一维数组时,系统会在内存中为该数组分配一个存储空间。其数组的名字就是数组在内存的首地址。如果在定义一个指针变量,并将数组的首地址传给指针变量,则该指针就指向了这个一维数组。如:int *p,a[10]; p = a;

        这里a是数组名,也就是数组的首地址,将它赋给指针变量p,也就是将数组a的首地址赋给p,也可以写成:int *p,a[10]; p = &a[0]。

【注意】在将数组名赋给指针变量时不需要写“&”,但是将数组首地址赋给指针变量时,需要加上“&”。

【示例2.1.1】输出数组的元素。

#include <stdio.h>int main(void) {int *p,*q,a[5],b[5],i;p = &a[0];q = b;printf("请输入数组a中的元素:\n");for(i = 0;i<5;i++){scanf("%d",&a[i]);}printf("请输入数组b中的元素:\n");for(i = 0;i<5;i++){scanf("%d",&b[i]);}printf("数组a的元素是:\n");for(i = 0;i<5;i++){printf("%5d",*(p+i));}printf("\n");printf("数组b的元素是“\n");for( i =0;i<5;i++){printf("%5d",*(q+i));}printf("\n");
}

2.2 使用指针访问数组

        对于一维数组的引用有两种方法:一种是下标法,另一种是指针法。下标法是采用a[i]的形式引用数组中的元素,而指针法时本篇主要介绍的,首先看下面两条语句:int *p,a[10]; p = &a;

        上面的语句将作以下几方面介绍:

        1、p + n 与 a + n表示数组元素a[n]的地址,即&a[n]。对整个a数组来说,共有10个元素,n的取值是0~9,则数组元素的地址就可以表示为p+0~p+9或者a+0~a+9,如下图 

        2、如何来表示数组中的元素用到了前面介绍的数组元素的地址,用*(p+n)和*(a+n)来表示数组中的各个元素。例如下面的语句:printf("%10d",*(p+i)) ;printf("%10d",*(q+i));分别表示输出数组a和数组b中对应的元素。

        上面提到可以用a+n表示数组元素的地址,*(a+n)表示数组元素,那么就可以把上面的程序进行改造:

#include <stdio.h>int main(void) {int *p,*q,a[5],b[5],i;p = &a[0];q = b;printf("请输入数组a中的元素:\n");for(i = 0;i<5;i++){scanf("%d",&a[i]);}printf("请输入数组b中的元素:\n");for(i = 0;i<5;i++){scanf("%d",&b[i]);}printf("数组a的元素是:\n");for(i = 0;i<5;i++){printf("%5d",*(a+i));}printf("\n");printf("数组b的元素是“\n");for( i =0;i<5;i++){printf("%5d",*(b+i));}printf("\n");
}

        3、表示指针的移动可以使用”++“和”--“运算符,利用”++“运算符可以把上面的程序再次改造

#include <stdio.h>int main(void) {int *p,*q,a[5],b[5],i;p = &a[0];q = b;printf("请输入数组a中的元素:\n");for(i = 0;i<5;i++){scanf("%d",p++);}printf("请输入数组b中的元素:\n");for(i = 0;i<5;i++){scanf("%d",q++);}p = a;q = b;printf("数组a的元素是:\n");for(i = 0;i<5;i++){printf("%5d",*p++);}printf("\n");printf("数组b的元素是“\n");for( i =0;i<5;i++){printf("%5d",*q++);}printf("\n");
}

【示例2.1】查找数列中的最值。输入10个整型数字,自动查找这些数中的最大值和最小值。

#include <stdio.h>
void max_min(int a[],int n,int *max,int *min){int *p;*max = *min = *a;for(p = a + 1;p < a + n; p ++){if(*p > *max){*max = *p;}else if(*p< *min){*min = *p;}}
}int main(void) {int i,a[10];int max,min;printf("请输入10个整数:\n");for(i = 0;i<10;i++){scanf("%d",&a[i]);}max_min(a,10,&max,&min);printf("最大值是:%d\n",max);printf("最小值是:%d\n",min);getchar();}

 

3 字符串与指针

        字符串常量是由双引号组成的字符序列,表示字符串可以用字符数组,即通过数组名来表示字符串,数组名就是数组的首地址,是字符串的起始地址。现在将字符串数组的名赋予一个指向字符类型的指针变量,让字符类型指针指向字符串在内存的首地址,对字符串的表示就可以用指针实现。

3.1 字符型指针

        字符型指针就是指向字符型内存空间的指针变量,一般形式为:char *p ,使用字符型指针可以访问字符串。

【示例3.1】字符型指针

#include <stdio.h>int main(int argc, const char * argv[]) {char *string = "生当作人杰,死亦为鬼雄\n";printf("%s\n",string);return 0;
}

【示例3.1.1】声明两个字符数组,将str1中的字符串复制到str2中

#include <stdio.h>int main(int argc, const char * argv[]) {char str1[]="生当作人杰,死亦为鬼雄",str2[15],*p1,*p2;p1 = str1;p2 = str2;while(*p1 != '\0'){*p2 = *p1;p1 ++;p2 ++;}*p2 = '\0';printf("第二个字符串的内容是:\n");puts(str2);return 0;
}

 

3.2 字符串数组

        这里提到的字符串数组与前面提到的字符数组不同,字符数组是一个一维数组,而字符串数组是以字符串作为数组元素的数组,可以将其看成一个二维字符数组。例如下面一个简单的字符串数组定义:

char country[5][20] ={"中国","美国","俄罗斯","英国","法国"}

        字符型数组变量country被定义为含有5个字符串的数组,每个字符串的长度要小于20(要考虑字符串最后的'\0')。通过观察上面定义的字符串数组会发现,元素的长度远远小于其定义的20个字节的空间,这样会造成空间浪费。为了解决这个问题,可以使用指针数组,每个指针指向所需要的字符常量,这种方法虽然需要再数组中保存字符指针,同样也占用空间,但要远少于字符串数组所需要的空间。 

        那么什么是指针数组呢?一个数组,其元素均为指针类型数据,成为指针数组。也就是说,指针数组中的每一个元素都相当于一个指针变量。一维指针数组的定义形式:

类型名   数组名[数组长度]

        【例3.2.1】输出一周7天

#include <stdio.h>int main() {int i;char *month[] = {"星期一","星期二","星期三","星期四","星期五","星期六","星期日"};for(i = 0;i<7;i++){printf("%s \n",month[i]);}return 0;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/173477.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LeetCode Hot100 236.二叉树的最近公共祖先

题目&#xff1a; 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为&#xff1a;“对于有根树 T 的两个节点 p、q&#xff0c;最近公共祖先表示为一个节点 x&#xff0c;满足 x 是 p、q 的祖先且 x 的深度尽可能大&#xff08;一个节…

HT97220与HT97230耳机放大器芯片对比

HT97230有两个不同开启时间(tON)版本&#xff0c;版本A、C和E的导通时间tON为5.5ms&#xff0c;用于耳机驱动&#xff1b;B和D则具有130ms的tON&#xff0c;用于机顶盒设计&#xff08;目前仅提供A版本&#xff0c;其他版本需预定&#xff09;。内部电荷泵对输入电源反相&#…

accelerate的使用说明

1 多卡(GPU)使用方法 终端输入指令&#xff0c;生成问答页面 accelerate config 这个方法也是可以的 2 后面修改直接找到这个yaml文件进行修改即可 cd ~/.cache/huggingface/accelerate vim default_config.yaml 进入vim进行修改 3 单卡(GPU)使用方法 vim default_config.…

边缘计算网关:智能制造的“智慧大脑”

一、智能制造的崛起 随着科技的飞速发展&#xff0c;智能制造已经成为了制造业的新趋势。智能制造不仅能够提高生产效率&#xff0c;降低生产成本&#xff0c;还能够实现个性化定制&#xff0c;满足消费者多样化的需求。然而&#xff0c;智能制造的实现离不开大量的数据处理和分…

2023.11.25电商项目平台建设2 -四大业务之核销主题建模

1.数仓建模步骤 自下而上 ADS-DWS-DWM-DWD 2.DWD方案(清洗转换,降维拉宽) DWD层的表 dwd_sale_store_sale_dtl_i 门店销售明细宽表 维度dim 销售sale合成成的宽表 dwd_dim_date_f 日期表 store_sale_dtl 门店销售明细表 dwd_sale_store_sale_dtl_i 门店销售明细表 …

快速上手Banana Pi BPI-R4 MediaTek MT7988A 开源路由器开发板

基础开发 准备开发 * 准备8G以上TF卡、USB转串口线、Ubuntu系统* 使用 USB 串行电缆&#xff08;3.3V TTL&#xff0c;波特115200&#xff09;连接到 BPI-R4 上的调试控制台G接地&#xff1b;RXBPI-R4输入&#xff1b;TXBPI-R4输出* BPI-R4 引导程序和设备选择跳线设置* 例子…

【leetcode】62. 不同路径

题目 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。 问总共有多少条不同的路径&#xff1f; …

基于SpringBoot的超市信息管理系

✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取项目下载方式&#x1f345; 一、项目背景介绍&#xff1a; 随着我国经济的不断发…

【Linux】第二十站:模拟实现shell

文章目录 一、shell的实现细节1.shell的一些细节2.用户名、主机名、工作目录2.输入命令3.改为循环4.切割字符串5.普通命令的执行6.内建命令的处理7.子进程的退出码8.总结 二、模式实现shell完整代码 一、shell的实现细节 1.shell的一些细节 shell操作系统的一个外壳程序。 s…

笔记:pycharm当有多个plt.show()时候,只显示第一个plt.show()

import matplotlib.pyplot as plt import numpy as np# 创建数据 x np.linspace(0, 10, 100) y1 np.sin(x) y2 np.cos(x) y3 np.tan(x) y4 np.exp(x)# 创建一个2x2的子图网格 # fig plt.figure() fig,((ax1, ax2), (ax3, ax4)) plt.subplots(nrows2, ncols2, figsize(8,…

【matlab程序】matlab画台风符号和实例应用

【matlab程序】matlab画台风符号和实例应用 没有看文献&#xff0c;不知道文献中的符号什么样子&#xff0c;据我理解为这样子的&#xff1a; 因此&#xff0c;按照自己的理解做了这期。 结果浏览&#xff1a; 台风符号一切可改&#xff0c;可细细改。可是我不发论文&#xf…

Oracle 中的操作符

1.union:对两个结果集进行并集操作&#xff0c;不包括重复行&#xff0c;同时进行默认规则的排序&#xff1b; SELECT * FROM emp WHERE sal < 1500 UNION SELECT * FROM emp WHERE sal BETWEEN 1000 AND 2000 order by 1 2.union All&#xff1a;对两个结果集进行并集操…

[C/C++]数据结构 堆排序(详细图解)

一:前言 在[C/C]数据结构 堆的详解中,介绍了什么是堆,并且完成了堆的实现和一系列接口,包括向上调整法和向下调整法等,接下来小编介绍一个有点量级的排序方法------堆排序,时间复杂度为O(n*lgn) 二:堆排序详解 2.1 方法介绍 1.首先将待排序数组建为大堆,此时堆顶元素就为数组…

给国外客户价格报低了怎么办

前一段时间有一个单子的货发出去了&#xff0c;被朋友提醒才发现自己报错了价格&#xff0c;造成了亏损&#xff0c;而报错价格的原因并不是自己看错了或者是抄错了价格&#xff0c;而是自己的脑子里记错了产品的价格列表。 如果不是朋友善意的提醒&#xff0c;大概我会一直错…

【心得】XXE漏洞利用个人笔记

XML中关于DTD类型(内部(SYSTEM)的和外部(PUBLIC)的区别) xxe的利用 XML Entity 实体注入 当程序处理xml文件时&#xff0c;没有禁止对外部实体的处理&#xff0c;容易造成xxe漏洞 危害 主流是任意文件读取 XML 文件 一般表示带有结构的数据 祖父 3个叔父 8个堂弟堂妹 …

python-opencv 人脸68点特征点检测

python-opencv 人脸68点特征点检测 不是很难&#xff0c;主要还是掉包&#xff0c;来看一下代码啊&#xff1a; # coding: utf-8 # 导包 import numpy as np import dlib import cv2class face_emotion(object):def __init__(self):# 人脸检测器对象&#xff0c;通过它拿到人…

Rust UI开发(三):iced如何打开图片(对话框)并在窗口显示图片?

注&#xff1a;此文适合于对rust有一些了解的朋友 iced是一个跨平台的GUI库&#xff0c;用于为rust语言程序构建UI界面。 这是一个系列博文&#xff0c;本文是第三篇&#xff0c;前两篇的链接&#xff1a; 1、Rust UI开发&#xff08;一&#xff09;&#xff1a;使用iced构建…

【LeetCode:1457. 二叉树中的伪回文路径 | 二叉树 + DFS +回文数】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

使用 OpenCV 发现圆角矩形的轮廓

OpenCV - 如何找到圆角矩形的矩形轮廓? 问题: 在图像中,我试图找到矩形对象的圆角轮廓。然而,我对两者的尝试 HoughLinesP 并 findContours 没有产生预期的结果。 我的目标是找到一个类似于以下形状的矩形: 。 代码: import cv2 import matplotlib.pyplot as plt…

能让PDF看起来像是扫描件的Look Scanned

什么是 Look Scanned ? Look Scanned 是一个能够让 PDF 看起来就像是扫描件一样的纯前端网站。你再也不需要麻烦地打印之后扫描了&#xff0c;你所需要的就是鼠标点几下。 这是个挺有意思的软件&#xff0c;但是老苏不确定什么场景下会用到这个软件&#xff0c;如果不想自己搭…