新书推荐:1.2 动态链接库与API

2024-06-03 08:12

本文主要是介绍新书推荐:1.2 动态链接库与API,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本节必须掌握的知识点:

        kernel32.dll

        user32.dll

        gdi32.dll

动态链接库

最早的软件开发过程,所有的功能实现都是有程序员独立完成的。在这个过程中,我们很快就会发现,有很多常用的功能模块是可以重复利用的,我们将其封装为功能函数。我们将这些功能函数独立编译成统一格式的OBJ目标文件,并存放于一个LIB库文件中。源程序中可以调用这些第三方功能函数,当编译器编译时,暂时将这些第三方功能函数的地址空下,待链接器链接时,在LIB库中查找对应的OBJ功能模块,并将其添加到源代码中,填写空下的函数地址。这就是我们所谓的静态链接库了。

在Windows操作系统中,采用的是另外一种形式的链接库,我们将特定功能的第三方功能函数封装到一个dll形式的库中,我们将其称为动态链接库。动态链接库dll文件通常保存在“C:\Windows\System32”系统目录下。源程序中同样直接调用第三方功能函数(需要头文件声明函数原型)。编译器编译链接时,会将第三方功能函数所在的dll动态链接库名称、函数名称、函数序号以及函数相对地址记录到一个PE文件(Windows指定的exe二进制格式文件)的节区。待编译后的EXE文件由操作系统加载到低2GB空间时,加载dll动态链接库(除了系统链接库之外,也可以是自定义的dll动态链接库)。此时,应用程序和dll动态链接库同处于一个4GB虚拟空间内,然后根据所引用的函数名或序号在动态链接库内查找该函数真实的地址,替换函数地址表内的相对地址,这样形成一个完整的程序就可以正常运行了。

为何要采用动态链接库这种形式呢?随着应用程序的功能越来越多,所引用的库函数也变得越来越多,如果继续采用静态链接库的形式,无疑会大大增加程序的体积,基于有限的存储空间和程序加载效率的考虑,设计动态链接库的形式可以很好地解决上述问题。动态链接库的缺点是对操作系统环境的依赖。

API函数

对于程序员来说,Windows操作系统的功能完全由API(Application Programming Interface首字母的缩写,意思为程序之间的接口)来定义。API涵盖了应用程序所能调用的全部操作系统函数,以及相关的数据类型和结构。在Windows中,API还隐含了一种特殊的程序结构,我们将在后续章节中详细探讨这种程序结构。

早期的Windows 1.0版本只能支持不到450个函数,而今天则支持数千个API函数。

我们把windows 1.0到3.1版本称为win16,win95及98之后的所有NT版本称为win32。从win16到win32转化过程中大部分函数保持不变,但也有一些需要扩展,比如图形坐标点数值从win16的16位扩展为32位。此外,win16中有些函数调用返回的二维坐标点被压缩到一个32位的整数里,这在win32中就行不通了。为解决这些问题,win32增加了一些新的函数。

所有32位版本的windows既支持win16 API以保证和原先的程序兼容,也支持win32 API以运行新的应用程序。有意思的是,windows NT与win95和win98的工作方式不同。在WINDOWS NT中,win16的函数调用通过一个翻译层先被转换成win32的函数调用,然后再由操作系统来处理。而在windows95和windows 98中正好相反,win32的函数调用通过一个翻译层先被转换成win16的函数调用,然后再由操作系统来处理。

       接下来,我们逐一介绍三个Windows应用程序必备的dll动态链接库。

1.2.1 kernel32.dll

       kernel32.dll是一个Windows操作系统核心动态链接库文件。库中包含了九百多个API函数,这些API函数主要负责内存管理、进程和线程管理、调试、错误处理和时间处理等功能。

内存管理:kernel32.dll包含了管理内存的API函数。例如:

●GlobalAlloc和LocalAlloc函数用于从堆中分配内存。

●GlobalFree和LocalFree函数用于释放之前分配的内存。

●VirtualAlloc和VirtualFree函数用于分配和释放大块的进程虚拟地址空间(4GB)中的内存。

●HeapAlloc和HeapFree函数用于管理堆中的内存。

●HeapCreate函数创建私有堆。

类似的虚拟空间和堆栈的操作函数还有很多,不再一一列举。

进程和线程管理:Windows操作系统作为多任务系统,需要创建、管理、维护和切换多个任务。Windows操作系统是进程强相关的操作系统,进程与进程之间可以进行读写操作。Windows操作系统中的进程可以包含一个或多个线程,进程是一些资源的集合,进程内的所有线程可以共享这些资源。在多任务系统中,任务切换其实就是线程切换。可以是同一进程中的线程切换,也可以是不同进程间的线程切换。kernel32.dll包含的常用的进程和线程函数有:

●CreateProcess函数创建进程。

●CreateThread函数创建线程。

●CloseHandle函数减少一个内核对象计数。

●WaitForSingleObject函数检测内核对象信号状态。

●CreateEvent函数创建事件内核对象。

●TerminateProcess函数结束进程。

●TerminateThread函数结束线程。

调试函数:kernel32.dll还包含了一些和调试相关的函数,包括进程间的远程读写。例如:

●DebugActiveProcess函数使调试程序能够附加到活动进程并对其进行调试。

●GetThreadContext函数可以检索指定线程的上下文。

●ReadProcessMemory函数从指定进程中的内存区域读取数据。

●WriteProcessMemory函数将数据写入到指定进程中的内存区域。

错误处理:在程序开发中,难免会出现一些错误和异常,检测和处理这些错误和异常需要使用kernel32.dll中的函数。例如:

●SetUnhandledExceptionFilter函数安装一个顶层的全局异常处理程序。

●GetExceptionCode函数用于获取异常的类型。

●GetExceptionInformation函数用于获取异常的信息。

●RaiseException函数在调用线程中引发异常。

●GetLastError函数检索最后错误代码。

●若要访问消息字符串,需要使用 FormatMessage 函数。

时间处理:Windows操作系统提供了各种各样的时间处理函数,这些函数无疑都是在动态链接库kernel32.dll中的。例如:

●系统时间函数GetSystemTime检索 UTC 格式的当前系统日期和时间。

●SetSystemTime函数设置当前系统时间和日期。

●本地时间函数GetLocalTime检索当前本地日期和时间。

●用于处理文件时间的函数GetFileTime检索指定文件或目录的创建、上次访问和上次修改的日期和时间。

●FileTimeToLocalFileTime函数将 UTC 文件时间转换为本地文件时间。●FileTimeToSystemTime函数将文件时间转换为系统时间格式。

1.2.2 user32.dll

user32.dll主要负责处理用户图形界面,是Windows用户界面相关应用程序接口。user32.dll中常用的API函数如下:

窗口操作

●CreateWindowEx函数用指定方式创建一个窗口。

●CreateMDIWindow函数创建一个多文档界面窗口。

●DestroyWindow函数销毁指定的窗口。

●FindWindow函数从类名或窗口名中返回一个相匹配的顶层窗口的句柄。

●GetWindow函数返回指定窗口的句柄。

●RegisterClass函数为以后调用CreatWindow函数注册一个窗口类。

菜单操作

●CreateMenu函数创建一个菜单,然后用AppendMenu函数填充菜单项。

●DestroyMenu 函数销毁指定的菜单。

●GetMenuItemInfo函数返回有关菜单项的信息。

●AppendMenu函数在给定菜单的尾不增加新项。

对话框操作

●CreateDialotParam函数从对话框模板资源中创建一个无模式对话框。

●DialogBoxParam 函数从对话框模板资源中创建一个模式对话框。

●MessageBox函数创建、显示并操作一个消息框。

GUI对象操作

●CreateCursor函数用指定大小、位模式、热点创建一个光标。

●CreateIcon 函数用指定大小、颜色和位模式创建一个图标。

●CopyRect函数拷贝一个矩形坐标。

●GetScrollInfo函数返回滚动条的参数,包括最小/最大滚动位置,页大小及拇指框的位置。

●LoadImage函数装入一个图标、光标或位图。

●SetScrollRange函数设置滚动条最大或最小位置值。

提示

【注】Windows对象可以分为GUI对象、GDI对象和内核对象三类。

GUI对象包括窗口、客户窗口、滚动条、菜单、标题栏等用户图形界面中的对象。

GDI对象包括画笔、画刷、字体、位图、路径、调色板等绘图工具。

内核对象包括进程、线程、事件、信号量、互斥量等内核中创建的对象。

消息操作

●GetMessage函数从指定线程的消息队列中检取一条消息。

●PeekMessage函数检查应用程序的消息队列。

●PostMessage函数在指定的窗口消息队列中放置一条消息。

●SendMessage函数把一消息发送给指定的多个窗口。

●TranslateMessage函数把虚键消息翻译为字符消息。

●DispatchMessage函数传送一个消息给指定的窗口过程。

1.2.3 gdi32.dll

gdi32.dll动态链接库用于图形设备接口(GDI)功能的实现。GDI是一个图形接口,负责处理Windows系统中的图形和字体操作,如绘制图形、显示文本、打印等操作。GDI函数有一个非常明显的特征是函数的第一个参数一定是设备环境上下文句柄(跟踪设备上下文地址的指针)。gdi32.dl中常用的API函数如下:

GDI对象操作

●CreateBitmap 函数创建具有指定宽度、高度和颜色格式的位图, (颜色平面和每像素位) 。

●CreateBrushIndirect 函数创建具有指定样式、颜色和图案的逻辑画笔。

●CreatePen 函数创建具有指定样式、宽度和颜色的逻辑笔。 笔随后可以选择到设备上下文中,并用于绘制线条和曲线。

●CreateSolidBrush 函数创建具有指定纯色的逻辑画笔。

●CreateCompatibleDC 函数创建与指定设备兼容的内存设备上下文 (DC) 。

●CreateDIBitmap 函数从 DIB 创建兼容位图 (DDB) ,并选择性地设置位图位。

●CreateFont 函数创建具有指定特征的逻辑字体。 随后可以选择逻辑字体作为任何设备的字体。

●CreatePalette 函数创建逻辑调色板。

●GetDC 函数检索设备上下文的句柄, (指定窗口的工作区或整个屏幕的 DC) 。

●getObject 函数 (wingdi.h) 检索指定图形对象的信息。

绘图函数

●LineTo 函数从当前位置绘制一条线,但不包括指定点。

●MoveToEx 函数将当前位置更新为指定点,并根据需要返回上一个位置。

●Rectangle 函数绘制矩形。 矩形使用当前笔轮廓,并使用当前画笔填充。

●RoundRect 函数绘制带圆角的矩形。 矩形使用当前笔轮廓,并使用当前画笔填充。

●Ellipse 函数绘制椭圆形。 椭圆的中心是指定边框的中心。 椭圆形使用当前笔轮廓,并使用当前画笔填充。

●CreateEllipticRgn 函数创建椭圆区域。

●CreatePolygonRgn 函数创建多边形区域。

●CreateRectRgn 函数创建矩形区域。

●DrawText 函数在指定矩形中绘制格式化文本。

●TextOut 函数使用当前所选字体、背景色和文本颜色在指定位置写入字符串。

●FillPath 函数关闭当前路径中所有打开的图形,并使用当前画笔和多边形填充模式填充路径的内部。

●FillRect 函数使用指定的画笔填充矩形。 此函数包括左边框和上边框,但不包括矩形的右边框和下边框。

●FrameRect 函数使用指定的画笔在指定矩形周围绘制边框。 边框的宽度和高度始终是一个逻辑单元。

●FrameRgn 函数使用指定的画笔在指定区域周围绘制边框。

●GetDIBits 函数检索指定兼容位图的位,并使用指定格式将其作为 DIB 复制到缓冲区中。

●PlayMetaFile 函数显示存储在指定设备上的给定 Windows 格式图元文件中的图片。PolyDraw 函数绘制一组线段和 B zier 曲线。

●PolyBezierTo 函数绘制一条或多条 B zier 曲线。

●PolylineTo 函数绘制一条或多条直线。

●SetMapMode 函数设置指定设备上下文的映射模式。 映射模式定义用于将页面空间单位转换为设备空间单位的度量单位,还定义设备的 x 和 y 轴的方向。

●BitBlt 函数将对应于像素矩形的颜色数据从指定的源设备上下文传输到目标设备上下文中。

●BeginPaint 函数准备用于绘制的指定窗口,并使用有关绘制的信息填充 PAINTSTRUCT 结构。

打印输出函数

●StartDoc 函数启动打印作业。

●EndDoc 函数结束打印作业。

●EndPage 函数通知设备应用程序已完成对页面的写入。 此函数通常用于指示设备驱动程序转到新页面。

●StartPage 函数准备打印机驱动程序以接受数据。

●GetPrinter 函数检索有关指定打印机的信息。

●EnumPrinters 函数枚举可用的打印机、打印服务器、域或打印提供程序。

本文摘自编程达人系列教材《Windows API每日一练》。

这篇关于新书推荐:1.2 动态链接库与API的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1026431

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

usaco 1.2 Palindromic Squares(进制转化)

考察进制转化 注意一些细节就可以了 直接上代码: /*ID: who jayLANG: C++TASK: palsquare*/#include<stdio.h>int x[20],xlen,y[20],ylen,B;void change(int n){int m;m=n;xlen=0;while(m){x[++xlen]=m%B;m/=B;}m=n*n;ylen=0;whi

usaco 1.2 Name That Number(数字字母转化)

巧妙的利用code[b[0]-'A'] 将字符ABC...Z转换为数字 需要注意的是重新开一个数组 c [ ] 存储字符串 应人为的在末尾附上 ‘ \ 0 ’ 详见代码: /*ID: who jayLANG: C++TASK: namenum*/#include<stdio.h>#include<string.h>int main(){FILE *fin = fopen (

usaco 1.2 Milking Cows(类hash表)

第一种思路被卡了时间 到第二种思路的时候就觉得第一种思路太坑爹了 代码又长又臭还超时!! 第一种思路:我不知道为什么最后一组数据会被卡 超时超了0.2s左右 大概想法是 快排加一个遍历 先将开始时间按升序排好 然后开始遍历比较 1 若 下一个开始beg[i] 小于 tem_end 则说明本组数据与上组数据是在连续的一个区间 取max( ed[i],tem_end ) 2 反之 这个

usaco 1.2 Transformations(模拟)

我的做法就是一个一个情况枚举出来 注意计算公式: ( 变换后的矩阵记为C) 顺时针旋转90°:C[i] [j]=A[n-j-1] [i] (旋转180°和270° 可以多转几个九十度来推) 对称:C[i] [n-j-1]=A[i] [j] 代码有点长 。。。 /*ID: who jayLANG: C++TASK: transform*/#include<

防近视护眼台灯什么牌子好?五款防近视效果好的护眼台灯推荐

在家里,灯具是属于离不开的家具,每个大大小小的地方都需要的照亮,所以一盏好灯是必不可少的,每个发挥着作用。而护眼台灯就起了一个保护眼睛,预防近视的作用。可以保护我们在学习,阅读的时候提供一个合适的光线环境,保护我们的眼睛。防近视护眼台灯什么牌子好?那我们怎么选择一个优秀的护眼台灯也是很重要,才能起到最大的护眼效果。下面五款防近视效果好的护眼台灯推荐: 一:六个推荐防近视效果好的护眼台灯的

智能交通(二)——Spinger特刊推荐

特刊征稿 01  期刊名称: Autonomous Intelligent Systems  特刊名称: Understanding the Policy Shift  with the Digital Twins in Smart  Transportation and Mobility 截止时间: 开放提交:2024年1月20日 提交截止日

【LabVIEW学习篇 - 21】:DLL与API的调用

文章目录 DLL与API调用DLLAPIDLL的调用 DLL与API调用 LabVIEW虽然已经足够强大,但不同的语言在不同领域都有着自己的优势,为了强强联合,LabVIEW提供了强大的外部程序接口能力,包括DLL、CIN(C语言接口)、ActiveX、.NET、MATLAB等等。通过DLL可以使用户很方便地调用C、C++、C#、VB等编程语言写的程序以及windows自带的大

如何更优雅地对接第三方API

如何更优雅地对接第三方API 本文所有示例完整代码地址:https://github.com/yu-linfeng/BlogRepositories/tree/master/repositories/third 我们在日常开发过程中,有不少场景会对接第三方的API,例如第三方账号登录,第三方服务等等。第三方服务会提供API或者SDK,我依稀记得早些年Maven还没那么广泛使用,通常要对接第三方