- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想了解拥有一个 void 指针的真正需要,例如在下面的代码中,我使用强制转换能够以不同的方式使用相同的 ptr,那么如果可以的话,为什么真的有一个 void 指针类型转换?
int main()
{
int x = 0xAABBCCDD;
int * y = &x;
short * c = (short *)y;
char * d = (char*)y;
*c = 0;
printf("x is %x\n",x);//aabb0000
d +=2;
*d = 0;
printf("x is %x\n",x);//aa000000
return 0;
}
最佳答案
基本 C 不支持将任何指针类型转换为任何其他指针类型(即,没有 C 标准要求的任何扩展或行为的 C)。 2018 C 标准在第 6.3.2.3 条第 7 段中说:
A pointer to an object type may be converted to a pointer to a different object type. If the resulting pointer is not correctly aligned for the referenced type, the behavior is undefined. Otherwise, when converted back again, the result shall compare equal to the original pointer…
在那段话中,我们看到两个限制:
int *
到 short *
不太可能失败,因为 int
通常比 short
具有更严格的对齐方式.但是,base C 不支持反向转换。假设您用 short x[20];
定义了一个数组。或 char x[20];
.然后数组将根据需要对齐 short
或 char
, 但不一定是 int
的需要, 在这种情况下 (int *) x
的行为不会由 C 标准定义。int *
转换而来的指针。访问 short
.标准确实对某些指针转换做了一些额外的保证。其中之一是上述段落的延续:
… When a pointer to an object is converted to a pointer to a character type, the result points to the lowest addressed byte of the object. Successive increments of the result, up to the size of the object, yield pointers to the remaining bytes of the object.
因此您可以使用从int *
转换而来的指针访问表示 int
的各个字节,并且您可以执行相同的操作来访问任何其他对象类型的字节。但是这种保证仅适用于访问具有字符类型的单个字节,而不是使用 short
。类型。
从上面我们知道在short * c = (short *)y;
之后在你的例子中,y
不一定指向 x
的任何部分它起源于——指针转换产生的值不能保证作为 short *
工作根本。但是,即使它确实指向了 x
所在的地方是,base C 不支持使用 c
访问这些字节,因为 6.5 7 说:
An object shall have its stored value accessed only by an lvalue expression that has one of the following types:
— a type compatible with the effective type of the object,
— a qualified version of a type compatible with the effective type of the object,
— a type that is the signed or unsigned type corresponding to the effective type of the object,
— a type that is the signed or unsigned type corresponding to a qualified version of the effective type of the object,
— an aggregate or union type that includes one of the aforementioned types among its members (including, recursively, a member of a subaggregate or contained union), or
— a character type.
所以 *c = 0;
C 不支持您的示例,原因有二:c
不一定指向 x
的任何部分或任何有效地址,即使是,修改部分 int
的行为x
使用 short
C 标准未定义类型。它可能在您的 C 实现中有效,甚至可能受您的 C 实现支持,但它并不严格符合 C 代码。
C 标准提供了 void *
当特定类型不合适时使用的类型。 6.3.2.3 1 对指向 void
的指针做出类似的保证就像它对对象指针所做的那样:
A pointer to
void
may be converted to or from a pointer to any object type. A pointer to any object type may be converted to a pointer tovoid
and back again; the result shall compare equal to the original pointer.
void *
用于必须处理任意对象类型的例程,例如 qsort
. char *
可以达到这个目的,但最好有一个单独的类型,明确表示没有特定类型与之关联。例如,如果函数的参数是 char *p
, 该函数可能会无意中使用 *p
得到一个它不想要的角色。如果参数是void *p
,则函数必须在使用指针访问对象之前将指针转换为特定类型。因此,为“通用指针”设置一个特殊类型有助于避免错误,并向阅读代码的人表明意图。
关于c - 如果指针可以转换为任何类型(在 c 中),为什么要 void pointer?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57980302/
我刚接触 C 语言几周,所以对它还很陌生。 我见过这样的事情 * (variable-name) = -* (variable-name) 在讲义中,但它到底会做什么?它会否定所指向的值吗? 最佳答案
我有一个指向内存地址的void 指针。然后,我做 int 指针 = void 指针 float 指针 = void 指针 然后,取消引用它们以获取值。 { int x = 25; vo
我正在与计算机控制的泵进行一些串行端口通信,我用来通信的 createfile 函数需要将 com 端口名称解析为 wchar_t 指针。 我也在使用 QT 创建一个表单并获取 com 端口名称作为
#include "stdio.h" #include "malloc.h" int main() { char*x=(char*)malloc(1024); *(x+2)=3; --
#include #include main() { int an_int; void *void_pointer = &an_int; double *double_ptr = void
对于每个时间步长,我都有一个二维矩阵 a[ix][iz],ix 从 0 到 nx-1 和 iz 从 0 到 nz-1。 为了组装所有时间步长的矩阵,我定义了一个长度为 nx*nz*nt 的 3D 指针
我有一个函数,它接受一个指向 char ** 的指针并用字符串填充它(我猜是一个字符串数组)。 *list_of_strings* 在函数内部分配内存。 char * *list_of_strings
我试图了解当涉及到字符和字符串时,内存分配是如何工作的。 我知道声明的数组的名称就像指向数组第一个元素的指针,但该数组将驻留在内存的堆栈中。 另一方面,当我们想要使用内存堆时,我们使用 malloc,
我有一个 C 语言的 .DLL 文件。该 DLL 中所有函数所需的主要结构具有以下形式。 typedef struct { char *snsAccessID; char *
指针, C语言的精髓 莫队先咕几天, 容我先讲完树剖 (因为后面树上的东西好多都要用树剖求 LCA). 什么是指针 保存变量地址的变量叫做指针. 这是大概的定义, 但是Defad认为
我得到了以下数组: let arr = [ { children: [ { children: [], current: tru
#include int main(void) { int i; int *ptr = (int *) malloc(5 * sizeof(int)); for (i=0;
我正在编写一个程序,它接受一个三位数整数并将其分成两个整数。 224 将变为 220 和 4。 114 将变为 110 和 4。 基本上,您可以使用模数来完成。我写了我认为应该工作的东西,编译器一直说
好吧,我对 C++ 很陌生,我确定这个问题已经在某个地方得到了回答,而且也很简单,但我似乎找不到答案.... 我有一个自定义数组类,我将其用作练习来尝试了解其工作原理,其定义如下: 标题: class
1) this 指针与其他指针有何不同?据我了解,指针指向堆中的内存。如果有指向它们的指针,这是否意味着对象总是在堆中构造? 2)我们可以在 move 构造函数或 move 赋值中窃取this指针吗?
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: C : pointer to struct in the struct definition 在我的初学者类
我有两个指向指针的结构指针 typedef struct Square { ... ... }Square; Square **s1; //Representing 2D array of say,
变量在内存中是如何定位的?我有这个代码 int w=1; int x=1; int y=1; int z=1; int main(int argc, char** argv) { printf
#include #include main() { char *q[]={"black","white","red"}; printf("%s",*q+3); getch()
我在“C”类中有以下函数 class C { template void Func1(int x); template void Func2(int x); }; template void
我是一名优秀的程序员,十分优秀!