C++11 中std::vector 的emplace_back 使用的一处注意

vector的emplace_back  可以就地构造对象放入vector 而不用调用拷贝构造, 已经在项目中大量使用, 对于之前使用对象指针存放时不存在动态扩容问题, 现在有一个问题,  如果使用 emplace_back 直接存储对象, 并且取操作只取引用, 可以很方便, 也可以达到存取指针得高效, 也不用管理指针,写好构造析构函数即可, 

但是有一种情况需要注意:  如果vector 容器存在动态扩容, 你恰恰在扩容前记录了里面对象得指针,  那么在扩容后这个对象的地址是发生了变化, 变成不可用,  需要重新取, 如下代码例子:

 


struct row;
struct subrow;
struct dataset
{
	std::string name;
	std::vector<row> m_rows;
};

struct row
{
	row(double tx, double ty)
	{
		x = tx; y = ty;
	}
	double x, y;
	std::vector<subrow> m_subrows;
};

struct subrow
{
	subrow(double tt)
	{
		t = tt;
	}
	double t;
	row * parent;
};

void test1()
{
	dataset a;
	for (int i = 0; i < 1000; i++) {
		a.m_rows.emplace_back(i, i);
		row& r = a.m_rows[i];
		for (int j = 0; j < 50; j++)
		{
			r.m_subrows.emplace_back(j);
			subrow& sr = r.m_subrows[r.m_subrows.size()-1];
            //此处存储的r的地址很有可能在下次vector扩容后改变
			sr.parent = &r;
		}
	
	}
}

void test2()
{
	dataset a;
	for (int i = 0; i < 1000; i++) {
		a.m_rows.emplace_back(i, i);
		row& r = a.m_rows[i];
		for (int j = 0; j < 50; j++)
		{
			r.m_subrows.emplace_back(j);
		}

	}
    //在全部初始化好后再绑定parent , 如果再改变就需要重新绑定parent
	for (int i = 0; i < 1000; i++) 
	{
		row& r = a.m_rows[i];
		for (int j = 0; j < 50; j++)
		{
			r.m_subrows[j].parent = &r;
		}
	}

}

运行比较:

 

目前看来此种情况还是比较恶心, 但是相对数据量小,对象比较小的时候,无循环引用等常见情况 还是比较适用这种方式.  有循环引用还是不建议这种方式, 容易出错.

  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 5
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值