c语言中形参与实参的关系

一、形参出现在函数定义中,在整个函数体内都可以使用,形参变量只有在被调用时才分配内存单元,在调用结束时即刻释放分配的内存单元因此,形参只有在函数内部有效。 函数调用结束返回主调函数后则不能再使用该形参变量。  离开该函数则不能使用,实参出现在主调函数中进入被调函数后实参变量也不能使用。 

二、形参和实参的功能是作数据传送。发生函数调用时, 主调函数把实参的值传送给被调函数的形参从而实现主调函数向被调函数的数据传送。

三、实参可以是常量、变量、表达式、函数等, 无论实参是何种类型的量,在进行函数调用时,它们都必须具有确定的值, 以便把这些值传送给形参。因此应预先用赋值,输入等办法使实参获得确定值。

四、实参和形参在数量上,类型上,顺序上应严格一致, 否则会发生“类型不匹配”的错误。 

五、实参和形参的数据传递方式有两种,

  1. 一种是值传递,实参和形参都不是指针,这种情况下函数调用中发生的数据传送是单向的。 即只能把实参的值传送给形参,而不能把形参的值反向地传送给实参。 在函数调用过程中,形参的值发生改变,而实参中的值不会变化。究其原因,是因为这两个参数在内存中位于不同的位置,值传递只是形参将实参的内容复制,在内存中被分配了新的内存单元,在函数执行完毕以后地址会立刻被释放掉,因此形参的改变不会对实参有任何影响。
  2. 另一种是地址传递,就是实参与形参共用同一个内存单元,在函数执行的过程中,实际就是对实参的地址进行操作,因此形参改变,实参同步变化,实际他们就是同一变量,因为在内存中占据的就是一个内存单元。
  3. 不过下面这个例子很狡猾,就是指针作为形参时,实参*s只是把地址传递给了形参*p,在函数体内,p指针指向了下一个内存地址,因此原内存地址中的内容并未改变,p指针变量的如何变化与S无关,除非p指针改变了实参所指向位置的内存单元中的数值,那么访问实参所得值也会随之变化。
void func(char *p)
{
    p=p+1;
}
int main()
{
    char s[]={'1','2','3','4'};
    func(s);
    printf("%c",*s);
    return 0;
}

输出结果为1

并不是2

<think>嗯,用户想了解C语言形参实参的区别。首先,我需要回忆一下这两个概念的基本定义。形参应该是函数定义时声明的数,而实参是调用函数时实际传入的值。对吧?然后得想怎么解释清楚它们的区别,最好用例子说明。 我记得之前看过的引用里提到,在函数定义处的数是形参,调用处的是实参。比如sum函数定义中的mn是形参,调用时传的ab是实参。用户可能需要知道它们在内存中的处理方式,比如形参是临时变量,实参的值会被复制给形参。 还有,形参实参的作用域不同,形参只在函数内部有效。这可能导致一些误解,比如在函数内修改形参不会影响实参的值,除非使用指针。这时候应该举一个例子,比如交换两个数的函数,如果不传指针,实参不会被改变,这样用户更容易理解。 另外,用户可能对为什么需要区分形参实参感到困惑。需要说明这样设计的目的是为了实现模块化编程,让函数更独立,减少副作用。不过,如果用户没有正确理解传值传址的区别,可能会有bug出现。 可能还需要提到形参实参的类型必须匹配,否则编译器会报错或者警告。比如,如果实参是float,而形参是int,可能会有类型转换的问题。不过C语言有时候会自动转换,但最好还是显式匹配类型。 用户的问题可能涉及到函数调用的机制,比如值传递引用传递。在C语言中,默认是值传递,所以形参实参的副本。如果要修改实参,必须通过指针来传递地址。这一点很重要,可能需要重点强调。 总结一下,我需要分点解释两者的定义、区别、作用域、生命周期,以及传值传址的不同效果。用例子代码片段来辅助说明,比如交换函数的例子,这样用户能更直观地理解。同时,结合引用中的例子,比如sum函数,说明形参实参如何传递数据。 另外,用户可能不知道如何正确使用形参实参,或者常见错误有哪些,比如在函数内修改形参期望影响实参却失败的情况。这时候需要指出问题所在,并给出正确的做法,比如使用指针。 最后,确保回答结构清晰,先定义再区别,然后例子注意事项,最后总结。这样用户能一步步理解,不会有信息过载的问题。同时,生成的三个相关问题要覆盖常见疑问,比如修改形参是否影响实参形参的作用域,以及如何正确传递数。</think>### C语言形参实参的区别 #### 1. 基本定义 - **形参数)**:函数定义时声明的数,用于接收调用时传入的值。例如: ```c int sum(int m, int n) { // m、n为形参 return m + n; } ``` - **实参(实际数)**:函数调用时传递给函数的具体值或变量。例如: ```c int a = 5, b = 3; sum(a, b); // a、b为实参 ``` #### 2. 核心区别 1. **存在阶段不同** - 形参在函数定义时声明,函数调用结束后销毁[^1]。 - 实参在函数调用前已存在,调用时将其值传递给形参[^2]。 2. **内存分配** - 形参是函数的局部变量,仅在函数内部有效,调用时为它分配内存[^1]。 - 实参可以是常量、变量或表达式,其内存独立于形参。 3. **数据传递方向** - 实参形参是单向值传递(默认行为)。例如: ```c void modify(int x) { x = 10; } int main() { int y = 5; modify(y); // y的值仍为5 } ``` #### 3. 关键特性对比 | 特性 | 形参 | 实参 | |------------|-----------------------|-----------------------| | 作用域 | 函数内有效 | 原作用域有效 | | 生命周期 | 函数调用期间存在 | 原作用域内存在 | | 修改影响 | 不影响实参(默认情况)| 直接反映原始数据变化 | #### 4. 通过指针修改实参 若需通过形参修改实参,需传递指针: ```c void swap(int *x, int *y) { // 形参为指针 int temp = *x; *x = *y; *y = temp; } int main() { int a = 1, b = 2; swap(&a, &b); // 传递地址作为实参 // 此时a=2, b=1 } ``` #### 5. 注意事项 - 形参实参的**类型需匹配**,否则可能导致隐式类型转换或编译错误。 - 数组作为数时,形参会退化为指针(例如 `int arr[]` 等价于 `int *arr`)。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值