最近用QT做一个监控系统的项目,需要显示目标的运动轨迹,每次目标移动后,就在目标的轨迹中(用vector记录)添加一条新轨迹。

但是在运行中画面里一直不出现轨迹,经过调试发现是记录轨迹的函数出错了。

原程序的逻辑是:目标轨迹类是ObjectTrack,通过调用getTrack()函数获得一个vector,然后将新的坐标append到vector里。

问题出现的原因在于,getTrack()方法返回的vector不是ObjectTrack对象的成员,因为在return的时候发生了vector的复制,所以对复制后的vector添加新坐标是不会被保存的。

解决办法1:

对返回后的vector进行添加新坐标后再set回ObjectTrack对象中

这个方法会产生额外的vector副本

解决办法2:

使用引用,将getTrack()函数的返回值设置成引用,并用引用变量接收

不会产生vector的副本,比解决办法1好

解决办法3:

在ObjectTrack中添加新的函数用于追加新坐标信息

符合封装性,使用也更简单,比1、2好

对于C++对象赋值操作(还有return对象)发生的复制现象,做了以下分析:

1.当函数返回值不是引用类型是,会发生复制,且接收函数返回值的变量不能是引用类型

#include <iostream>
using namespace std; class C1
{
public:
int a;
C1(int a){
this->a = a;
}
/*复制构造函数,不修改被拷贝的对象,所以参数为const;参数是引用,
是因为如果不是引用,在传递参数的时候就会被调用复制构造函数,而这
个函数本身就是复制构造函数,会造成无穷的递归调用
*/
C1(const C1 &c){
this->a = c.a;
cout << "copy constructor called" << endl;
}
};
class C2
{
public:
C1 c1;
C2():c1(10){
}
C1 getC1(){
return c1;
}
};
int main(){
C2 c2;
C1 c1 = c2.getC1();
cout<<(int)(&c1)<<endl<<(int)(&c2.c1)<<endl;
return 0;
}

  运行结果是:

copy constructor called
2686712
2686716

2.当函数返回值是引用类型,接收函数返回值的变量不是引用时,仍然发生复制

#include <iostream>
using namespace std; class C1
{
public:
int a;
C1(int a){
this->a = a;
}
/*复制构造函数,不修改被拷贝的对象,所以参数为const;参数是引用,
是因为如果不是引用,在传递参数的时候就会被调用复制构造函数,而这
个函数本身就是复制构造函数,会造成无穷的递归调用
*/
C1(const C1 &c){
this->a = c.a;
cout << "copy constructor called" << endl;
}
};
class C2
{
public:
C1 c1;
C2():c1(10){
}
C1 &getC1(){
return c1;
}
};
int main(){
C2 c2;
C1 c1 = c2.getC1();
cout<<(int)(&c1)<<endl<<(int)(&c2.c1)<<endl;
return 0;
}

  运行结果是:

copy constructor called
2686712
2686716

3.函数返回值是引用类型,接收函数返回值的变量是引用类型,不会发生复制

#include <iostream>
using namespace std; class C1
{
public:
int a;
C1(int a){
this->a = a;
}
/*复制构造函数,不修改被拷贝的对象,所以参数为const;参数是引用,
是因为如果不是引用,在传递参数的时候就会被调用复制构造函数,而这
个函数本身就是复制构造函数,会造成无穷的递归调用
*/
C1(const C1 &c){
this->a = c.a;
cout << "copy constructor called" << endl;
}
};
class C2
{
public:
C1 c1;
C2():c1(10){
}
C1 &getC1(){
return c1;
}
};
int main(){
C2 c2;
C1 &c1 = c2.getC1();
cout<<(int)(&c1)<<endl<<(int)(&c2.c1)<<endl;
return 0;
}

运行结果是:

2686712

2686712

另外,数组返回值也会发生复制

String str = strs[i];

str与strs[i]也是不同的对象

C++函数返回值发生的对象复制的更多相关文章

  1. C++函数返回值为类对象但未调用复制构造函数

    参考资料:https://blog.csdn.net/sxhelijian/article/details/50977946 不要迷信书本,要学会自己调试程序.

  2. Python3_函数参数传递、可变与不可变对象、变量作用域、函数返回值

    参数传递: 在 python 中,类型属于对象,变量是没有类型的:(变量都仅仅是一个对象的引用,没有类型之分)a=[1,2,3] a="Runoob" 以上代码中,[1,2,3] ...

  3. 获得函数返回值类型、参数tuple、成员函数指针中的对象类型

    //function_traits.h,获得函数返回值类型.参数tuple.成员函数指针中的对象类型 //参考https://github.com/qicosmos/cosmos/blob/maste ...

  4. C++ const修饰函数、函数参数、函数返回值

    const修饰函数 在类中将成员函数修饰为const表明在该函数体内,不能修改对象的数据成员而且不能调用非const函数.为什么不能调用非const函数?因为非const函数可能修改数据成员,cons ...

  5. 速战速决 (3) - PHP: 函数基础, 函数参数, 函数返回值, 可变函数, 匿名函数, 闭包函数, 回调函数

    [源码下载] 速战速决 (3) - PHP: 函数基础, 函数参数, 函数返回值, 可变函数, 匿名函数, 闭包函数, 回调函数 作者:webabcd 介绍速战速决 之 PHP 函数基础 函数参数 函 ...

  6. 关于QT中evaluateJavaScript()函数返回值的处理问题

    关于QT中evaluateJavaScript()函数返回值的处理问题 - 寒风问雪的专栏 - 博客频道 - CSDN.NET 关于QT中evaluateJavaScript()函数返回值的处理问题 ...

  7. C++函数返回值(02)

    对象作为返回值 编译器会将函数栈中的返回值数据拷贝到返回栈中 指针作为返回值 函数的返回值可以是存储某种类型数据的内存地址,称这种函数为指针函数.它们的一般定义形式如下:  类型标识符 *函数名(参数 ...

  8. Python第七天 函数 函数参数 函数里的变量 函数返回值 多类型传值 函数递归调用 匿名函数 内置函数

    Python第七天   函数  函数参数   函数里的变量   函数返回值  多类型传值     函数递归调用   匿名函数   内置函数 目录 Pycharm使用技巧(转载) Python第一天   ...

  9. 转 C++函数返回值,你必须注意的问题

    归根结底,C++所面临的问题要求它提供各种各样的机制以保证性能,也许,这辈子也见不到C++能安全有效的自己进行内存垃圾回收..... 老程序猿都会提醒菜鸟,注意函数的返回值,因为,很可能,你的函数返回 ...

随机推荐

  1. ECharts插件的使用

    ECharts插件:官网下载echarts.js开发者可以选择源码.下载地址:http://echarts.baidu.com/download.html 下载之后,echarts.js放在js文件夹 ...

  2. c++---天梯赛---N个数求和

    ★题目: ★难点:要求只能以有理数和分数去输出结果. ★分析:可以对输入的数据进行通分处理,随后把结果按格式输出. ★代码: #include<iostream> #include< ...

  3. 访问网时出现403 Forbidden错误的原因:

    1.你的IP被列入黑名单.2.你在一定时间内过多地访问此网站(一般是用采集程序),被防火墙拒绝访问了.3.网站域名解析到了空间,但空间未绑定此域名.4.你的网页脚本文件在当前目录下没有执行权限.5.在 ...

  4. Cannot declare class app\home\controller\Cases because the name is already in use

    Cannot declare class app\home\controller\Cases because the name is already in use 命名空间冲突了 use 模型类的时候 ...

  5. Python3 的注释

    单行注释 # 这是一个注释 print("Hello, World!") 多行注释 1:3个单引号 ''' 这是多行注释,用三个单引号 这是多行注释,用三个单引号 这是多行注释,用 ...

  6. JDBC 元数据 (DatabaseMetaData,ResultSetMetaData )

    Java 通过JDBC获得连接以后,得到一个Connection 对象,可以从这个对象获得有关数据库管理系统的各种信息,包括数据库中的各个表,表中的各个列,数据类型,触发器,存储过程等各方面的信息.根 ...

  7. Java多线程异常处理

    在java多线程程序中,所有线程都不允许抛出未捕获的checked exception,也就是说各个线程需要自己把自己的checked exception处理掉.这一点是通过java.lang.Run ...

  8. Hyperledger Fabric Transaction Flow——事务处理流程

    Transaction Flow 本文概述了在标准资产交换过程中发生的事务机制.这个场景包括两个客户,A和B,他们在购买和销售萝卜(产品).他们每个人在网络上都有一个peer,通过这个网络,他们发送自 ...

  9. PHP数据核心:Zend HashTable详解

    最近看了篇关于php内的hashtable的文章,PHP数据存储的核心,各种常量.变量.函数.类.对象等都用它来组织的.转载地址 http://www.phppan.com/2009/12/zend- ...

  10. python_怎么格式化字符串?

    案例: 如何对下面字典的key左对齐 {'dhqbl': 30, 'psfgj': 40, 'ontpqsb': 90, 'mrean': 110, 'klespjtr': 60, 'lprnkqhb ...