回 帖 发 新 帖 刷新版面

主题:[讨论]C语言如何通过在结构中设置一个 void* 来得到“封装的结构”

标题:How can inline functions help with the tradeoff of safety vs. speed?
    In straight C, you can achieve "encapsulated structs" by putting a void* in a struct, in which case the void* points to the real data that is unknown to users of the struct. Therefore users of the struct don't know how to interpret the stuff pointed to by the void*, but the access functions cast the void* to the appropriate hidden type. This gives a form of encapsulation. 
    Unfortunately it forfeits type safety, and also imposes a function call to access even trivial fields of the struct (if you allowed direct access to the struct's fields, anyone and everyone would be able to get direct access since they would of necessity know how to interpret the stuff pointed to by the void*; this would make it difficult to change the underlying data structure). 
    Function call overhead is small, but can add up. C++ classes allow function calls to be expanded inline. This lets you have the safety of encapsulation along with the speed of direct access. Furthermore the parameter types of these inline functions are checked by the compiler, an improvement over C's #define macros. 

   有几句没翻译。
大概的翻译:
   在传统的 C 中,你可以通过在结构中设置一个 void* 来得到“封装的结构”,在这种情况下,指向实际数据的 void* 指针对于结构的用户来说是未知的。因此结构的用户不知道如何解释void*指针所指内容,但是存取函数可以将 void* 转换成适当的隐含类型。这样给出了封装的一种形式。
    不幸的是这样做丧失了类型安全,并且也将繁琐的对结构中的每个域的访问强加于函数调用。(如果你允许直接存取结构的域,那么对任何能直接存取的人来说,了解如何解释 void* 指针所指内容就是必要的了;这样将使改变底层数据结构变的困难)。
    虽然函数调用开销是很小的,但它会被累积。C++类允许函数调用以内联展开。这样让你在得到封装的安全性时,同时得到直接存取的速度。此外,内联函数的参数类型由编译器检查,这是对 C 的 #define 宏的一个改进。 


问题:
    如何通过在结构中设置一个 void* 来得到“封装的结构”,我不太清楚!
哪位高手能否举个例子!谢谢!

回复列表 (共4个回复)

沙发

void * CreateHiddenStruct();   //创建

Process1HiddenStruct(void *); //实现1
Process2HiddenStruct(void *); //实现2
....

DestroyHiddenStruct(void *);   //销毁

用户必须保证process和destory的传入参数必须从CreateHiddenStruct获得。
但是,万一用户不保证呢?没办法了。编译器不会发现任何问题。

板凳

这段话说到“不可直接存取”的意思是
因为你不知道void *的类型,所以你不可以通过
a.val=xx;这种方式直接存取,而必须通过函数。
但是在c++中,允许函数内连。这样,就可以用函数的形式来达到直接存取的目的。

3 楼

谢谢你的解答!

4 楼

是这样的,
typedef struct apple {
    void *weight; /* 作float或double类型,取决于哪个函数 */
} APPLE;

void setWeightF (APPLE *apple, float *weight)
{
    apple->weight = weight;
}

void setWeightD (APPLE *apple, double *weight)
{
    apple->weight = weight;
}

void showWeightF (APPLE *apple)
{
    printf ("%f\n", *(float *)(apple->weight));
}

我来回复

您尚未登录,请登录后再回复。点此登录或注册