如何取消索引结构?例:

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

10-06 14:40