C/C++ 高质量编程--动态内存传递疑问解析 1(一)

2014-11-24 08:55:45 · 作者: · 浏览: 0

高质量编程 :中指针高级部分遇到动态内存传递疑问解析


总结:

值传递:对于函数来说,值传递就是一个人来了,给你一些数据,你对数据处理。

地址传递:地址传递就是你通过地址找到一个人,然后直接对这个人处理。

引用传递:而引用传递就是你要直接对一个人进行处理,通过名字对人访问,但你不知道那个人叫什么,就先起个随便的名字。谁来了,他的别名就是这个名字。但通过这个别名对人处理有一些限制,相比地址对人全权操作安全。

另一段别人的解释:值传递、地址传递、引用传递的区别:

值传递好比是你把文件复制一份,通过网络传给他,然后他可以在他本机上对文件做任何的修改,修改会保存下来,但是你机器上的文件不会发生任何的变化。即形参与实参是两个不同的变量,各自占用不同的存储单元。

地址传递好比是你把文件在网络上的地址告诉他人,他人通过网络访问你机器上的文件,他可以对文件进行修改并保存,此时,文件的内容就会发生变化。即形参与实参是相同的变量,占用同一段内存空间。

引用传递是变量的别名,对别名操作就是对变量本身操作。

地址传递是通过把地址传递给函数,然后函数根据地址要对存储单元操作。打个比方说:你告诉某人一个房间号,让他根据房间号去拿去东西。
交换二个数的理解:
值传递没有办法交换两个数。
用地址传递可以实现。
打个比方说: 在两个房间门口有两个服务人员,房间里有一本书和字典。现在要求交换这两件东西。服务员就是地址,如果找到服务员,然后让她把东西拿出来和别一个房间交换。这样达到了交换的目的。地址传递能够实现交换的目的。
另一种方法:在两个服务人员中还有一个人是管理人员,如果管理人员让这两个服务员换一下位置,这样以后按服务员找各自房间的东西,也起到了交换的目的。这是通过指针的指针来实现的。
以上二种情况是效率最高提第二种了。
第一种是交换房间的东西,人没有换。
第二种是交换人,东西没有换。
从以可以感性的知道,用指针操作会更快。 
例子:

void GetMemory(char *p, int num)
{
p = (char *)malloc(sizeof(char) * num);
}
void Test(void)
{
char *str = NULL;
GetMemory(str, 100); // str 仍然为 NULL
strcpy(str, "hello"); // 运行错误
}


这种情况下: p=str; 是值传递。
void GetMemory2(char **p, int num)
{
*p = (char *)malloc(sizeof(char) * num) ;
}

void Test2(void)
{
char *str = NULL;
GetMemory2(&str, 100); // 注意参数是 &str,而不是str
strcpy(str, "hello");
cout<< str << endl;
free(str);
}


这种情况下:  *p=&str; 是地址传递。
可以这样理解,理管人员(&str), 然后告诉服务员(*p)去找一个房间,这时管理人员和服务一起去,(相当于共占一个存储空间),服务员找到了房间,管理员也就知道了。
char *GetMemory3(int num)
{
char *p = (char *)malloc(sizeof(char) * num);
return p;
}
void Test3(void)
{
char *str = NULL;
str = GetMemory3(100);
strcpy(str, "hello");
cout<< str << endl;
free(str);
}


这种情况可以这样理解:服务员(p)找到了一个房间((char *)malloc(sizeof(char) * num);),然后告诉(RETURN p;)管理员(str).

一. 三道考题
开讲之前,我先请你做三道题目

1. 考题一:程序代码如下:

void Exchg1(int x, int y)  
{
  int tmp;
  tmp=x;
  x=y;
  y=tmp;
  printf(“x=%d,y=%d\n”,x,y)
}
void main()
{
  int a=4,b=6;
  Exchg1 (a,b) ;
  printf(“a=%d,b=%d\n”,a,b)
}


输出的结果:
x=____, y=____
a=____, b=____
问下划线的部分应是什么,请完成。

2. 考题二:代码如下。

Exchg2(int *px, int *py)
{
  int tmp=*px;
  *px=*py;
   *py=tmp;
  print(“*px=%d,*py=%d\n”,*px,*py);
}
main()
{
  int a=4;
  int b=6;
      Exchg2(&a,&b);
      Print(“a=%d,b=%d\n”, a, b);
}


输出的结果为:
*px=____, *py=____
a=____, b=____
问下划线的部分应是什么,请完成。

3. 考题三:

Exchg2(int &x, int &y)
{
   int tmp=x;
   x=y;
   y=tmp;
  print(“x=%d,y=%d\n”,x,y);
}
main()
{
  int a=4;
  int b=6;
      Exchg2(a,b);
      Print(“a=%d,b=%d\n”, a, b);
}


输出的结果:
x=____, y=____
a=____, b=____
问下划线的部分输出的应是什么,请完成。

你不在机子上试,能作出来吗?你对你写出的答案有多大的把握?
正确的答案,想知道吗?(呵呵,让我慢慢地告诉你吧!)
好,废话少说,继续我们的探索之旅了。
我们都知道:C语言中函数参数的传递有:值传递,地址传递,引用传递这三种形式。题一为值传递,题二为地址传递,题三为引用传递。不过,正是这几种参数传递的形式,曾把我给搞得晕头转向。我相信也有很多人与我有同感吧?
下面请让我逐个地谈谈这三种传递形式。

二. 函数参数传递方式之一:值传递

1. 值传递的一个错误认识
先看题一中Exchg1函数的定义:

void Exchg1(int x, int y)   //定义中的x,y变量被称为Exchg1函数的形式参数
{
  int tmp;
  tmp=x;
  x=y;
  y=tmp;
  printf(“x=%d,y=%d\n”,x,y)
}


问:你认为这个函数是在做什么呀?
答:好像是对参数x,y的值对调吧?
请往下看,我想利用这个函数来完成对a,b两个变量值的对调,程序如下:
void main()
{
  int a=4,b=6;
  Exchg1 (a,b)     //a,b变量为Exchg1函数的实际参数。
/  printf(“a=%d,b=%d\n”,a,b)
}


我问:Exchg1 ()里头的 printf(“x=%d,y=%d\n”,x,y)语句会输出什么啊?
我再问:Exchg1 ()后的 printf(“a=%d,b=%d\n”,a,b)语句输出的是什么?
程序输出的结果是:
x=6 , y=4
a=4 , b=6 //为什么不是a=6,b=4呢?

奇怪,明明我把a,b分别代入了x,y中,并在函数里完成了两个变量值的交换,为什么a,b变量值还是没有交换(仍然是a==4,b==6,而不是a==6,b==4)?如果你也会有这个疑问,那是因为你跟本就不知实参a,b与形参x,y的关系了。

2. 一个预备的常识
为了说明这个问题,我先给出一个代码:

int a=4