重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
我来详细解释一下: void test1(int a)
创新互联专业为企业提供谢家集网站建设、谢家集做网站、谢家集网站设计、谢家集网站制作等企业网站建设、网页设计与制作、谢家集企业网站模板建站服务,10年谢家集做网站经验,不只是建网站,更提供有价值的思路和整体网络服务。
{
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);
} 附图:
函数参数有传值和传址两种,你只要把数组的首地址传过去就可以了,函数参数是个指针,接收数组首地址,就可以在子函数中用指针调用数组中的数据。
子函数与母函数,你这个说法头一次听说,呵呵。
函数间只有调用与被调用关系,没有亲属关系。除了MAIN函数外,其他所有函数间都是平等关系。
一个函数在定义时,就会声明两大项内容:一、函数返回值类型;二、输入、输出参数及类型,如:
int func1( int x , char *str ); //返回值为INT类型,参数有两个,一个整型,一个字符指针。
调用者在调用此函数时,就需要传参数。
void fun2( void ); //无返回值,无参数
调用者在调用此函数时,就不需要传参数。
在写程序时,我们使用一个函数前,一定要先查询到这个函数的具体定义(一般系统提供的函数在相应的头文件中都有说明),这样我们才能正确的使用C函数。
p函数就是个冒泡递减排序功能。标准函数头写法应该是void p(int a[])而不是void p(int a[5]),即使写成void p(int a[5]),编译器也是按void p(int a[])解释而不按void p(int a[5])解释,而且这里a[]的意思就是定义a是int *型指针。函数p中并没有改变a,改变的是a指向的目标a[0]~a[4]。所以不知你问的具体是啥意思……