如何取消索引结构?例:
typedef struct String_s {
int current_location;
int size;
char data[0];
} String;
char* String_getCString(String *str){
return &str->data[0];
}
//this is supposed to take the result of 'String_getCString' and reverse the process to get the String*
//i.e. String_getCString(CString_getString(str)) == str
String* CString_getString(char *str){
//???
}
int foo(char *cstr){
printf("%s\n", cstr);
fflush(0);
free(CString_getString(cstr));
}
int main(int argc, char *argv[]){
const char *hello_world = "hello world";
String *str = (String*)malloc(sizeof(String)+1000*sizeof(char));
str->size = 1000;
str->count = strlen(hello_world);
char *cstr = String_getCString(str);
strcpy(cstr, hello_world);
foo(cstr);
return 0;
}
最佳答案
我不是100%肯定我理解您想要CString_getString
做什么,但是如果您希望它在传递嵌入式String
字段的地址时返回整个data
对象的地址,那很简单,但很危险:
#include <stddef.h>
String *CString_getString(char *str)
{
return (String *)(str - offsetof(String, data));
}
如果希望“取消索引”的字段类型不是
[signed/unsigned/] char
,则需要在减法之前将输入指针转换为char *
,然后再转换为所需的返回类型。这很危险,因为
CString_getString
无法知道您是否传入了确实是str
对象的嵌入式data
字段的String
。如果弄错了,C编译器会坐下来,看着它在运行时爆炸。但是,可以说,这并不比人们一直在C语言中所做的任何事情都要糟糕,这可能是一种有用的技术。例如,它在Linux中非常常用:http://lxr.free-electrons.com/ident?i=container_of