- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
对于两个(或更多)struct
:Base
和 Sub
,第一个(未命名)struct
>,从 Base
到 Sub
的转换/转换是否安全,反之亦然?
struct Base{
struct{
int id;
// ...
};
char data[]; // necessary?
}
struct Sub{
struct{
int id;
// same '...'
};
// actual data
};
这些功能是否保证安全且技术上正确? (另外:零长度的 char data[]
成员是否必要且有用?)
struct Base * subToBase(struct Sub * s){
return (struct Base*)s;
}
struct Sub * baseToSub(struct Base * b){
if(b->id != SUB_ID){
return NULL;
}
return (struct Sub*)b;
}
编辑
我没有计划在 Sub
中嵌套比 Base
更远的部分,而是保留添加其他子类型的可能性(直接在 Base
下) code>) 之后无需更改 Base
。我主要关心的是指向 struct
的指针是否可以在 Base
和任何 sub 之间安全地来回转换。如果能引用 (C11) 标准,我们将不胜感激。
编辑 v2
稍微更改了措辞以阻止 OOP/继承讨论。我想要的是一个没有 union
的标记 union ,以便以后可以扩展。我没有做额外嵌套的计划。需要其他子类型功能的子类型可以明确地这样做,而无需进行任何进一步的嵌套。
对于脚本解释器 1 我做了一个伪面向对象 tagged-union类型系统,没有 union
。它有一个(抽象的) generic 基类型 Object
和几个(特定的) 子类型,例如 String
, Number
, List
等。每个类型-struct
都有以下未命名的 struct
作为第一个成员:
#define OBJHEAD struct{ \
int id; \
int line; \
int column; \
}
id
标识对象的类型,line
和column
应该(也)是不言自明的。各种对象的简化实现:
typedef struct Object{
OBJHEAD;
char data[]; // necessary?
} Object;
typedef struct Number{
OBJHEAD;
int value; // only int for simplicity
} Number;
typedef struct String{
OBJHEAD;
size_t length;
char * string;
} String;
typedef struct List{
OBJHEAD;
size_t size;
Object * elements; // may be any kind and mix of objects
} List;
Object * Number_toObject(Number * num){
return (Object*)num;
}
Number * Number_fromObject(Object * obj){
if(obj->type != TYPE_NUMBER){
return NULL;
}
return (Number*)obj;
}
我知道最优雅和技术上最正确的方法是使用 enum
作为 id
和 union
作为各种子类型。但我希望类型系统是可扩展的(通过某种形式的类型注册表),以便以后可以添加类型而无需更改所有与 Object
相关的代码。
稍后/外部添加可能是:
typedef struct File{
OBJHEAD;
FILE * fp;
} File;
无需更改 Object
。
这些转换是否保证安全?
(至于小的宏滥用:OBJHEAD
当然会被广泛记录,因此其他实现者将知道哪些成员名称不应该使用。这个想法不是隐藏标题,而是每次都粘贴它。)
最佳答案
将指向一种对象类型的指针转换为指向不同对象类型的指针(例如,通过强制转换)是允许的,但如果生成的指针未正确对齐,则行为未定义 (C11 6.3.2.3/7) .根据 Base
和 Sub
的成员以及依赖于实现的行为,Base *
不一定会转换为 Sub *
正确对齐。例如,给定...
struct Base{
struct{
int id;
};
char data[]; // necessary?
}
struct Sub{
struct{
int id;
};
long long int value;
};
...可能是实现允许 Base
对象在 32 位边界上对齐,但要求 Sub
对象在 64 位边界上对齐,甚至更严格的。
这一切都不受 Base
是否具有灵活的数组成员的影响。
取消引用通过转换不同类型的指针值获得的一种类型的指针值是否安全是一个不同的问题。一方面,C 对实现如何选择布局结构的限制很少:成员必须按照它们声明的顺序进行布局,并且在第一个之前不能有任何填充,否则,实现可以自由支配。据我所知,在你的情况下,如果你的两个结构的匿名 struct
成员有多个成员,则不需要以相同的方式布置它们。 (如果他们只有一个成员,那么为什么要使用匿名结构?)假设 Base.data
的起始偏移量与 中匿名结构之后的第一个元素相同也是不安全的>子
。
在实践中,取消引用您的 subToBase()
的结果可能没问题,您当然可以实现测试来验证这一点。此外,如果您有一个 Base *
是通过从 Sub *
转换获得的,那么将其转换回来的结果,例如通过 baseToSub()
,保证与原始 Sub *
相同(又是 C11 6.3.2.3/7)。在这种情况下,转换为 Base *
并返回对将指针取消引用为 Sub *
的安全性没有影响。
另一方面,虽然我很难在标准中找到它的引用,但我不得不说 baseToSub()
在一般情况下是非常危险的。如果一个 Base *
实际上并不指向一个 Sub
被转换为 Sub *
(这本身是允许的),那么它是 < em>不 取消引用该指针以访问未与 Base
共享的成员是安全的。特别是,鉴于我上面的声明,如果引用的对象实际上是一个 Base
,那么声明 Base.data
绝不会阻止 ((Sub *) really_a_Base_ptr)->value
产生未定义的行为。
为了避免所有未定义和实现定义的行为,您需要一种避免转换并确保一致布局的方法。 @LoPiTaL 建议在您的 Sub
结构中嵌入一个类型化的 Base
结构,在这方面是一个很好的方法。
关于c - 没有 union 的结构类型别名/标记 union ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32638712/
上下文:我已经在域根上设置了一个用于 GIS 测试和 dokuwiki 的虚拟机服务器。我想在子域上提供 gis Web 应用程序,这样 dokuwiki url 重命名就不会发生冲突(而且感觉更干净
这个问题在这里已经有了答案: If two variables point to the same object, why doesn't reassigning one variable affe
代码如下: Select Branch.BranchName, Sum(Inventory.OnHand) From Inventory, Branch Inventory.BranchNum = B
我的目标是创建一种“Javascript 库”,如果您可以这样调用它的话。我打算在浏览网页时使用它来操作页面,将其作为 greasemonkey 脚本动态加载。这个想法是将“win”映射到window
是否可以在 javascript 中以某种方式为本地变量分配别名/引用? 我的意思是类似 C 的东西: function foo() { var x = 1; var y = &x; y+
SQLite 别名 您可以暂时把表或列重命名为另一个名字,这被称为别名。使用表别名是指在一个特定的 SQLite 语句中重命名表。重命名是临时的改变,在数据库中实际的表的名称不会改变。 列别名用来
我有以下导入(在同一个 TS 源文件中): import {Vector as sourceVector} from "ol/source"; import {Vector} from "ol/lay
我想弄清楚 this 到底是什么行是在这个 github json 项目的 cmake 文件中, add_library(${NLOHMANN_JSON_TARGET_NAME} INTERFACE)
使用集合时如何创建别名 (AS)。 $collection->addAttributeToSelect('total_qty') 可以这样做: $collection->getSelect()->co
这个问题在这里已经有了答案: Pass an argument to a Git alias command (4 个回答) 去年关闭。 我的团队在不同的分支上工作,通常,我需要快速从分支切换来帮助他
我正在寻找一种向我的 vimrc 添加自定义命令或别名的方法,以便快速切换到 Vim 中的不同目录。我曾经设置过一些东西,但丢失了那个配置。这是用于 Linux 设置。 基本上,我想设置一个别名(如果
我使用 Delphi 和 Firebird 1.5 开发了一个应用程序,其中服务器与应用程序位于同一台计算机上。我现在正在将应用程序部署到另一个站点,其中 Firebird 服务器( super 服务
我创建了一个全新的 React 应用 create-react-app demo 我需要为某些目录/组件创建别名,例如: import { Header } from '@uicomponents'
我正在尝试打印有向图,并且不断更改各种节点属性,例如颜色和形状。有什么方法可以使用一次定义并用于多个节点的变量?理想情况下,我想要这样的东西: digraph g { building_color =
server { listen 80; server_name pwta; root html; location /test/{ alia
我想匹配多个数组,并在匹配时构建另一个数组。该键可以在任意数量的数组中匹配,也可以根本不匹配。 [ [{ 'a': 13 }, { 'b': 62 }, { 'c': 93 }, { 'd': 52
经过审查后,我需要更改一些代码并将所有更改代码重新推送到远程分支上。 所以我想使用别名(例如:git repushall)在 git 上自动化这些不同的命令: git add . git commit
我有一个这样的查询: select a1.name, b1.info from (select name, id, status from table1 a) as a1 righ
我想给一些 racket 2htdp 函数/宏起别名,这样我就可以为我的 child 将它们翻译成另一种语言。 我可以简单地用define给函数起别名。我在使用 big-bang 结构时遇到了麻烦;例
经过审查后,我需要更改一些代码并将所有更改代码重新推送到远程分支上。 所以我想使用别名(例如:git repushall)在 git 上自动化这些不同的命令: git add . git commit
我是一名优秀的程序员,十分优秀!