在循环体(for, while等)内定义对象和变量的注意事项(C++)

循环体内定义变量容易产生莫名其妙的 bug ,其实是对循环理解不透彻导致的,本文通过几段代码来理清楚这个问题。

    for(int i = 0; i < 5; i++){
        int x;
        cout << "x: " << x << "    x的地址: " << &x << endl;
        x++;
    }

运行结果:

x: 1976155616    x的地址: 0x7ffe7abdd990
x: 1976155617    x的地址: 0x7ffe7abdd990
x: 1976155618    x的地址: 0x7ffe7abdd990
x: 1976155619    x的地址: 0x7ffe7abdd990
x: 1976155620    x的地址: 0x7ffe7abdd990

可以看出在循环体内定义变量每次循环变量的地址不变,值也被保留了下来,很容易让人认为变量 x 只定义一次,第二次循环会跳过这个步骤,真是这样的吗?
如果真的这样认为,我们很容易写出下面这样的代码,并且期望 std::vector xx; 这句只在第一次循环执行;每次循环 xx 会被push_back()一个元素进去,最后 xx 是一个长度为5的向量,但事与愿违,请看:

    for(int i = 5; i > 0; i--){
        std::vector<int> xx;
        cout << xx << endl;
        xx.push_back(i);
        cout << "xx: " << xx << endl;
    }

运行结果:

xx: 5
xx: 4
xx: 3
xx: 2
xx: 1

咦,长度都是1,难道每次循环 xx 都被定义一次?
我们动手去实验:

class Test{
public:
    int x;
    Test(){
        x = 10086;
        std::cout << "进入构造函数!" << std::endl;
    }

    ~Test(){
        cout << "析构函数!" << endl;
    }
};
int main(){
	    for(int i = 2; i > 0; i--){
	        Test test;
	        test.x++;
	        std::cout << "test.x: " << test.x << "    aq的地址: "<< &test << std::endl;
  	  }
}

运行结果:

进入构造函数!
test.x: 10087    test的地址: 0x7ffd9463e110
析构函数!
进入构造函数!
test.x: 10087    test的地址: 0x7ffd9463e110
析构函数!

可以看到每一次循环 class Test 都执行了一次构造函数和一次析构函数。也就是说,每单次循环结束程序都会跳出循环体的作用域,循环体内的局部变量都被释放;下一次循环重新进入循环体,里边的每个局部变量和对象都重新构造。
也许你会想,第一段代码不是变量 x 的地址没变,里边的值也保留下来了吗?其实这是由于代码复杂度低,编译器每次申请空间地址没变,当然里面的值也不会变了,可以把这种情况看成巧合。我们理解c++代码不应该把这种巧合拉进来考虑,因为它是不受 c++ 语法保证的。也就是说如果一个变量,你希望在一次循环结束后被保留到下一次循环,不希望被销毁,那么你就不应该这循环体内定义它。

  • 53
    点赞
  • 91
    收藏
    觉得还不错? 一键收藏
  • 6
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值