指向数组的指针给 C++ 带来麻烦

Fra*_*cik 4 c++ pointers

好的,所以我从学习指南中获得了一些代码,并将其放入编译器中,但我并不完全理解结果。

#include <iostream>
using namespace std;

int main()
{
  int *p, *p1;

  p = new int[10];
  p1 = p++;

  for (int i=0; i<10;i++)
  {
    *p1 = i*10;
    p1++;
  }

  for (int i=0; i<10;i+=2)
  {
    p[i] = i*100;
  }

  for (int i=0; i<5;i++)
  {
    cout << *p++ << " ";
  }
}
Run Code Online (Sandbox Code Playgroud)

我基本上认为我理解除了读取的行之外的所有内容p1 = p++; 我认为这只是说p1现在指向相同的数组,p但我不完全知道p++它的一部分是做什么的。

当我将它放入编译器时,我得到0 20 200 40 400了我不明白的信息。有人能解释一下那一行是什么意思,然后为什么我会得到那个输出吗?

Wee*_*ble 8

p = new int[10];
Run Code Online (Sandbox Code Playgroud)

我们为 10 个整数分配一个空间,并指向p它的头部:

  p-> 0:[ - ]
      1:[ - ]
      2:[ - ]
      3:[ - ]
      4:[ - ]
      5:[ - ]
      6:[ - ]
      7:[ - ]
      8:[ - ]
      9:[ - ]
Run Code Online (Sandbox Code Playgroud)
p1 = p++;
Run Code Online (Sandbox Code Playgroud)

这看起来有点奇怪,并不是很好的风格。这会将 p 的值复制到 p1(因此它指向 p1 指向的位置),然后增加 p 以指向数组中的下一项。不保留指向动态分配数组开头的指针通常是一个坏主意,因为这意味着您无法释放它。(原则上,您当然可以将指针递减回到起始位置,但要做到这一点,您需要知道要递减的位置,并且保留原始指针要容易得多。如果您认为这是很多麻烦,就是这样!这就是 std::array 和 std::vector 存在的原因。但让我们回到手头的问题......)

      0:[ - ] <- p1
  p-> 1:[ - ]
      2:[ - ]
      3:[ - ]
      4:[ - ]
      5:[ - ]
      6:[ - ]
      7:[ - ]
      8:[ - ]
      9:[ - ]
Run Code Online (Sandbox Code Playgroud)
for (int i=0; i<10;i++)
{
    *p1 = i*10;
    p1++;
}
Run Code Online (Sandbox Code Playgroud)

这会循环十次,每次循环时,它都会设置 p1 指向的数组中的值,并将 p1 前进一位。最后,p1 指向数组末尾的一个位置。(没关系,但前提是没有任何东西试图读取或写入该位置。)

      0:[ 0 ]
  p-> 1:[ 10 ]
      2:[ 20 ]
      3:[ 30 ]
      4:[ 40 ]
      5:[ 50 ]
      6:[ 60 ]
      7:[ 70 ]
      8:[ 80 ]
      9:[ 90 ]
              <- p1
Run Code Online (Sandbox Code Playgroud)
for (int i=0; i<10;i+=2)
{
    p[i] = i*100;
}
Run Code Online (Sandbox Code Playgroud)

这有点令人困惑。请记住,p 指向原始数组的索引 #1,而不是索引 #0。这循环了五次,其中 i 的值为 0、2、4、6 和 8。由于 p 已经指向索引 #1 而不是 #0,这实际上更新了索引 #1、#3、#5、#7 和原始数组中的#9。

      0:[ 0 ]
  p-> 1:[ 0 ]
      2:[ 20 ]
      3:[ 200 ]
      4:[ 40 ]
      5:[ 400 ]
      6:[ 60 ]
      7:[ 600 ]
      8:[ 80 ]
      9:[ 800 ]
              <- p1
Run Code Online (Sandbox Code Playgroud)
for (int i=0; i<5;i++)
{
    cout << *p++ << " ";
}
Run Code Online (Sandbox Code Playgroud)

最后循环五次,每次输出 p 当前指向的元素,同时将 p 移动到下一项。

      0:[ 0 ]
      1:[ 0 ]     <
      2:[ 20 ]    <
      3:[ 200 ]   < These five elements were output.
      4:[ 40 ]    <
      5:[ 400 ]   <
 p->  6:[ 60 ]
      7:[ 600 ]
      8:[ 80 ]
      9:[ 800 ]
              <- p1
Run Code Online (Sandbox Code Playgroud)