指针应用场景一:交换两个变量的值

  在学习函数时,交换两个数的值,做一个swap函数,传递值进去,也可以将两个值交换过来,没问题,可是离开swap就没有用了,为什么?因为传进去的是两个值。

 #include <stdio.h>
void swap(int a,int b); int main(void) {
int i=;
int j=;
swap(i,j);
printf("i=%d,j=%d\n",i,j);
return ;
} void swap(int a,int b){
int p;
p=a;
a=b;
b=p;
printf("a=%d,b=%d\n",a,b);
}

  现在学过指针后,就可以用指针来交换变量了,传两个地址进去,试一下:

 #include <stdio.h>
void swap(int *pa,int *pb); int main(void) {
int a=;
int b=;
swap(&a,&b);
printf("a=%d,b=%d\n",a,b);
return ;
} void swap(int *pa,int *pb){
int t;
t=*pa;
*pa=*pb;
*pb=t;
}

  swap函数的参数是两个指针,在swap内部使用的都是*运算符,在main里将a,b的地址传递过去,运行我们发现值交换成功了。

指针应用场景二:函数通过指针返回多个值

  在上面的例子中,swap函数在做交换后,返回来2个值,只返回a不够,还应该返回b。所以某些值就只能通过指针带回,也就是说传入的参数实际上是需要保存带回的结果的变量。再看一个例子:

 #include <stdio.h>
void minmax(int a[],int len,int *max,int *min); int main(void) {
int a[]={,,,,,,,,,,,,,,,,};
int min,max;
minmax(a,sizeof(a)/sizeof(a[]),&min,&max);
printf("min=%d,max=%d\n",min,max);
return ;
} void minmax(int a[],int len,int *min,int *max){
int i;
*min =*max=a[];
for(i=;i<len;i++){
if(a[i]<*min){
*min=a[i];
}
if(a[i]>*max){
*max=a[i];
}
}
}

  minmax函数有4个参数,第一个参数a[]是数组,在这个参数中找出最大和最小的数组元素,第二个len是表示数组的长度,最后两个是*max和*min,minmax函数需要算出两个结果,最小值和最大值。两个结果不可能通过函数的返回值来获得,返回值只能返回一个,那么需要通过两个指针来返回两个结果。

  在minmax函数中,首先使*min和*max等于a的第一个元素。然后使用for循环从第二个元素起遍历整个数组,如果发现某个元素比*min小了,就将该值赋值给*min。最后在main里调用minmax函数,使用sizeof函数得到数组的长度传递过去,然后将main里的int类型min和max取地址传递给minmax函数。查看下运行的结果,min是1,max是55.

  这就是指针应用非常常见的一个场景,需要函数处理的结果不止一个,通过指针把要接受结果的变量的地址传递过去,然后在函数内部把这些变量的值填好,同时传递回来。所以虽然*min和*max是传递进去的参数,但是他们的作用是把值带出来。

指针应用场景二b:函数返回运算的状态,结果通过指针返回

  函数在运算的过程中遇到一些特殊的状态,或者说在函数运算时没有得到正确的结果,常用的套路是什么那?是让函数返回特殊的不属于有效范围内的值来表示出错。在C语言的标准库里,和文件操作相关的函数里能看到大量的这样的例子,这些函数要么返回-1要么返回0来表示这个操作没有成功。如果当函数返回任何值都是有效的可能结果时,0或者-1也是属于有效的,就没有办法通过返回值来表示结果是否正确,这时需要分开返回:函数的状态通过返回值来返回,实际的值通过指针参数来返回。这样做的好处是,可以容易的将结果放在if语句里去,看下两个值做除法的例子:

 #include <stdio.h>
//return 如果除法成功,返回1;否则返回0
int divide(int a,int b,int *result); int main(void) {
int a=;
int b=;
int c;
if(divide(a,b,&c)){
printf("%d/%d=%d\n",a,b,c);
}
return ;
} int divide(int a,int b,int *result){
int ret=;
if(b==) ret = ;
else{
*result=a/b;
}
return ret;
}

  divide函数有三个参数,整数a和整数b,a除以b的结果放在第三个参数*result中,函数的返回值是如果除法是成功的,则返回值返回1,如果除数是0,让返回值为0,程序就终止不去做除法运算了产生异常返回0.在main里,如果返回值是1,条件满足打印输出abc的结果,如果返回值是0,条件不满足,不打印,因为如果除数b为0,c里是不能产生任何有意义的值。

  这种场景就是函数运算可能回出错,这种错误通过另外的场景表达出来,在c语言里只能这样。在后续的C++,java语言中,采用异常机制来解决这个问题。

指针最常见的错误:

  初学者在学习指针时最长犯的错误是,定义来指针变量,还没有指向任何变量,就开始使用指针。比如:

int *p;
int k;
k=;
*p=;

  首先定义*p,*p没有指向任何一个变量,这个时候定义k,为k赋值。然后*p不是相当于一个变量吗,也为*p赋值一个值。这是不可行的,为什么?因为在定义时,*p是空的,没有指向任何一个具体的变量,没有被赋予一个实际的值,这个时候给*p赋值12,这个时候会发生的什么情况那,首先所有的本地变量都不会有默认的初始值,如果没有赋值就不会是个明确的值,可能是个乱七八糟的值,如果把这个乱七八糟的值当成一个地址的时候,p可能会指向一个莫名其妙的地方,所以当*p=12时,则就是把这个莫名其妙的地方写入12,但这个莫名其妙的地方恰巧是不可写的,这个时候程序就会崩溃。

  试一下,如果int *p=0,让*p指向一个明确的不应该去的地方,这个时候程序可能就会出现一个段错误(和编译器有关),也可能编译可以过去,但是运行时结果出现问题,另外如果*p不赋值,同样可能编译能够过去,但是总有一天会出错,你不能保证*p里不指向一个不可写或者不存在的地方。这两种情况,在devc++环境下,都是编译可过,运行时出现错误:

听翁恺老师mooc笔记(4)--指针的应用场景的更多相关文章

  1. 听翁恺老师mooc笔记(6)--指针运算

    指针值加1就是将指针值加上sizeof(指针所指变量的类型) 1+1=2,那么指针加1是加上了1这个数字吗?试一下,在代码中定义了char数组,char也是整数,数组名是ac,ac中有10个元素,0- ...

  2. 听翁恺老师mooc笔记(5)--指针与数组

    如果我们通过函数的参数将一个数组传递到参数中去,那么在函数里接收到的是什么东西呢?我们知道如果传递一个普通变量,那么参数接收到的是值,如果传递一个指针变量,参数接收到的也是值,只不过这时的值是地址.那 ...

  3. 听翁恺老师mooc笔记(3)--指针的定义

    在上一个blog学习了&运算符,使用&取了变量.数组等地址,有什么用那?如果能够将取得的变量的地址传递给函数,能否通过这个地址在函数内访问到外部这个变量?答案是肯定的,scanf(&q ...

  4. 听翁恺老师mooc笔记(16)--程序设计与C语言

    问题1:计算机遍布生活的各个方面,若你需要一个功能可以下载APP,我们需要的大部分功能都可以找到对应的APP,如果没有可以自己写一个软件,但是很少人需要这么做,那么我们为什么学习计算机编程语言? 学习 ...

  5. 听翁恺老师mooc笔记(12)--结构中的结构

    结构数组: 和C语言中的int,double一样,一旦我们做出一个结构类型,就可以定义这个结构类型的变量,也可以定义这个结构类型的数组.比如下面这个例子: struct date dates[100] ...

  6. 听翁恺老师mooc笔记(11)--结构和函数

    结构作为函数参数: 声明了一个结构就有了一种自定义的数据类型,这个数据类型和int.float.double一样,int等基本类型可以作为函数的参数,那么这种个自定义的结构类型也应该可以作为函数参数, ...

  7. 听翁恺老师mooc笔记(10)--结构

    定义结构: 在程序里,如果想要表达一个数据就需要一个变量,而每个变量又都需要一个类型,之前学过C语言中有int.double.float.char等这些基础类型,还有指针.数组等.如果你要表达的数据比 ...

  8. 听翁恺老师mooc笔记(8)--字符串2

    字符串的赋值 字符串的输入与输出 对C语言的基础类型,比如int.double等类型,scanf.printf有专门的格式转换,而对字符串,scanf.printf使用%s格式字符进行输入与输出.当使 ...

  9. 听翁恺老师mooc笔记(7)--字符串1

    C语言中字符串的定义 如果定义一个字符数组word,并使用大括号对其初始化,如下图所示: 但是这个不是C语言的字符串,只是字符数组,不是字符串,因为不能使用字符串的方式进行计算.那么C语言的字符串长什 ...

随机推荐

  1. Java 第一章 初识Java

    第一章笔记 什么是计算机程序:算机为完成某些功能生产的一系列有序指令集合 Java技术包括: java SE:标准版 java EE:企业版 Java ME:移动版 开发Java程序步骤:1.编写 2 ...

  2. MFC关于多线程中传递窗口类指针时ASSERT_VALID出错的另类解决 转

    MFC关于多线程中传递窗口类指针时ASSERT_VALID出错的另类解决   在多线程设计中,许多人为了省事,会将对话框类或其它类的指针传给工作线程,而在工作线程中调用该类的成员函数或成员变量等等. ...

  3. C#图解教程 第二十二章 异常

    异常 什么是异常try语句 处理异常 异常类catch 子句使用特定catch子句的示例catch子句段finally块为异常寻找处理程序更进一步搜索 一般法则搜索调用栈的示例 抛出异常不带异常对象的 ...

  4. 【转载】Spark运行架构

    1. Spark运行架构 1.1 术语定义 lApplication:Spark Application的概念和Hadoop MapReduce中的类似,指的是用户编写的Spark应用程序,包含了一个 ...

  5. jquery中的事件与应用

    mouseover和mouseenter的区别 mouseenter的定义是当鼠标穿过该元素才会被执行,而mouseover是当鼠标穿过该元素或者其子元素皆会触发该事件 mouseleave一般是与m ...

  6. 【BZOJ1305】跳舞(网络流)

    [BZOJ1305]跳舞(网络流) 题面 Description 一次舞会有n个男孩和n个女孩.每首曲子开始时,所有男孩和女孩恰好配成n对跳交谊舞.每个男孩都不会和同一个女孩跳两首(或更多)舞曲.有一 ...

  7. [BZOJ1046] [HAOI2007] 上升序列 (dp)

    Description 对于一个给定的S={a1,a2,a3,…,an},若有P={ax1,ax2,ax3,…,axm},满足(x1 < x2 < … < xm)且( ax1 < ...

  8. python3下Django2.0配置最新xadmin详解

    1,打开pycharm,创建一个Django项目 2,安装Django,默认是最新版本,pip3 install -i https://pypi.douban.com/simple/ django 3 ...

  9. 3.2.2 break 与 continue 语句

    break 语句和 continue语句在while循环和for循环中都可以使用,并且一般常与选择结构结合使用.一旦break语句被执行,将使得break语句所属层次的循环提前结束.continue语 ...

  10. PHP/JAVA 杂谈 一(php 槽点)

    [本文为个人意见,不喜就喷吧!] 最近,同事问到我,『那时候为什么从PHP转成Java?』,我想了很久,且撇开主观上的原因,当初业务重构使用java确实有很多可以说道的地方. 槽点1:哪有最好的语言, ...