1 模块化设计要求库接口隐藏实现细节

作为一个函数库来说,尽力降低和其调用方的耦合。是最主要的设计标准。

C语言,作为经典“程序=数据结构+算法”的践行者,在实现函数库的时候,必定存在大量的结构体定义,接口函数须要对这些结构体进行操作。同一时候,程序设计的模块化要求库接口尽量少的暴露事实上现细节,接口參数尽量使用基本数据类型。尽量避免在形參中暴露库内结构体的定义。

2 隐藏结构体的两种方法

以笔者粗浅的认识,有两种最经常使用的方法。可以实现库内结构体定义的隐藏:接口函数形參使用结构体指针,接口函数形參使用句柄。

2.1 通过结构体指针引用结构体

为了说明方便。先给出使用VC++写的一段样例代码。

库接口头文件 MySDK.h

#pragma once

#ifdef MYSDK_EXPORT
#define MYSDK_API __declspec(dllexport)
#else
#define MYSDK_API __declspec(dllimport)
#endif typedef struct _Window Window; /*预先声明*/ #ifdef __cplusplus
extern "C" {
#endif MYSDK_API Window* CreateWindow();
MYSDK_API void ShowWindow(Window* pWin); #ifdef __cplusplus
}
#endif

库实现文件MySDK.c

#define MYSDK_EXPORT

#include "MySDK.h"
#include <stdlib.h> struct _Window
{
int width;
int height;
int x;
int y;
unsigned char color[3];
int isShow;
}; MYSDK_API Window* CreateWindow()
{
Window* p = malloc(sizeof(Window));
if (p) {
p->width = 400;
p->height = 300;
p->x = 0;
p->y = 0;
p->color[0] = 255;
p->color[1] = 255;
p->color[2] = 255;
p->isShow = 0;
}
return p;
}
MYSDK_API void ShowWindow(Window* pWin)
{
pWin->isShow = 1;
}

库使用者代码

#include <stdio.h>
#include "../myDll/MySDK.h"
#pragma comment(lib, "../Debug/myDll.lib") int main(int argc, char** argv)
{
Window* pWin = CreateWindow();
ShowWindow(pWin); return 0;
}

当中MySDK.h和MySDK.c是库的实现; main.cpp是调用方程序实现。两方使用了相同的接口头文件MySDK.h。

可是从使用者角度,main.cpp里面仅仅知道库中有名为Window的一种结构体类型,可是却不能知道此机构体的实现细节(定义)。因为C/C++编译器是延迟依赖型编译器,仅仅要源代码中没有涉及到Window结构体内存布局的代码。编译时不须要知道Window的完整定义。可是仍然可以检查类型名称的正确性,比方假设client代码例如以下则会被编译器检查出问题:

    int* p = 0;
ShowWindow(p);

编译器尽管不知道ShowWindow(pWin)中pWin指向的结构体的实现细节,可是仍然可以确保实參类型为Window*。这也方便了调用方检查错误。

2.2 通过“句柄”(handle)来引用结构体

最先接触句柄的概念。是在Win32API中。可以断定Windows系统的内部定义了大量的结构体,如线程对象、进程对象、窗体对象、….。可是编程接口Win32API中却非常少提供这些结构体的定义。调用者通过一个称为“句柄”的值来间接引用要使用的结构体对象。

Win32API 中的句柄

比如,例如以下Win32API

   HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr); ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd);

窗体类型在Windows中一定是一个非常复杂的结构体,为了隐藏事实上现细节,微软採取了窗体句柄的概念来间接引用窗体结构体对象。为了实现这样的相应关系。库内部必须维护句柄和结构体对象的相应关系。

Linux API中的句柄

句柄的概念也广泛的应用在Linux平台API中。如

 int open(const char *pathname, int flags);
ssize_t read(int fd, void *buf, size_t count);

在Linux内部。文件一定是通过一个复杂的结构体来表示,可是在API中使用了一个简单整数对其进行引用,避免了向调用者暴露文件结构体的细节。

OpenGL API中的句柄

句柄相同应用到了OpenGL库中。

void WINAPI glGenTextures(
GLsizei n,
GLuint *textures
);
void WINAPI glBindTexture(
GLenum target,
GLuint texture
);

纹理在OpenGL库内部也是一个复杂的结构体,相同使用句柄的概念对外隐藏了实现细节。

3 句柄和指针的比較

3.1 句柄的优势与不足

句柄看起来真的不错。那么局部究竟是怎样映射到相应的结构体的呢?一个最easy想到的答案就是:直接把结构体对象的内存地址作为句柄。

然而实际上,大多数的库实现都不是这么做的。之所以不直接把内存地址作为句柄的值,我个人觉得有例如以下几个原因:

  • 从源代码保护角度,内存地址更easy被Hack。知道了结构体的内存地址,就行读取这块内存的内容,从而为推測结构体细节提供了方便。

  • 从程序稳定性角度,对于库内部维护的对象,调用者仅仅应该通过接口函数来訪问。假设调用者得到了对象的内存地址,那么就有可能有意或无意的进行直接改动,从而影响库的稳定执行。

  • 从可移植性角度,指针类型在32位和64位系统中具有不同的长度,这样就须要为定义两个名称反复的接口函数,造成各种不便。而比如OpenGL,使用int型作为句柄类型。则可以一个接口函数跨越多个平台。

  • 从简化接口头文件角度,使用指针至少须要事先声明结构体类型,如 struct Window; 而使用基本数据类型作为句柄,无需这样做。

句柄存在的不足有:

  • 编译器无法识别详细的结构体类型

    因为句柄的数据类型实际上是基本数据类型,所以编译器仅仅能进行常规的检查,不能识别详细的结构体类型。

   SECURITY_ATTRIBUTES sa;
HANDLE h = CreateMutex(&sa, TRUE, L"Mutex");
ReadFile(h, NULL, 0, 0, 0);

上述代码编译器并不会报错。因为相互排斥体对象和文件对象都是使用相同的句柄类型。

  • 效率可能稍差

    毕竟存在一个 依据句柄值-查找内存指针的过程,可能会稍稍影响执行效率。

3.2 指针的优势与不足

事实上指针和句柄是相对的,句柄的不足就是指针的优势,句柄的优势也是指针的不足。

4 怎样选择

对于大型跨平台库的设计,採用句柄;对于专用小型库。採用指针。

就我眼下的项目而言,是一个小型的C库project,库的目标群体也相对单一,所以本着简单够用的原则,我选择了使用指针的方式对外隐藏库内结构体的实现细节。

C语言开发函数库时利用不透明指针对外隐藏结构体细节的更多相关文章

  1. 13-C语言字符串函数库

    目录: 一.C语言字符串函数库 二.用命令行输入参数 回到顶部 一.C语言字符串函数库 1 #include <string.h> 2 字符串复制 strcpy(参数1,参数2); 参数1 ...

  2. Go语言【第十一篇】:Go数据结构之:结构体

    Go语言结构体 Go语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型,结构体是由一系列具有相同类型或不同类型数据构成的集合.结构体表示一项记录,比如:保存图书馆的书籍记 ...

  3. C语言结构体指针(指向结构体的指针)详解

    C语言结构体指针详解 一.前言 一个指向结构体的变量的指针表示的是这个结构体变量占内存中的起始位置,同样它也可以指向结构体变量数组. *a).b 等价于 a->b. "."一 ...

  4. c语言学习之基础知识点介绍(十二):结构体的介绍

    一.结构体的介绍 /* 语法: struct 结构体名{ 成员列表; }; 切记切记有分号! 说明:成员列表就是指你要保存哪些类型的数据. 注意:上面的语法只是定义一个新的类型,而这个类型叫做结构体类 ...

  5. c语言指针数组和结构体的指针

    指向数组的指针,先初始化一个数组,使用传统方式遍历 void main() { ] = { ,,,, }; ; i < ; i++) { printf("%d,%x\n", ...

  6. larave5.6 引入自定义函数库时,报错不能重复定义

    方法一:使用function_exists判断 方法二:使用命名空间 namespace test; function test(){ echo 'test/test'; } namespace te ...

  7. linux驱动开发( 五) 字符设备驱动框架的填充file_operations结构体中的操作函数(read write llseek unlocked_ioctl)

    例子就直接使用宋宝华的书上例子. /* * a simple char device driver: globalmem without mutex * * Copyright (C) 2014 Ba ...

  8. 【C语言入门教程】5.6 函数库和文件

    函数库是为代码复用建立的,将同一类型,需要在不同的程序里使用的函数放置在一起,就组成了一个函数库.如 C 语言的标准库,它集合了开发者常用的函数.开发者自行编写的函数也可以组成函数库,通常称之为自定义 ...

  9. 转:在 C# 中使用 P/Invoke 调用 Mupdf 函数库显示 PDF 文档

    在 C# 中使用 P/Invoke 调用 Mupdf 函数库显示 PDF 文档 一直以来,我都想为 PDF 补丁丁添加一个 PDF 渲染引擎.可是,目前并没有可以在 .NET 框架上运行的免费 PDF ...

随机推荐

  1. Winform 异步调用2 时间

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  2. idea UML快捷键

  3. MVC简单的解释

    MVC (Model-View-Controller,模型视图控制器)是一种软件的设计模式,它最早是由 20 世纪 70 年代的 Smalltalk 语言提出的,即把一个复杂的软件工程分解为三个层 ...

  4. Android Studio 将module打成jar包

    1.新建测试工程,工程里面有两个module,app是Android工程,mylibrary是Android Library库. 2.打开mylibrary目录下的build.gradle文件,加入下 ...

  5. 关于基础的Set 和Get

    先附上一篇文章,讲的很清楚 在Core中,我们要是先这样设置了.在我们对这个上下文做查询工作的时候,例如: var head = _OMSECDatabase.OmsEcorderHead.Where ...

  6. 网页前端状态管理库Redux学习笔记(一)

    最近在博客园上看到关于redux的博文,于是去了解了一下. 这个Js库的思路还是很好的,禁止随意修改状态,只能通过触发事件来修改.中文文档在这里. 前面都很顺利,但是看到异步章节,感觉关于异步说得很乱 ...

  7. dubbo之事件通知

    事件通知 在调用之前.调用之后.出现异常时,会触发 oninvoke.onreturn.onthrow 三个事件,可以配置当事件发生时,通知哪个类的哪个方法 1. 服务提供者与消费者共享服务接口 in ...

  8. 图像局部显著性—点特征(FREAK)

    参考文章:Freak特征提取算法  圆形区域分割 一.Brisk特征的计算过程(参考对比): 1.建立尺度空间:产生8层Octive层. 2.特征点检测:对这8张图进行FAST9-16角点检测,得到具 ...

  9. POJ_2411_Mondriaan's Dream_状态压缩dp

    Mondriaan's Dream Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 15407   Accepted: 888 ...

  10. WPF动态折线图

    此项目源码下载地址:https://github.com/lizhiqiang0204/WpfDynamicChart 效果图如下: 此项目把折线图制作成了一个控件,在主界面设置好参数直接调用即可,下 ...