C++基础入门教程(八):函数指针

 更新时间:2014年11月17日 10:15:22   投稿:junjie  
这篇文章主要介绍了C++基础入门教程(八):函数指针,本文讲解了函数原型和函数定义、const限定符与指针、函数的指针参数、为什么要使用指针参数等内容,需要的朋友可以参考下

最近事情比较多,其实并不忙,就是事情比较影响思绪,所以都没心思写文章了。
今天主要说说函数的一些基本情况吧,同时也解释一下新手最容易迷糊的——什么时候要用指针参数?

一、函数原型和函数定义

大家都知道,C++定义函数之前,还需要声明函数原型,对于习惯Java等其他高级语言的朋友来说,真心觉得这很烦人。
如下代码:

复制代码 代码如下:

// 声明函数原型
void startGame(int param);
// 函数定义
void startGame(int param)
{
    // 各种逻辑
}

函数原型主要是给编译器用的,在编译的时候会通过函数原型来检查函数返回值、参数数量、参数类型等。
总而言之,方便编译器,编译器爽了,我们才能更爽。

但实际中,函数原型也方便我们快速理解某个类的功能。

这些很简单,就不多唠叨了。

二、const限定符与指针

之前也有简单介绍过const,比如 const int num = 10; 那么num就是常量,不可再次进行赋值操作了。
如果把const用在指针上呢?

复制代码 代码如下:

    int num = 10;
    const int *p = #
 
    // 编译会报错
    *p = 100;

如上代码,编译的时候就会报错,因为指针p指向一个const int类型,这是一个常量。
所以*p的值是一个常量,不能被修改。

再来理一理,不然等会会混乱的:
1.p是一个指针
2.p指向一个内存地址,这个地址里存放的是一个const int类型的值
3.*p代表是p指向的内存地址里存放的那个值
4.所以,*p就是一个const int类型的值
5.综上所述,*p不能再次被赋值。

这里要区分p和*p,这是两个概念,一个是指针,一个是指针指向的值。
p是可以被再次赋值的,但是*p是不能被赋值的。

三、函数的指针参数

先来看看下面的代码:

复制代码 代码如下:

void notChangeNum(int num);
void changeNum(int* num);
int _tmain(int argc, _TCHAR* argv[])
{
    int num = 10;
  
    // 这个函数不会改变num的值
    notChangeNum(num);
    cout << num << endl;
    // 这个函数会改变num的值
    int* p = &num;
    changeNum(p);
    cout << num << endl;
    return 0;
}
void notChangeNum(int num)
{
    // 参数不是指针
    num = 999;
}
void changeNum(int* num)
{
    // 参数是指针
    *num = 999;
}

这里有两个函数,一个是普通参数(值传递),一个是指针参数(地址传递)。

第一个notChangeNum函数是不会改变num的值的,因为num传递给函数时,是拷贝了一份新的值,原来的num是不受影响的。

当离开notChangeNum函数后,函数的num参数会被释放。

第二个changeNum函数的参数是指针,我们都知道,指针是指向某个内存地址的,所以,函数的参数指向的内存地址就是num的内存地址。

直接修改内存地址上的值,会影响原来的num,所以,离开changeNum函数后,num的值也会被改变,最终值是999.

这就是指针参数的作用,某些情况下,我们希望函数里对参数的修改能够真正产生影响。

四、为什么要使用指针参数

为什么要用指针作为参数呢?因为指针可以直接指向内存地址,可以直接在函数里修改值,并且离开函数后仍然生效。
说是这么说,但,肯定还有人会迷糊,为什么呢?为什么要这样呢?

比如,我们的函数参数是某个类:

复制代码 代码如下:

void play(Sprite* sp) {
}

Sprite和Value都是Cocos2d-x常用的,这里的参数为什么是指针?
因为值引用的参数是会拷贝一份的,这样才不会影响原本的值,拷贝一份就会有额外的开销。
一般类的开销都比较大(相对于int、float等基本类型而言),所以拷贝一份不太合适。
而且我们通过都需要在函数里改变Sprite的坐标、大小等属性,如果使用值传递的话,就无法修改了(修改的只是拷贝的那一份)。

当然,这个还要看具体项目的情况,我不唠叨了,太深入不好吹水。

五、不想拷贝,又不想值被修改,怎么办?

拷贝开销大,使用指针参数又很可能在函数被修改了值,怎么办呢?
这时候就要用const限定符了,如下代码:

复制代码 代码如下:

void play(const Sprite* sp) {
}

这样在函数内部既不会修改sp指向的值,又可以避免值传递的额外开销。

六、函数内部的变量离开函数时就会被释放

我们之前说过,只要不是new出来的变量,那么,在离开作用范围后,就会被自动释放。
但是,来看看这个函数:

复制代码 代码如下:

int getNum() {
    int num = 10;
    return num;
}

既然变量离开作用范围后会被释放,那么,num在离开getNum函数后,就会被释放。
这时候return num的意义何在呢?getNum函数真的能成功获取到数字10吗?
答案是肯定的。

因为return 在返回num的时候,实际上是拷贝了一份的,返回的是拷贝的值,释放的是原来的变量。
这就是return的秘密了。

但是,指针就不行了,看看下面的代码:

复制代码 代码如下:

// 假设有这样一个结构体
struct People {
   int age;
};
People* getNewPeople() {
    People nPeople;
    nPeople.age = 20;
    return &nPeople;
}

这个函数返回的是一个指向People结构体类型的内存地址。
按照return的规则,返回的时候实际上是拷贝了一份,但这个时候拷贝的只是一个指针,也就是一个内存地址。
这个内存地址仍然指向函数内部的nPeople变量。

所以即使getNewPeople函数成功返回了一个指针,但这个指针指向的内存地址上的值仍然是被释放了。
也就是说,我们获取的只是一个野指针。

七、结束

好了,这篇写得有点糟糕,太多内容了,我只是抽取部分来吹吹水~

相关文章

  • C++ class和struct到底有什么区别详解

    C++ class和struct到底有什么区别详解

    这篇文章主要介绍了C++ class和struct到底有什么区别详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03
  • C++实现图片转base64的示例代码

    C++实现图片转base64的示例代码

    Base64就是一种 基于64个可打印字符来表示二进制数据的表示方法,本文主要为大家详细介绍了如何使用C++实现图片转base64,需要的可以参考下
    2024-04-04
  • C语言合并排序及实例代码

    C语言合并排序及实例代码

    本篇文章主要介绍C语言合并排序算法,这里对合并排序通过实例代码进行了详细讲解,希望能帮助到大家学习
    2016-07-07
  • C++ 中

    C++ 中"priority_queue" 优先级队列实例详解

    这篇文章主要介绍了C++ 中"priority_queue" 优先级队列实例详解的相关资料,需要的朋友可以参考下
    2017-04-04
  • C语言运用函数指针数组实现计算器功能

    C语言运用函数指针数组实现计算器功能

    这篇文章主要为大家详细介绍了C语言运用函数指针数组实现计算器功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-10-10
  • C++如何采用Daemon进行后台程序的部署

    C++如何采用Daemon进行后台程序的部署

    这篇文章主要介绍了C++采用Daemon进行后台程序的部署,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-04-04
  • C++简明分析临时对象是什么

    C++简明分析临时对象是什么

    对性能来说,许多的问题都需要和出现频率及本身执行一次的开销挂钩,有些问题虽然看似比较开销较大,但是很少会执行到,那也不会对程序有大的影响;同样一个很小开销的函数执行很频繁,同样会对程序的执行效率有很大影响。本章中作者主要根据临时对象来阐述这样一个观点
    2022-04-04
  • C++通过TerminateProess结束进程实例

    C++通过TerminateProess结束进程实例

    这篇文章主要介绍了C++通过TerminateProess结束进程实例,是Windows应用程序设计中非常实用的技巧,需要的朋友可以参考下
    2014-10-10
  • 详解用C语言实现三子棋游戏流程

    详解用C语言实现三子棋游戏流程

    三子棋是一种民间传统游戏,又叫九宫棋、圈圈叉叉、一条龙、井字棋等。将正方形对角线连起来,相对两边依次摆上三个双方棋子,只要将自己的三个棋子走成一条线,对方就算输了
    2021-11-11
  • C++聚合关系类的构造函数的调用顺序详解

    C++聚合关系类的构造函数的调用顺序详解

    下面小编就为大家带来一篇C++聚合关系类的构造函数的调用顺序详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考,一起跟随小编过来看看吧
    2016-05-05

最新评论