C语言 传值 传址 传引用
我们知道C语言向子函数传递参数有3种形式(准确的讲引用是C++风格的,并不能算在C中)
传值的话很简单,就是子函数构造形参,将实参的值赋给形参,即子函数中的操作并不会影响实参.
传址,将实参地址传递给形参,使实参形参指向同一块内存
传引用,真正的以地址方式传递参数
现在的问题是
传址这种形式中,若我在子函数中修改了形参地址,实参的地址是不变的吗?
那在传引用这种形式中,情况又是如何?
如果说我的推论是正确的话,那是不是说传值和传址本质是一样的,只是传递的内容不同?
追问:那也就是说,传址实际上是把地址当值传给子函数了?也就是传值的一种特殊情况?
答案:我来详细解释一下:
void test1(int a)
{
a = 100; //这里的a是根据参数a而构造出的一个副本,暂存于栈中,与实际参数a完全处于不同内存。
//当函数退出时,a自动无效,修改a的值只是修改栈中这个a的值,无法修改实参。
//如果使用的是C++传递的是一个类的话,同样需要构造一个副本,函数退出时也会析构
//该副本,因此会加大构造和析构的开销,同时会加大内存开销用于保存临时副本
}
void test2(int* a)
{
*a = 1000; //这里的a和传值一样,也是构造的一个副本,不过这个副本是一个指针,同样暂存域栈中
//同样需要加大创建这个指针副本的内存,32位机器为4字节,即存放这个副本用的
//同样与实参a指针处于完全不同的内存
//但是,虽然这两个指针处于不同内存,他们指向的地址却是同一个地址
//我们通过*操作符来解析指针指向的地址的值(简称解引用)
//*a = 5;这句代码的意思是将副本a指向的地址的值改变为5
//由于副本a指向的地址与实参指向的地址相同,即改变了实参指向的地址的值
//因此实参a指向的地址的值变成了5
//可以解释的形象一些,
//假如实参a自身处于内存0x01,它指向地址0x03
//那么这个副本处于的内存不会是0x01(因为它是构造出来的一个副本指针),我们假设它自身地址
//为0x02(只是假设,这个地址值肯定是在栈当中的),而它指向的地址一样是0x03
}
void test3(int* a)
{
int tmp;
a = &tmp; //接着test2讲,由于这里的a是一个构造的副本,如果我们改变它,让它指向tmp的地址
//就不会改变实参指向的地址的值,这里只是将副本指向的地址改变了而已
//解释的形象一些,
//同理假设实参a自身内存为0x01,它指向地址0x03
//这里的a的自身内存为0x02,它也指向地址0x03
//tmp地址为0x04
//a = &tmp;这句代码的意思是让a指向tmp的地址,即副本a指向了tmp地址0x04
//而实参a指向的地址的值没有发生任何变化
*a = 10000; //这里对形参a的任何操作已经于实参a毫无任何关系了
//因为它自身地址是0x02,它指向的地址是0x04
}
void test4(int& a)
{
a = 100000; //这里的a与实参a完全一样了,在这里对a的任何操作等价于对实参a的操作
}
void main()
{
int a = 1;
int b = 2;
int c = 3;
int d = 4;
printf("操作前各值\na:%d\tb:%d\tc:%d\td:%d\n", a, b, c, d);
test1(a);
test2(&b);
test3(&c);
test4(d);
printf("操作后各值\na:%d\tb:%d\tc:%d\td:%d\n", a, b, c, d);
}
附图:
传址这种形式中,若你在子函数中修改了形参地址,实参的地址是不变的
你说的对的,就是传值的话只是作用于函数内,传地址才能真正修改数值
呵呵 传址是将实参的地址传递 给了形参 也就是说形参里保存的是实参的地址 而形参本身也分配了内存空间 也会有自己的地址 所以你改变形参的地址 是不会改变实参的地址的
而传值则是将实参的值传递给了形参 在形参中有一个和实参相同的参数 而改变形参的值 是不会改变实参的值
传引用则是将实参从新定义了一个别名 对形参的的改变等于就是对实参的改变 不知道我的回答你能理解不 呵呵'
希望对你有所启发
上一个:怎样用C语言绘制直线
下一个:c语言快速入门?高手进!