概述

初学 \(c++\),一直搞不懂其参数传递方式。故找到一篇不错的文章:刘志华的深入探讨C++语言中参数传递问题。亲自实践一遍,并作此记录,以加深印象。

 


 

主要内容

本文主要分为五个小部分,下面依次作总结。

 

简单变量作参数

简单变量作参数的传递方式是按值传递,在这种传递方式下,在函数体中对形参的修改不会影响实际参数的值,因为它们使用各自的存储空间。

下面便是一个简单变量作参数的实例。调用 swap 参数同时把 \(x\) 的值传送给形参 \(a\),把 \(y\) 的值传送给形参 \(b\),在函数体中对形参 \(a\) 和 \(b\) 的操作是与对应的实参 \(x\) 与 \(y\) 无关的,因为它们使用各自的存储空间。

void swap(int a, int b)
{
int temp = a;
a = b;
b = temp;
}
int main()
{
int x=45, y = 12;
swap(x, y)
cout<<"x="<<x<<' '<<"y="<<y<<endl;
return 0;
}

输出结果是:\(x=45,\, y=12\). 可见在这种传递方式下,在函数体中对形参的修改不会影响实际参数的值。

 

指针作参数

指针本身是一个变量,特殊之处在于它的值是一个地址,因而可以通过指针来间接访问另外一个内存地址。当函数的形式是指针时,它的实际参数的值必须是一个地址。由于指针作形参,因此对应的形参的实参共用同一个内存单元,形参值的改变将影响实参值。下面继续看一个实例。

void swap(int* a, int* b)
{
int temp = *a;
*a = *b;
*b = temp;
}
int main()
{
int x = 45;
int y = 12;
swap(&x, &y); cout<<"x="<<x<<' '<<"y="<<y<<endl; return 0;
}

输出结果是:\(x=12,\, y=45\). 可见在这种传递方式下,在函数体中对形参的修改将影响实际参数的值。

 

引用作参数

由于引用作形参,因此对应的形参和实参就是同一个内存单元,形参值的改变也将影响实参值。

void swap(int& a, int& b)
{
int temp = a;
a = b;
b = temp;
}
int main()
{
int x = 45;
int y = 12;
swap(x, y); cout<<"x="<<x<<' '<<"y="<<y<<endl; return 0;
}

输出结果是:\(x=12,\, y=45\). 可见在这种传递方式下,在函数体中对形参的修改将影响实际参数的值。

引用传送的好处是不需要为形参分配新的存储空间,从而节省存储,另外就是如实例这样,能够使对形参的操作反映到实参上。有时,既为了使形参共享实参的存储空间,又不希望通过形参改变实参的值,则应当把该形参说明为常量引用,如:

void swap(const int& a, const int& b)

在这种情况下,只能读取引用参数 \(a\) 和 \(b\) 的值,不能够修改它们的值,因为它们是对应实参的别名,从而杜绝了对实参的有意或无意的破坏。

 

数组作参数

看下面一个实例。

int sum(int a[], int n)
{
int f = 1;
for (int i=0; i<n; i++)
{
f *=a[i];
}
return f;
}
int main()
{
const int N = 6;
int b[N] = {1,2,3,4,5,6}; int x = sum(b, 6);
int y = sum(&b[2], 4);
int z = sum(b+3, 3); cout<<"x="<<x<<' '<<"y="<<y<<' '<<"z="<<z<<endl; return 0;
}

输出结果是:\(x=720,\, y=360,\,z=120\).

该函数包含一个主函数和一个 sum 函数,其功能是求出一维整型数组所有元素之积并返回。

  • 在主函数第一次调用 sum 函数时,把数组 \(b\) 的首地址传送给 \(a\),把数组 \(b\) 的长度 \(6\) 传送给 \(n\). 执行函数体对数组 \(a\) 的操作实际上就是对主函数中数组 \(b\) 的操作,因为它们同时指向数组 \(b\) 的存储空间;
  • 第二次调用 sum 函数是把数组 \(b\) 中 \(b[2]\) 元素的地址传送给 \(a\), 把整数 \(4\) 传送给 \(n\). 执行函数体对数组 \(a[n]\) 的操作实际上是对数组 \(b[2]\) 至 \(b[5]\) 之间元素的操作;
  • 第三次调用 sum 函数是把数组 \(b\) 中 \(b[3]\) 元素的地址传送给 \(a\), 把整数 \(3\) 传送给 \(n\), 执行函数体对数组 \(a[n]\) 的操作实际上是对数组 \(b[3]\) 至 \(b[5]\) 之间元素的操作。

 

字符串作参数

看下面一个实例。

char* sss(char *sp, char* dp)
{
if (*sp == '\0')
{
*dp = '\0';
return dp;
}
int i=0; // i 是记录 dp 中字符个数
int j;
for (char* p=sp; *p; p++)
{
//扫描 sp 所指字符串中的每个字符位置
for (j=0; j<i; j++)
{
if (*p == dp[j])
break; // 当 *p 与 dp[0] 至 dp[i-1] 之间的任一元素相同则比较过程结束
}
if (j >= i)
dp[i++] = *p; // 若 dp 数组的前 i 个元素不等于 *p, 则把 *p 写入 dp[i] 元素中
}
dp[i] = '\0'; // 写入字符串结束符
return dp;
}
int main()
{
char a[15] = "abcdeabcde";
char b[15]; char* c1 = sss(a, b);
cout<<"c1="<<c1<<' '<<"a="<<a<<' '<<"b="<<b<<endl; char* c2 = sss(a+4, b);
cout<<"c2="<<c2<<' '<<"a="<<a<<' '<<"b="<<b<<endl; return 0;
}

sss 函数的功能是把 sp 所指向的字符串去掉重复字符后拷贝到 dp 所指向的字符数组中,并返回 dp 指针。在主函数中第一次调用 sss 函数时,分别以 \(a\) 和 \(b\) 作为实参,第二次调用时分别以 \(a+4\) (即 \(a[4]\) 的地址)和 \(b\) 作为实参。该程序运行结果是:

\(c1=abcde\,\,a=abcdeabcde\,\, b=abcde\)

 

\(c2=eabcd\,\,a=abcdeabcde\,\, b=eabcd\)

c++ 中的函数调用中的参数传递的更多相关文章

  1. C++函数调用时的参数传递-3中传递方式

    参数传递即实参向形参传递信息,使形参获得对应的存储空间及初值,C++中函数传递主要有3种方式: 1.按值传递. 以按值传递方式进行参数传递的过程为:首先计算出实参的值,然后给它所对应的形参变量分配存储 ...

  2. Jni中C++和Java的参数传递 参数对照

    Jni中C++和Java的参数传递 如何使用JNI的一些基本方法和过程在网上多如牛毛,如果你对Jni不甚了解,不知道Jni是做什么的,如何建立一个基本的jni程序,或许可以参考下面下面这些文章:利用V ...

  3. C++中构造函数调用构造函数

    今天想做道矩阵的题目时,却卡在一些编程细节上了,找了好久才发现原来是在构造函数处出了问题,然后上网百度了下,发现这篇文章说得很好:从一道题谈C++中构造函数调用构造函数,很棒! 还补充一点: 看来自己 ...

  4. ASP.NET中实现页面间的参数传递

    ASP.NET中实现页面间的参数传递   编写人:CC阿爸 2013-10-27 l  近来在做泛微OA与公司自行开发的系统集成登录的问题.在研究泛微页面间传递参为参数,综合得了解了一下现行页面间传参 ...

  5. Jni中C++和Java的参数传递

    Jni中C++和Java的参数传递 如何使用JNI的一些基本方法和过程在网上多如牛毛,如果你对Jni不甚了解,不知道Jni是做什么的,如何建立一个基本的jni程序,或许可以参考下面下面这些文章:利用V ...

  6. SSIS中执行SQL任务组件参数传递的问题

    原文:SSIS中执行SQL任务组件参数传递的问题 症状: 执行SQL任务,传递参数到子查询中,执行报错. 错误: 失败,错误如下:"无法从使用 sub-select 查询的 SQL 语句中派 ...

  7. 理解js中的函数调用和this

    概述 这是我看typescript的时候看引用资源看到的,原文在这里:Understanding JavaScript Function Invocation and "this" ...

  8. JS循环中使用bind函数的参数传递问题

    JS循环中使用bind函数的参数传递问题,问题代码如下: for (var sc in result) { var tempp = '<div class="sidebar_todo_ ...

  9. 在chrome开发者工具中观察函数调用栈、作用域链与闭包

    在chrome开发者工具中观察函数调用栈.作用域链与闭包 在chrome的开发者工具中,通过断点调试,我们能够非常方便的一步一步的观察JavaScript的执行过程,直观感知函数调用栈,作用域链,变量 ...

随机推荐

  1. Unity3D游戏开发之Unity与Android交互调用研究

    各位朋友,大家好,我是秦元培,欢迎大家关注我的博客,我的博客地址是blog.csdn.net/qinyuanpei.在前一篇文章中,我们研究了Android平台上Unity3D的手势操作并在之前的基础 ...

  2. bzoj 3778: 共鸣【计算几何+dp】

    枚举起点,然后设f[i][j]为上凸壳上一个点是i当前点是j的最大面积,g是下凸壳,然后合并的时候枚举结束点t合并上下凸壳即可 这样的好处是每次转移都是往凸多边形里加一个三角形(s,i,j),所以判断 ...

  3. 从MySQL高可用引出对高可用架构设计的一些思考

    高可用HA(High Availability)是分布式系统架构设计中必须考虑的因素之一,它通常是指,通过设计减少系统不能提供服务的时间. 假设系统一直能够提供服务,我们说系统的可用性是100%.如果 ...

  4. 单片机的C语言中位操作用法

    在对单处机进行编程的过程中,对位的操作是经常遇到的.C51对位的操控能力是非常强大的.从这一点上,就可以看出C不光具有高级语言的灵活性,又有低级语言贴近硬件的特点.这也是在各个领域中都可以看到C的重要 ...

  5. Java反射学习笔记01

  6. python_argparse

    使用python argparser处理命令行参数 #coding:utf-8 # 导入模块 import argparse # 创建ArgumentParser()对象 parser = argpa ...

  7. 反射和__setattr__,__delattr__,__getattr__用法

    反射 什么是反射:反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问.检测和修改它本身状态或行为的一种能力(自省).这一概念的提出很快引发了计算机科学领域关于应用反射性的研究.它首先 ...

  8. #13:人十我一Orz——6

    水题放送,写得依旧丑: #include <cstdio> #include <cstring> #include <cmath> #include <alg ...

  9. AtCoder Regular Contest 078 D

    D - Fennec VS. Snuke Time limit : 2sec / Memory limit : 256MB Score : 400 points Problem Statement F ...

  10. 1-24List三个子类的特点

    List的三个子类的特点 因为三个类都实现了List接口,所以里面的方法都差不多,那这三个类都有什么特点呢? ArrayList: 底层数据结构是数组,查询快,增删慢. 线程不安全,效率高. Vect ...