C/C++ ===复习==函数返回值问题(集合体==网络)

按值传递

地址传递:

应该明白只有这2种传递,下面讨论函数的按值传递

#include <stdio.h>
#include <stdlib.h>
int add_rtVal(int a,int b)
{
    int c = 0;
    c = a + b;
    return c;
}
int main(int argc,char* argv[])
{
    int a = 0,b = 0;
    int c = 0;
    a = 3;
    b = 5;
    c = add_rtVal(a,b);
    printf("c=%d/n",c);

    return 1;
}

这个地方是正确返回了,测试c也对,why?我们返回的是一个值 ,在返回值后我们将其赋给了变量C,相当于创建了一个副本,不论add()中的局部变量是否销毁都没有关系了。

但是如果我们返回的是一个地址,指向局部变量的地址,就不行了。因为局部变量的地址在函数调用完成后销毁,我们返回的地址就指向的区域内就不是我们想要的值了,因为这块区域已是公共的,可被其它函数占用的,其内容是不定的。看如下例子:

#include <stdio.h>
#include <stdlib.h>
char* hello()
{
    char p[]= "hello world";
    return p;
}
int main(int argc,char argv[])
{
    char *str;

    str = hello();
    printf("%s/n",str);
    return 1;
}

结果:"/n
Process returned 1 (0x1) execution time : 0.094 s
Press any key to continue.

====================================================================================

#include<iostream>
int a,i;
int fun()
{
  int b=1,c=2;
  a=b+c;
  return a;
}
i=fun()
首先,这里必须是传递a的副本其次,为什么传递副本,不是由a的生命周期来定义的,比如a是否是全局,或者局部,甚至是static,都与是否传递副本没有关系的最后,这里传递副本是取决于fun函数的,fun函数返回的类型是 int,也就是说返回值的传递时采用 ” 值传递 “,你应该懂这个名词,不是地址传递,而值传递,明显特征是要传递副本换句话说,如果定义fun函数如下:int& fun();那么可以return a,不会有副本产生。注意必须是a为全局的情况下,否则会引起警告,禁止传递局部变量的引用,因为a的声明周期会马上结束掉。

???为什么不写成int& fun();这种形式呢?答案是没必要。int是内部自定义类型。对于自定义变量而言,引用与否引起的副本拷贝带来的工作量不是很大,相反,如果是自定义类型,不知道楼主学习了class没有,那么引用返回就非常有必要了。因为带来的副本拷贝可能非常耗时,而这也是C++之父strup反复强调了,也是我们需要常常使用的
void   function(int   a) <-这里是传值调用,所以要建立一个参数的副本再传给函数 

这样一来a这个参数的原始数据就会保留,不会在函数中被改变: 

void   function(int   a)
{
          a++;
}
这里如果a=1;函数执行完毕后a还是等于1. 

void   function(int   &a) <-这里是模拟一个指针,当编译器编译这个函数的时候,直接到a的地址去操作a,这样避免了产生一个副本可能带来的额外开销.但是增加了危险性

void   function(int   &a)
{
          a++;
}
如果a的原始数据是1的话,那么a++后a的数据将变成2
不管a是全局变量还是局部变量
下面我们再看一种情况,这是返回引用给变量的情况:

#include <iostream>
#include <string>
using namespace std;  

float c;
float& test(float,float);
void main(int argc,char* argv[])
{
    float pn=test(3.0f,1.2f);
    cout<<pn;
    cin.get();
}  

float &test(float a,float b)
{
    c=a*b;
    return c;
}
  这种返回引用给变量的情况下,在内存中,test()所在的栈空间内并没有产生临时变量,而是直接将全局变量c的值给了变量pn,这种方式是我们最为推荐的操作方式,因为不产生临时变量直接赋值的方式可以节省内存空间提高效率,程序的可读性也是比较好的。
#include <iostream>
#include <string>
using namespace std;  

float c;
float& test(float,float);
void main(int argc,char* argv[])
{
    float &pn=test(3.0f,1.2f);
    cout<<pn<<endl;
    test(3.0f,1.2f)=12.1;//把函数作左值进行计算!
    cout<<pn;
    cin.get();
}  

float &test(float a,float b)
{
    c=a*b;
    return c;
}
  通常来说函数是不能作为左值,因为引用可以做为左值,所以返回引用的函数自然也就可以作为左值来计算了。

  在上面的代码中:

float &pn=test(3.0f,1.2f); 

  进行到这里的时候pn已经指向到了目标c的地址了。 

  接下来运行了 

test(3.0f,1.2f)=12.1; 

  把函数作左值进行计算,这里由于test是返回引用的函数,其实返回值返回的地址就是c的地址,自然c的值就被修改成了12.1。

C/C++ ===复习==函数返回值问题(集合体==网络),布布扣,bubuko.com

时间: 2024-10-15 13:32:30

C/C++ ===复习==函数返回值问题(集合体==网络)的相关文章

以函数返回值做参数时,函数调用的顺序

环境:vs2013 在下面的代码中 1 //类似于下面的代码 2 3 foo(char*,char*,char*); 4 5 char* str ="A#B#C"; 6 7 foo(strtok(str,"#"),strtok(NULL,"#"),strtok(NULL,"#")); 预计让函数foo得到("A","B","C")的参数,程序编译的时候没问题,但是运行

函数指针与指针函数返回值的区别

指针函数是指带指针的函数,即本质是一个函数.函数返回类型是某一类型的指针定义: 类型标识符 *函数名(参数表)eg: int *f(x,y);函数指针是指向函数的指针变量,即本质是一个指针变量.int (*f) (int x); /* 声明一个函数指针 */ f=func; /* 将func函数的首地址赋给指针f */ 函数指针与指针函数返回值的区别,码迷,mamicode.com

对象做函数参数和函数返回值时,调用复制构造函数,构造函数,析构函数的情况

// 对象做函数参数和返回值.cpp : 定义控制台应用程序的入口点.//exit(0)表示正常退出程序,exit(0)表示异常退出 //在调用input时,编译器用对象A去创建了形参对象temp,调用了复制构造函数,对象A中的数据复制给了对象temp// 在input函数中,执行temp.set(s),为对象temp中数据成员str申请了动态储存空间,并设置了输入的字符串//并没有改变实参A中的数据成员str的储存空间,故在执行语句A.show()后输出的字符串并没有改变.在函数调用结束后 /

shell调用函数返回值深入分析

编写shell脚本过程中,我们经常会自定义一些函数,并根据函数的返回值不同来执行相应的流程,那么我们如何来获取函数的返回值呢? 首先shell中调用函数有两种方式: 第一种:value=`function_name [arg1 arg2 ......]` 或 第二种:function_name [arg1 arg2 ......] echo $? 这两种有什么区别呢? 举个例子来说: [[email protected] ~]# cat test.sh #!/bin/sh function aa

const修饰函数参数 const修饰函数返回值 const修饰成员函数

看到const 关键字,C++程序员首先想到的可能是const 常量.这可不是良好的条件反射.如果只知道用const 定义常量,那么相当于把火药仅用于制作鞭炮.const 更大的魅力是它可以修饰函数的参数.返回值,甚至函数的定义体. const 是constant 的缩写,"恒定不变"的意思.被const 修饰的东西都受到强制保护,可以预防意外的变动,能提高程序的健壮性.所以很多C++程序设计书籍建议:"Use const whenever you need". 1

string 中的 length函数 和size函数 返回值问题

string 中的 length函数 和 size函数 的返回值  (  还有 char [ ] 中 测量字符串的  strlen 函数 ) 应该是 unsigned int 类型的 不可以 和 -1 比较. 应尽量避免 unsigned int 类型 和 int类型 数据 的比较 .当unsigned int 类型 和 int类型 数据 比较 时 ,会 把int 类型 转换 为 unsigned int类型 .如果 int是负数 ,转换 为 unsigned int 会是 一个 很大 的正整数

C语言的函数返回值

一:背景 谈到C语言的函数返回值,可能会感觉很亲切,不就是一个函数返回值嘛,当初学C语言的时候早就学过了很easy嘛,我曾经也是这么想的.后来要上研究生了,研究生阶段搞得就是C,所以又重新开始学习C,学习C的过程中遇到了很多问题,在此博客中一一记录.实际过程中遇到的第一个问题自然就是函数返回值了.如果有人问你在一个函数中声明一个字符串数组,最后再return这个数组.这可以实现嘛?如果是问我我可能会毫不犹豫的说OK.那事实呢?由此本文诞生了...... 二:问题 先看几个实际的例子: #incl

将引用作为函数返回值的优缺点

格式:类型标识符 &函数名(形参列表及类型说明){ //函数体 }好处:在内存中不产生被返回值的副本:(注意:正是因为这点原因,所以返回一个局部变量的引用是不可取的.因为随着该局部变量生存期的结束,相应的引用也会失效,产生runtime error!注意事项:(1)不能返回局部变量的引用.这条可以参照Effective C++[1]的Item 31.主要原因是局部变量会在函数返回后被销毁,因此被返回的引用就成为了"无所指"的引用,程序会进入未知状态.(2)不能返回函数内部new

linux编程中接收主函数返回值以及错误码提示

程序A创建子进程,并调用进程B,根据不调用的不同情况,最后显示结果不同. #include <stdio.h> #include <unistd.h> #include <sys/wait.h> #include <sys/types.h> #include <errno.h> int main() { pid_t pid, rpid; int stat; if ((pid = fork()) < 0) { perror("for