当前位置: 首页 > 知识库问答 >
问题:

关于数组外指针算法的C标准

经慈
2023-03-14

我读了很多关于指针算术和未定义行为的东西(链接,链接,链接,链接,链接)。最后总是得出相同的结论:指针算法只在数组类型上以及在数组[0]和数组[array_size+1]之间定义得很好(对于C标准来说,超过结尾的一个元素是有效的)。

我的问题是:这是否意味着当编译器看到与任何数组无关的指针算术(未定义的行为)时,它可以发出它想要的(甚至什么也不发出)?或者更多的是一种高级的“未定义行为”,即您可能到达未映射的内存、垃圾数据等,并且不能保证地址的有效性?

在本例中:

char test[10];
char * ptr = &test[0];
printf("test[-1] : %d", *(ptr-1))
#include <stdint.h>
#include <stdio.h>

void func1()
{}

void func2()
{}

int main()
{
  uint8_t * ptr1 = (uint8_t*) &func1;
  uint8_t * ptr2 = (uint8_t*) &func2;

  printf("Func 1 size : %ld", ptr2-ptr1);

  return 0;
}

由于ptr1ptr2不是数组的一部分,因此它被认为是未定义的行为。同样,这是否意味着编译器不能发出这些代码?或者“未定义的行为”是否意味着减法是毫无意义的,取决于系统(函数在内存中不连续,有填充等),但仍按预期发生?有没有定义好的方法来计算两个无关指针之间的减法?

共有1个答案

段干浩荡
2023-03-14

C标准没有为未定义的行为定义未定义的程度。如果它没有定义,它总是所有的赌注都失败了。

此外,现代编译器会搞乱指针起源的问题,编译器甚至会观察一个可能有效的指针是否正确导出,如果不正确,它可以调整程序行为。

如果您希望数学指针算术不可能使用UB,可以尝试在进行数学运算之前将指针强制转换到UINTPTR_T

#include <stdio.h>
int main()
{
    char a,b;
    printf("&a=%p\n", &a);
    printf("&b=%p\n", &b);
    printf("&a+1=%p\n", &a+1);
    printf("&b+1=%p\n", &b+1);
    printf("%d\n", &a+1==&b || &b+1==&a);
}
&a=0x7ffee4e36cae
&b=0x7ffee4e36caf
&a+1=0x7ffee4e36caf
&b+1=0x7ffee4e36cb0
0
 类似资料:
  • C++ 指针 在我们讲解指针数组的概念之前,先让我们来看一个实例,它用到了一个由 3 个整数组成的数组:#include <iostream> using namespace std; const int MAX = 3; int main () { int var[MAX] = {10, 100, 200}; for (int i = 0; i < MAX; i++) { cout << "Va

  • C++ 数组 您可以先跳过本章,等了解了 C++ 指针的概念之后,再来学习本章的内容。 如果您对 C++ 指针的概念有所了解,那么就可以开始本章的学习。数组名是一个指向数组中第一个元素的常量指针。因此,在下面的声明中: double balance[50]; balance 是一个指向 &balance[0] 的指针,即数组 balance 的第一个元素的地址。因此,下面的程序片段把 p 赋值

  • 本文向大家介绍在C中使用指针算法的数组总和,包括了在C中使用指针算法的数组总和的使用技巧和注意事项,需要的朋友参考一下 在此程序中,我们需要使用指针算法来查找数组元素的总和。 在这里,我们使用*表示存储在内存地址中的值,并且该地址将保留存储在变量中。因此,“ int * ptr”表示ptr是包含地址的变量,并且地址的内容是整数。 * p表示它是一个指针变量。使用此方法,我们将找出数组元素的总和。

  • C++ 指针 指针和数组是密切相关的。事实上,指针和数组在很多情况下是可以互换的。例如,一个指向数组开头的指针,可以通过使用指针的算术运算或数组索引来访问数组。请看下面的程序:#include <iostream> using namespace std; const int MAX = 3; int main () { int var[MAX] = {10, 100, 200}; int *pt

  • 主要内容:关于数组指针的谜题数组(Array)是一系列具有相同类型的数据的集合,每一份数据叫做一个数组元素(Element)。数组中的所有元素在内存中是连续排列的,整个数组占用的是一块内存。以 为例,该数组在内存中的分布如下图所示: 定义数组时,要给出数组名和数组长度,数组名可以认为是一个 指针,它指向数组的第 0 个元素。 在C语言中,我们将第 0 个元素的地址称为数组的首地址。以上面的数组为例,下图是 arr 的指向:

  • 本文向大家介绍指针数组、数组指针、函数指针的区别?相关面试题,主要包含被问及指针数组、数组指针、函数指针的区别?时的应答技巧和注意事项,需要的朋友参考一下 指针数组:首先它是一个数组,数组的元素都是指针,数组占多少个字节由数组本身的大小决定,每一个元素都是一个指针,在32 位系统下任何类型的指针永远是占4 个字节。它是“储存指针的数组”的简称。 数组指针:首先它是一个指针,它指向一个数组。在32

  • C++ 指针 指针是一个用数值表示的地址。因此,您可以对指针执行算术运算。可以对指针进行四种算术运算:++、--、+、-。 假设 ptr 是一个指向地址 1000 的整型指针,是一个 32 位的整数,让我们对该指针执行下列的算术运算: ptr++ 在执行完上述的运算之后,ptr 将指向位置 1004,因为 ptr 每增加一次,它都将指向下一个整数位置,即当前位置往后移 4 个字节。这个运算会在

  • 我想知道实际上是如何工作的。我认为这是一种计算数组长度的简单方法,并希望在使用它之前适当地理解它。我对指针算术不是很有经验,但根据我的理解,给出了数组第一个元素的地址。将按地址转到数组的末尾。但是不应该给出这个地址的值吗。而是打印出地址。我真的很感激你帮我把指针的东西弄清楚。 下面是我正在研究的一个简单示例: