- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
注意:这个问题试图改进我试图问 here 的问题,但没有达到目的。
此外,我还看到了 this 和 this 。他们讨论相似的概念,但不回答这些问题。
我的环境是 Windows 10,为了测试我使用了两个编译器,CLANG 和 GCC。
我通过 void *
函数参数传递变量,需要转换它们。我想获得一些关于我在不同类型的方法之间看到的不一致的反馈。
以下是测试函数的精简描述,它使用 void *
参数和一个枚举值参数来指示传入的类型,以适应多种输入类型。
void func(void *a, int type)
{
switch(type) {
case CHAR://char
char cVar1 = (char)a; //compiles with no warnings/errors, seems to work
char cVar2 = *(char *)a; //compiles with no warnings/errors, seems to work
break;
case INT://int
int iVar1 = (int)a; //compiles with no warnings/errors, seems to work
int iVar2 = *(int *)a; //compiles with no warnings/errors, seems to work
break;
case FLT://float
float fVar1 = (float)a; //compile error: (a1)(b1)
float fVar2 = *(float *)a; //requires this method
case DBL://double
double dVar1 = (double)a; //compile error: (a1)(b1)(b2)
double dVar2 = *(double *)a;//this appears to be correct approach
break;
};
}
调用方法:
int main(void)
{
char c = 'P';
int d = 1024;
float e = 14.5;
double f = 0.0000012341;
double g = 0.0001234567;
void *pG = &g;
func(&c, CHAR);//CHAR defined in enumeration, typical
func(&d, INT);
func(&e, FLT);
func(&f, DBL);
func(pG, DBL);
return 0;
}
上面评论中与标志相关的确切错误文本如下:
CLANG - 3.3 版
海湾合作委员会 - (tdm-1) 5.1.0
供下面讨论引用
type var = (type)val;
type var = *(type *)val;
我的结果表明转换 float
和 double
需要方法 2。
但是对于 char
和 int
方法 2 似乎是可选的,即方法 1 编译得很好,并且似乎始终如一地工作。
问题:
从 void *
函数中恢复一个值似乎参数应该总是需要方法 2,那么为什么方法 1(看起来to) 使用char
和int
类型?这是未定义的行为吗?
如果方法 1 适用于 char
和 int
,为什么它至少不能适用于 float
类型?这不是因为它们的大小不同,即:sizeof(float) == sizeof(int) == sizeof(int *) == sizeof(float *)
。是因为严格的别名违规吗?
最佳答案
C standard明确允许指针和整数类型之间的转换。这在第 6.3.2.3 节中关于指针转换有详细说明:
5 An integer may be converted to any pointer type. Except as previously specified, the result is implementation-defined, might not be correctly aligned, might not point to an entity of the referenced type, and might be a trap representation.
6 Any pointer type may be converted to an integer type. Except as previously specified, the result is implementation-defined. If the result cannot be represented in the integer type, the behavior is undefined. The result need not be in the range of values of any integer type.
假设您在将整数类型传递给函数时将其转换为 void *
,然后将其转换回正确的整数类型,这可以完成提供实现允许它。特别是 GCC 将允许这样做,假设所讨论的整数类型至少与 void *
一样大。
这就是转换适用于 char
和 int
情况的原因,但是您需要传入值(转换为 void *
) 而不是地址。
例如,如果您这样调用函数:
func4((void *)123, INT);
那么函数可以这样做:
int val = (int)a;
val
将包含值 123。但是如果您这样调用它:
int x = 123;
func4(&x, INT);
然后函数中的 val
将包含 main
中的 x
的地址转换为整数值。
根据第 6.5.4p4 节关于转换运算符的规定,明确不允许在指针类型和浮点类型之间进行转换:
A pointer type shall not be converted to any floating type. A floating type shall not be converted to any pointer type.
当然,通过 void *
传递值最安全的方法是将值存储在适当类型的变量中,传递其地址,然后转换 void *
在函数中返回正确的指针类型。这保证有效。
关于c - `void *` 函数参数类型不一致的转换方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59754887/
我想了解 Ruby 方法 methods() 是如何工作的。 我尝试使用“ruby 方法”在 Google 上搜索,但这不是我需要的。 我也看过 ruby-doc.org,但我没有找到这种方法。
Test 方法 对指定的字符串执行一个正则表达式搜索,并返回一个 Boolean 值指示是否找到匹配的模式。 object.Test(string) 参数 object 必选项。总是一个
Replace 方法 替换在正则表达式查找中找到的文本。 object.Replace(string1, string2) 参数 object 必选项。总是一个 RegExp 对象的名称。
Raise 方法 生成运行时错误 object.Raise(number, source, description, helpfile, helpcontext) 参数 object 应为
Execute 方法 对指定的字符串执行正则表达式搜索。 object.Execute(string) 参数 object 必选项。总是一个 RegExp 对象的名称。 string
Clear 方法 清除 Err 对象的所有属性设置。 object.Clear object 应为 Err 对象的名称。 说明 在错误处理后,使用 Clear 显式地清除 Err 对象。此
CopyFile 方法 将一个或多个文件从某位置复制到另一位置。 object.CopyFile source, destination[, overwrite] 参数 object 必选
Copy 方法 将指定的文件或文件夹从某位置复制到另一位置。 object.Copy destination[, overwrite] 参数 object 必选项。应为 File 或 F
Close 方法 关闭打开的 TextStream 文件。 object.Close object 应为 TextStream 对象的名称。 说明 下面例子举例说明如何使用 Close 方
BuildPath 方法 向现有路径后添加名称。 object.BuildPath(path, name) 参数 object 必选项。应为 FileSystemObject 对象的名称
GetFolder 方法 返回与指定的路径中某文件夹相应的 Folder 对象。 object.GetFolder(folderspec) 参数 object 必选项。应为 FileSy
GetFileName 方法 返回指定路径(不是指定驱动器路径部分)的最后一个文件或文件夹。 object.GetFileName(pathspec) 参数 object 必选项。应为
GetFile 方法 返回与指定路径中某文件相应的 File 对象。 object.GetFile(filespec) 参数 object 必选项。应为 FileSystemObject
GetExtensionName 方法 返回字符串,该字符串包含路径最后一个组成部分的扩展名。 object.GetExtensionName(path) 参数 object 必选项。应
GetDriveName 方法 返回包含指定路径中驱动器名的字符串。 object.GetDriveName(path) 参数 object 必选项。应为 FileSystemObjec
GetDrive 方法 返回与指定的路径中驱动器相对应的 Drive 对象。 object.GetDrive drivespec 参数 object 必选项。应为 FileSystemO
GetBaseName 方法 返回字符串,其中包含文件的基本名 (不带扩展名), 或者提供的路径说明中的文件夹。 object.GetBaseName(path) 参数 object 必
GetAbsolutePathName 方法 从提供的指定路径中返回完整且含义明确的路径。 object.GetAbsolutePathName(pathspec) 参数 object
FolderExists 方法 如果指定的文件夹存在,则返回 True;否则返回 False。 object.FolderExists(folderspec) 参数 object 必选项
FileExists 方法 如果指定的文件存在返回 True;否则返回 False。 object.FileExists(filespec) 参数 object 必选项。应为 FileS
我是一名优秀的程序员,十分优秀!