English 中文(简体)
C部分通过
原标题:Passing by reference in C

如果 C 不支持按引用传递变量,为什么这个可以工作?

#include <stdio.h>

void f(int *j) {
  (*j)++;
}

int main() {
  int i = 20;
  int *p = &i;
  f(p);
  printf("i = %d
", i);

  return 0;
}

Output:

$ gcc -std=c99 test.c
$ a.exe
i = 21 
最佳回答

因为您正在将指针的值传递给该方法,然后取消引用以获取指向的整数。

问题回答

那不是传引用,而是传值,正如其他人所述。

The C language is pass-by-value without exception. Passing a pointer as a parameter does not mean pass-by-reference.

规则如下:

一个函数无法改变实际参数的值。

(上述引文实际上来自K&书;R)


让我们尝试看一下函数的标量参数和指针参数之间的区别。

Scalar variables

这个简短的程序展示了使用标量变量的传值方式。 param 被称为形式参数,函数调用时的 variable 被称为实际参数。请注意,在函数中递增 param 不会改变 variable

#include <stdio.h>

void function(int param) {
    printf("I ve received value %d
", param);
    param++;
}

int main(void) {
    int variable = 111;

    function(variable);
    printf("variable %dm", variable);
    return 0;
}

结果是

I ve received value 111
variable=111

Illusion of pass-by-reference

我们稍微修改了代码片段。param现在是一个指针。

#include <stdio.h>

void function2(int *param) {
    printf("I ve received value %d
", *param);
    (*param)++;
}

int main(void) {
    int variable = 111;

    function2(&variable);
    printf("variable %d
", variable);
    return 0;
}

结果是

I ve received value 111
variable=112

这让你相信该参数是按引用传递的。实际上它并不是。它是按值传递的,参数值是一个地址。int类型的值被加1,这是让我们认为这是一次按引用传递的函数调用的副作用。

Pointers - passed-by-value

我们怎么展示或证明这个事实呢?好的,也许我们可以尝试使用标量变量的第一个示例,但是我们使用地址(指针)而不是标量。让我们看看这是否有所帮助。

#include <stdio.h>

void function2(int *param) {
    printf("address param is pointing to %d
", param);
    param = NULL;
}

int main(void) {
    int variable = 111;
    int *ptr = &variable;

    function2(ptr);
    printf("address ptr is pointing to %d
", ptr);
    return 0;
}

结果将是这两个地址相等(不要担心精确值)。

例子結果:

address param is pointing to -1846583468
address ptr   is pointing to -1846583468

在我看来,这清楚地证明了指针是按值传递的。否则,在函数调用之后,ptr将会是NULL

In C, Pass-by-reference is simulated by passing the address of a variable (a pointer) and dereferencing that address within the function to read or write the actual variable. This will be referred to as "C style pass-by-reference."

来源:www-cs-students.stanford.edu

Because there is no pass-by-reference in the above code. Using pointers (such as void func(int* p)) is pass-by-address. This is pass-by-reference in C++ (won t work in C):

void func(int& ref) {ref = 4;}

...
int a;
func(a);
// a is 4 now

你的例子有效是因为你将变量的地址传递给一个使用解引用运算符操作它的函数。

虽然C语言不支持引用数据类型,但您仍然可以通过显式传递指针值来模拟按引用传递,就像您的示例一样。

C++参考数据类型比从C继承的指针类型强度较小,但被认为更安全。这是您的示例,使用C++引用进行适应:

void f(int &j) {
  j++;
}

int main() {
  int i = 20;
  f(i);
  printf("i = %d
", i);

  return 0;
}

您通过值传递了一个指针(地址位置)。

这就像是告诉你:“这里是我希望你更新数据的地方。”

C中没有传递引用,但p“引用”i,而您通过值传递p。 (Traditional Chinese) C中沒有傳遞引用,但p“引用”i,而您通過值傳遞p。 (Simplified Chinese) C中没有传递引用,但p“引用”i,而您通过值传递p。

p 是一个指针变量。它的值是 i 的地址。当你调用 f 时,你传递了 p 的值,这是 i 的地址。

在 C 中,所有东西都是传值调用的。使用指针使我们产生了一种错觉,即我们正在通过传递,因为变量的发生了变化。然而,如果您打印指针变量的地址,您会发现它没有受到影响。函数将地址的的一个副本传递给它。以下是一段代码片段,用于说明这一点。

void add_number(int *a) {
    *a = *a + 2;
}

int main(int argc, char *argv[]) {
   int a = 2;

   printf("before pass by reference, a == %i
", a);
   add_number(&a);
   printf("after  pass by reference, a == %i
", a);

   printf("before pass by reference, a == %p
", &a);
   add_number(&a);
   printf("after  pass by reference, a == %p
", &a);

}

before pass by reference, a == 2
after  pass by reference, a == 4
before pass by reference, a == 0x7fff5cf417ec
after  pass by reference, a == 0x7fff5cf417ec

简短回答:是,C语言使用指针实现参数按引用传递。

在实现参数传递时,编程语言的设计者使用三种不同的策略(或语义模型):将数据传输到子程序中,从子程序中接收数据,或两者都进行。这些模型分别被称为输入模式、输出模式和输入输出模式。

许多语言设计师已经设计出了几种模型来实现这三种基本的参数传递策略:

Pass-by-Value (in mode semantics) Pass-by-Result (out mode semantics) Pass-by-Value-Result (inout mode semantics) Pass-by-Reference (inout mode semantics) Pass-by-Name (inout mode semantics)

Pass-by-reference is the second technique for inout-mode parameter passing. Instead of copying data back and forth between the main routine and the subprogram, the runtime system sends a direct access path to the data for the subprogram. In this strategy the subprogram has direct access to the data effectively sharing the data with the main routine. The main advantage with this technique is that its absolutely efficient in time and space because there is no need to duplicate space and there is no data copying operations.

Parameter passing implementation in C: C implements pass-by-value and also pass-by-reference (inout mode) semantics using pointers as parameters. The pointer is send to the subprogram and no actual data is copied at all. However, because a pointer is an access path to the data of the main routine, the subprogram may change the data in the main routine. C adopted this method from ALGOL68.

Parameter passing implementation in C++: C++ also implements pass-by-reference (inout mode) semantics using pointers and also using a special kind of pointer, called reference type. Reference type pointers are implicitly dereferenced inside the subprogram but their semantics are also pass-by-reference.

因此,这里的关键概念是传递引用实现了对数据的访问路径,而不是将数据复制到子程序中。数据访问路径可以是显式解引用指针或自动解引用指针(引用类型)。

如欲了解更多情况,请参阅Robert Sebesta编程语言概念书,第10版,第9章。

在C中,要通过引用传递,应该使用地址操作符&,该操作符应该用于变量,但在您的情况下,由于您使用了指针变量p,因此不需要在其前面加上地址操作符。如果您使用&i作为参数,则会是真的:f(&i)

你也可以添加这个操作符来取消引用 p 并查看该值如何与 i 匹配:

printf("p=%d 
",*p);

因为你正在将指向变量p的指针(内存地址)传递到函数f中。换句话说,你正在传递一个指针而不是一个引用。

你不是通过引用传递整数,而是通过值传递整数指针。不同的语法,相同的含义。

指针和引用是两个不同的东西。

我还没有看到提到的一些事情。

一个指针是某个东西的地址。指针可以像任何其他变量一样被存储和复制。因此它具有大小。

参考应当被视为《亚太贸易协定》。 它没有规模,不能储存。 司法部提到一些东西,即它不能无效或改变。 有时,汇编者需要把参考文件作为点,但这是执行细节。

有了引用,您就不必担心指针问题,如所有权处理,空值检查和使用时的解除引用。

What you are doing is pass by value not pass by reference. Because you are sending the value of a variable p to the function f (in main as f(p);)

使用传递引用方式的C程序将如下所示,(!!!此程序出现2个错误,因为C不支持传递引用)

#include <stdio.h>

void f(int &j) {    //j is reference variable to i same as int &j = i
  j++;
}

int main() {
  int i = 20;
  f(i);
  printf("i = %d
", i);

  return 0;
}

产出:- (chǎn chū)

3:12: error: expected  ; ,  ,  or  )  before  &  token
             void f(int &j);
                        ^
9:3:  warning: implicit declaration of function  f 
               f(a);
               ^

将指针称为引用(如Java和Javascript所做的那样)是与按引用传递不同的一种使用引用的方法。C不支持按引用传递。这是您的示例的重新编写,以显示它不是真正按引用传递值,只是按值传递指针。

#include <stdio.h>

void f(int *j) {
  int k = (*j) + 1;
  j = &k;
}

int main() {
  int i = 20;
  int *p = &i;
  f(p);
  printf("i = %d
", i);
  printf("j = %d
", *p);


  printf("i(ptr) = %p
", &i);
  printf("j(ptr) = %p
", p);


  return 0;
}

这是输出。

i = 20
j = 20
i(ptr) = 0x7ffdfddeee1c
j(ptr) = 0x7ffdfddeee1c

正如您所看到的,值保持不变,但更重要的是指针也不会改变。然而,C++允许通过引用传递参数。在这里,同样的例子通过C++编译器转换,但在标题中添加了一个&符号,这使其成为引用参数。

#include <stdio.h>

void f(int *&j) {   // note the & makes this a reference parameter.
                    // can t be done in C
  int k = (*j) + 1;
  j = &k;
}

int main() {
  int i = 20;
  int *p = &i;
  f(p);
  printf("i = %d
", i);
  printf("j = %d
", *p);


  printf("i(ptr) = %p
", &i);
  printf("j(ptr) = %p
", p);


  return 0;
}

这是输出。

i = 20
j = 21
i(ptr) = 0x7ffcb8fc13fc
j(ptr) = 0x7ffcb8fc13d4

请注意,我们能够更改实际的指针!

这个主题的一个绝佳参考文献是《编译原理》 (The Dragon Book) - 一本经典的计算机科学教科书。自从我上大学以来,它是最广泛使用的编译器教科书之一,我猜大部分设计语言或编写编译器的人也是从这本书中学习的。该书第1章非常清晰地解释了这些概念,并解释了为什么 C 只支持值传递。

那个代码片段(略有修改)

void add_number(int * const a) {
    *a = *a + 2;
}

也存在于C++中,并在语义上等同于

void add_number(int &a) {
    a = a + 2;
}

编译器应该在两种情况下生成函数add_number的等效二进制代码。现在,当您将整数视为值时,该值按其引用传递,在上面的模式中,引用在技术上出现为指针。

Conclusion
C supports the semantics of passing an instance by reference.
Even technically with int *a you pass *a, which is a reference.

传递引用(使用指针)从C语言的开始就存在了。您认为它不是吗?

我认为C实际上支持按引用传递。

大多数语言需要语法糖才能按引用而不是按值传递。(例如,C++需要在参数声明中使用&)。

为此,C语言需要语法糖。在参数类型声明中,需要使用*,而在参数上需要使用&。因此,*和&是C语言中的引用传递语法。

现在可以争论说,真正的按引用传递应该只需要在参数声明上使用语法,而不是在参数方面使用语法。

但现在出现了C#,它支持引用传递,而且在参数和参数两侧都需要语法糖。

C没有按引用传递的论点导致其语法元素无法表达其基础技术实现,这根本不是一个论点,因为这几乎适用于所有实现。

唯一剩下的争议是,在C中传递引用并不是一个整体的特性,而是结合了两个现有的特性。(通过&获取参数的引用,期望以*获取类型的引用。)例如,C#需要两个语法元素,但它们不能单独使用。

这显然是一个危险的论点,因为语言中许多其他功能由其他功能组成。(例如 C++ 中的字符串支持)





相关问题
Fastest method for running a binary search on a file in C?

For example, let s say I want to find a particular word or number in a file. The contents are in sorted order (obviously). Since I want to run a binary search on the file, it seems like a real waste ...

Print possible strings created from a Number

Given a 10 digit Telephone Number, we have to print all possible strings created from that. The mapping of the numbers is the one as exactly on a phone s keypad. i.e. for 1,0-> No Letter for 2->...

Tips for debugging a made-for-linux application on windows?

I m trying to find the source of a bug I have found in an open-source application. I have managed to get a build up and running on my Windows machine, but I m having trouble finding the spot in the ...

Trying to split by two delimiters and it doesn t work - C

I wrote below code to readin line by line from stdin ex. city=Boston;city=New York;city=Chicago and then split each line by ; delimiter and print each record. Then in yet another loop I try to ...

Good, free, easy-to-use C graphics libraries? [closed]

I was wondering if there were any good free graphics libraries for C that are easy to use? It s for plotting 2d and 3d graphs and then saving to a file. It s on a Linux system and there s no gnuplot ...

Encoding, decoding an integer to a char array

Please note that this is not homework and i did search before starting this new thread. I got Store an int in a char array? I was looking for an answer but didn t get any satisfactory answer in the ...