动态创建的对象使用基于范围的 for 循环显示为 nullptr

Dynamically created objects appear as nullptr using ranged-based for-loop

本文关键字:for 循环 范围 显示 nullptr 对象 创建 于范围 动态      更新时间:2023-10-16

我有一个指向一些粒子对象的指针数组:Particle* particles[ TOTAL_PARTICLES ]; TOTAL_PARTICLES = 10;在哪里。

当我在下面运行这段代码时,

for( Particle* p : particles )
{
    cout << "Attempting to create particle!n";
    p = new (nothrow) Particle( x, y );
    cout << p << "n";

    if ( p == nullptr )
    {
        cout << "Error assigning memory!n";
    }
    else
    {
        cout << "Particle created!n";
    }
}
for( Particle* p : particles )
{
    cout << "Nullptr? " << ( p == nullptr ) << "n";
}

这是输出:

 Attempting to create particle!
 0x2393c00
 Particle created!
 Attempting to create particle!
 0x2393c20
 Particle created!
 Attempting to create particle!
 0x2393c40
 Particle created!
 Attempting to create particle!
 0x2393c60
 Particle created!
 Attempting to create particle!
 0x2393c80
 Particle created!
 Attempting to create particle!
 0x2393ca0
 Particle created!
 Attempting to create particle!
 0x2393cc0
 Particle created!
 Attempting to create particle!
 0x2393ce0
 Particle created!
 Attempting to create particle!
 0x2393d00
 Particle created!
 Attempting to create particle!
 0x2393d20
 Particle created!
 Nullptr? 0
 Nullptr? 0
 Nullptr? 0
 Nullptr? 0
 Nullptr? 1
 Nullptr? 0
 Nullptr? 1
 Nullptr? 0
 Nullptr? 1
 Nullptr? 1

我不明白为什么,即使控制台告诉我所有粒子都已成功创建并且具有指向它们的非空指针,但当我再次迭代它们时,其中一些突然变为 nullptr。

最令人困惑的是,这只有在我使用基于范围的 for 循环时才会发生。如果我将第一个for( Particle* p : particles )替换为 for( int i = 0; i < TOTAL_PARTICLES; ++i )并将第二个 for 循环保留为基于范围的 for 循环,则代码将按预期工作。

有没有人知道为什么指向粒子的指针数组突然变成 nullptr?还有一件事需要注意:我尝试在不实际创建任何粒子的情况下迭代指针数组,并且我也收到了类似的输出,这让我怀疑第一个基于 ranged 的 for 循环中的代码以某种方式没有以某种方式"保存"。

for( Particle* p : particles )
{
    //...
    p = new (nothrow) Particle( x, y );
    //...
}

请注意,p是一个Particle*。这意味着您只是将指针从数组中复制出来,因此赋值不会更改数组。

您可以改为引用指针:

for (Particle*& p : particles)
{
    //...
}

引用指针意味着当您分配给 p 时,数组中的指针将被更新。

在基于范围的循环中,pparticles 数组中每个指针的副本。所以在这个作业中

p = new (nothrow) Particle( x, y );

particles数组中的原始指针没有任何变化。

在第一个循环中,您不是分配给数组中的指针,而是分配给它们的临时副本。

for( Particle*& p : particles )

相反。