上 期 我 们 介
绍 了 调 用WindowsAPI 函 数 的 方 法, 本 期 我 们 继 续 介 绍 几 个
常 用 的API 函 数。
Windows 操 作 系 统 在 象 文 件 操 纵 一 类 的 低 级 函 数 调 用 方 面 是 相 当 灵 活 的。 而 在PowerBuilder 中 为 了 实 现 这 一 些 低 级 操 作, 开 发 者 们 必 须 用 第 三 方 开 发 库( 如:FUNCkyforPowerBuilder 库) 或 用C 语 言 之 类 的 语 言 开 发 自 己 的 库, 这 种 方 法 我 们 将 在 后 面 的 章 节 中 讨 论。 在 这 里, 我 们 首 先 展 示 如 何 通 过 调 用WindowsAPI 函 数 实 现 简 单 而 又 常 用 的 低 级 操 作: 文 件 拷 贝。
使 用WindowsAPI 而
不 用 第 三 方 开 发 库 的 主 要 好 处 是, 如 果 这 是 您 所 要 的
唯 一 的 低 级 操 作, 在 发 布 您 的 应 用 时 就 不 需 要 同 时 将
额 外 的 库 打 包 进 您 的 应 用 中。 但 如 果 您 还 需 使 用 了 第
三 方 开 发 库 提 供 的 其 它 服 务, 为 简 单 起 见, 您 使 用 第
三 方 开 发 库 所 提 供 的 例 程 可 能 要 更 方 便 些。
Function long LZOpenFile(string FileName, REF ws_ofstruct FileStructure, uint style) Library "lzexpand.dll" Function long CopyLZFile(uint SourceHandle, uint DestHandle) Library "lzexpand.dll" Function long LZClose (uint FileHandle) Library "lzexpand.dll" ws_ofstruct的窗口级结构如下: Variable Name Data Type
byte[1] char
fixed_disk[1] char
errorcode uint
reserverd[4] char
pathname[128] char键入下面的一个窗口级函数: wf_copy // 返 回: boolean // 参 数: string as_source string as_dest long ll_SourceFileHandle, ll_DestFileHandle, ll_result ws_ofstruct lstr_SourceFileStructure, lstr_DestFileStructure // 打 开 源 文 件 ll_SourceFileHandle = LZOpenFile ( as_source, lstr_SourceFileStructure, 0 ) IF ll_SourceFileHandle = -1 THEN MessageBox ( " 程 序 错 误", " 无 法 打 开 文 件: " + as_source ) Return FALSE END IF // 打 开 目 标 文 件 ll_DestFileHandle = LZOpenFile ( as_dest, lstr_DestFileStructure, 4096 ) IF ll_DestFileHandle = -1 THEN MessageBox ( " 程 序 错 误", " 无 法 创 建 文 件: " + as_dest ) LZClose ( ll_SourceFileHandle ) Return FALSE END IF // 拷 贝 ll_result = CopyLZFile ( ll_SourceFileHandle, ll_DestFileHandle ) IF li_result <0 THEN MessageBox (" 程 序 错 误"," 无 法 拷 贝 文 件: " + as_source & + " to file: " + as_dest ) LZClose ( ll_SourceFileHandle ) LZClose ( ll_DestFileHandle ) Return FALSE END IF // 关 闭 文 件 LZClose ( ll_SourceFileHandle ) LZClose ( ll_DestFileHandle ) Return TRUE
Windows 所 带 的COMPRESS.EXE 是 一 个 处 理 采 用Lempel-ziv 算 法 压 缩 的 文 件 的 可 执 行 文 件。 此 外Windows 内 部 还 包 含 了 几 个 处 理 这 一 算 法 的 函 数, 其 中 之 一 是LzCopy 函 数, 该 函 数 可 以 用 来 生 成 解 压 缩 文 件。 同 时LzCopy 函 数 的 文 档 中 还 说 明 了:“ 如 果 原 文 件 未 被 压 缩, 该 函 数 复 制 原 文 件。” 于 是LzCopy 函 数 也 可 以 用 作 一 个 快 速 的 文 件 拷 贝 函 数。
LzCopy 函 数 有 两 个 参 数, 原 文 件 和 目 标 文 件 的 句 柄。 但 是 不 幸 的 是,PowerBuilder 提 供 的FileOpen 函 数 返 回 的 文 件 句 柄 并 不 是LzCopy 所 需 要 的 那 个 句 柄。 因 此, 如 果 调 用WindowsAPI 中 的LzCopy 函 数, 我 们 必 须 同 时 使 用LzOpenFile 和LzClose 函 数, 以 确 保 获 得 所 需 要 的 文 件 句 柄。
LzFileOpen 函 数
有 三 个 参 数。 第 一 个 参 数 是 将 要 打 开 的 文 件 名, 第 二
个 参 数 是 一 个 结 构, 这 个 结 构 将 在 函 数 调 用 时 填 入 有
关 这 个 文 件 的 信 息, 与PowerBuilder 的FileOpen 函 数 一 样,LzFileOpen
函 数 可 以 创 建 或 打 开 一 个 文 件。 该 函 数 的 第 三 个 参 数
指 明 该 函 数 是 被 用 来 以 只 读 方 式 打 开 文 件(0) 还 是 以 只
写 方 式 创 建 文 件(4096)。 当 我 们 打 开 了 源 文 件 和 目 标 文
件 后, 我 们 只 要 简 单 地 将 返 回 的 这 两 个 文 件 的 句 柄 传
递 给LzCopy 函 数, 就 可 实 现 文 件 拷 贝。 最 后, 在 退 出 之 前
调 用LzClose 函 数 来 关 闭 这 两 个 文 件。
实 现: 首 先
声 明 下 列localexternal 函 数
Function uint GetModuleHandle(string ModuleName) Library "kernel" Function uint GetModuleUsage(uint ModuleName) Library "kernel" Function uint FindExecutable(string FileName, & REF string Directory, REF string Result) Library "shell" 键入下面的一个窗口级函数: fw_check_app_status // 返 回: boolean // 参 数: // string as_filename // string as_directory integer li_size = 144, li_result string ls_executable = Space ( li_size ) integer li_module_usage uint li_module_handle li_result = FindExecutable ( as_filename, as_directory, ls_executable ) IF li_result <32 THEN Return FALSE //如可执行文件已载入,获取这个执行文件的句柄 li_module_handle="GetModuleHandle" ( ls_executable ) IF li_module_handle < 1 THEN // 没 有 载 入 Return FALSE ELSE // 获 取 有 多 少 个 实 例 被 载 入 li_module_usage="GetModuleUsage" ( li_module_handle ) Return TRUE END IF
这 段 程 序 可 以 获 取 某 个 应 用 是 否 正 在 运 行。
我 们 用windowsAPI 中 的GetModueHandle 函 数 来 判 定 是 否 另 一 个 应 用 正 在 运 行。 为 了 做 到 这 一 点, 我 们 需 要 知 道 文 件 的 全 路 径 名( 包 括 驱 动 器 名 和 路 径)。 我 们 可 以 将 它 包 含 在 应 用 的INI 文 件 中, 也 可 以 通 过 写 入 源 代 码 将 它 嵌 入 可 执 行 程 序。 这 里 向 大 家 推 荐 的 是 采 用 前 者 的 方 法, 这 样 作 可 使 应 用 具 有 更 多 的 灵 活 性。 在 上 面 的 窗 口 级 函 数 中, 我 们 首 先 在 用 户 的Windows 目 录 中 查 找win.ini 文 件。 因 为 对 于 我 们 需 要 进 行 查 询 的 应 用 与 特 定 扩 展 名 的 文 档 之 间 的 关 联 被 记 录 在WIN.INI 文 件 中。 于 是 我 们 所 要 做 的 第 一 件 事 就 是 在WIN.INI 文 档 中 查 找 这 一 关 联, 以 知 道 我 们 所 要 寻 找 应 用 是 从 通 过 哪 个 软 件 装 入 的。 我 们 可 以 自 己 搜 索WIN.INI 文 件, 或 者 调 用WindowsAPI 中 的FindExecutable 函 数 来 得 到 这 一 信 息。 这 里, 我 们 使 用 了 第 二 种 方 法。 我 们 只 要 将 与 我 们 所 希 望 寻 找 的 可 执 行 程 序 相 关 联 的 一 个 文 件 的 文 件 名 与 路 径 传 递 给FindExecutable 函 数, 就 会 得 到 该 可 执 行 程 序 的 全 路 径 名。 这 在 我 们 判 断 出 该 可 执 行 程 序 没 有 在 运 行 以 后 启 动 该 程 序 时 是 很 有 用 的。
一 旦 获 得 了 可 执 行 程 序 的 全 路 径 名, 我 们 将 它 传 递 给GefModuleHandle 函 数。 如 果 应 用 正 在 运 行, 则 该 函 数 返 回 应 用 的 句 柄。 如 果 返 回0, 说 明 应 用 并 没 有 装 入, 此 时 退 出 函 数。 如 果 应 用 正 在 运 行, 我 们 将 获 得 的 应 用 的 句 柄 传 递 给GetModuleUsage 函 数。 这 个 函 数 的 返 回 值 会 告 诉 我 们 这 个 应 用 有 几 个 不 同 的 实 例。 在 我 们 的 例 子 中, 如 查 询 的 应 用 是MicrosoftWord, 该 应 用 是 不 允 许 多 个 实 例 同 时 运 行, 但 象NotePad 这 样 的 应 用, 这 些 信 息 对 还 是 非 常 有 用 的。
激 活 另 一 个 应 用
实 现: 首 先
声 明 下 列localexternal 函 数
Subroutine BringWindowToTop(int hWnd) Library "User" Function uint GetNextWindow(uint hWnd, int Flag) Library "user" Function uint GetWindowText(uint hWnd, REF string Title, uint Size ) Library "user" Function uint ShowWindow(uint hWnd, int CmdShow) Library "user" Function integer IsIconic(uint hWnd) Library "user" Function integer OPenIcon(uint hWnd) Library "user" Function integer FindWindow(string ClassName, string WindowName) Library "user" Function integer FindWindow(long ClassName, string WindowName) Library "user" Function integer FindWindow(string ClassName, long WindowName) Library "user" 键入下面的一个窗口级函数: fw_make_app_active // 返 回:boolean // 参 数: //string as_title boolean lb_cont = TRUE integer li_result, li_size string ls_title uint li_handle li_handle = Handle ( this ) li_size = Len ( at_title ) + 1
//li_handle = FindWindow ( 0, "Control Panel" )
DO WHILE lb_cont li_handle = GetNextWindow ( li_handle, 1 ) If li_handle = Handle ( this ) THEN Return FALSE li_result = GetWindowText ( li_handle, ls_title, li_size ) IF ls_title = as_title THEN lb_cont = FALSE LOOP
//ShowWindow ( li_handle, 3 )
IF IsIconic ( li_handle ) > 0 THEN OpenIcon ( li_handle ) ELSE BringWindowToTop ( li_handle ) END IF Return TRUE
获 得 另 一 个 应 用 最 顶 层 窗 口 引 用 的 最 简 单 的 方 法 就 是 用FindWindow 函 数 来 查 找。FindWindow 函 数 有 两 个 参 数: 您 要 找 的 窗 口 的 类 名 和 窗 口 的 标 题。 注 意, 我 们 在localexternal 函 数 中 用 三 种 方 式 声 明 了 这 一 函 数。 这 是 因 为 您 可 以 用null 作 为“ 通 配 符” 分 别 传 递 两 个 参 数。 为 了 传 递null, 您 必 须 将 参 数 类 型 定 义 成long 型, 其 值 为0。 所 以 我 们 需 要 为 每 一 种 可 能 的 调 用 形 式 定 义 一 个localexternal 函 数。 当 我 们 真 正 使 用 这 个 函 数 时,PowerBuilder 自 动 判 断 我 们 要 用 哪 一 种 形 式。
例 如 在 窗 口 函 数fw_make_app_active 中, 我 们 可 以 用 这 种 手 段 使Findwindow 函 数 搜 索ControlPanel 窗 口。 接 下 来 只 要 将Findwindow 函 数 返 回 的 窗 口 句 柄 传 递 给BringWindowToTop 函 数 就 可 以 激 活ControlPanel 窗 口 了。
但 是 这 样 的 做 法 有 这 样 两 点 缺 陷:
您 并 不 总 是 知 道 某 个 应 用 的 顶 层 窗 口 的 全 名。 例 如 当 您 使 用MicosotWord, 当 前 打 开 的 文 档 的 名 字 也 将 出 现 在 窗 口 的 标 题 上。 因 此, 这 种 情 况 下, 寻 找 顶 层 窗 口 需 要 另 一 种 方 法。
如 果 您 希 望 激 活 的 窗 口 正 处 于 最 小 化 状 态,BringWindowToTop 函 数 将 不 能 正 常 工 作。
实 际 上, 上 面 提 到 的 第 二 个 问 题 比 较 容 易 解 决。 我 们 加 入IsIconic 函 数 判 断 我 们 所 要 激 活 的 窗 口 是 否 最 小 化。 如 果 不 是 最 小 化 状 态, 就 使 用BringWindowsToTop 函 数。 如 果 是 最 小 化 状 态, 就 用OpenIcon 函 数 来 激 活 该 窗 口。 这 种 方 法 只 假 定 我 们 想 以 窗 口 最 小 化 前 的 大 小 及 位 置 显 示 该 窗 口。 我 们 也 可 以 用ShoWindow 函 数 来 对 要 显 示 的 其 它 应 用 的 窗 口 施 以 更 多 的 控 制。
本 例 中, 我 们 传 送 一 个 参 数 使 被 激 活 窗 口 以 最 大 化 方 式 显 示。 如 果 您 想 以 不 同 的 方 式 激 活 窗 口, 请 参 照 下 表 的 参 数 对 应:
0
隐 藏 窗 口
1
以 窗 口 原 来 的 大 小 和 位 置( 如 果
当 前 窗 口 处 于 最 大 化 或 最 小 化 状 态) 并 激 活 该 窗 口( 与
值 相 同)
2
以 最 小 化 方 式 显 示 并 激 活 窗 口
3
以 最 大 化 方 式 显 示 并 激 活 窗 口
4
以 窗 口 最 近 一 次 的 大 小 和 位 置
显 示 但 不 激 活 窗 口
5
以 窗 口 最 近 一 次 的 大 小 和 位 置
显 示 并 激 活 窗 口
6
最 小 化 窗 口 并 在 系 统 列 表 中 激
活 顶 层 窗 口
7
最 小 化 窗 口 但 不 激 活 窗 口( 保 持
当 前 窗 口 激 活)
8
以 窗 口 的 当 前 状 态 显 示 但 并 不
激 活 它( 保 持 当 前 窗 口 激 活)
9
恢 复 窗 口 原 先 的 大 小 和 位 置( 如
果 窗 口 处 于 最 小 化 或 最 大 化 状 态) 并 激 活 它 解 决 第 二
个 问 题 时, 我 们 声 明 了 另 外 两 个 函 数,GetNextWindow 和GetWindowsText。
如 果 我 们 将 一 个 窗 口 句 柄 传 递 给GetNextWindow 函 数, 它 将
在 父 窗 口 所 属 的 所 有 子 窗 口 之 间 循 环。 如 果 将 一 个 顶
层 窗 口 句 柄 传 递 给 该 函 数, 该 函 数 将 在 所 有 顶 层 窗 口
之 间 循 环。 我 们 希 望 得 到 的 是 后 一 种 方 式, 所 以 我 们
首 先 要 获 取 自 己 应 用 的 顶 层 窗 口 句 柄。
当 我 们 在 顶
层 窗 口 之 间 循 环 时, 用GetWindowText 函 数 返 回 它 们 的 标 题。
这 个 函 数 有 三 个 参 数, 窗 口 句 柄, 一 个 用 以 返 回 标 题
的 字 符 串 以 及 返 回 字 符 串 的 最 大 长 度。 我 们 要 好 好 利
用 最 后 一 个 参 数, 因 为 我 们 并 不 想 要 匹 配 整 个 窗 口 标
题, 而 是 要 匹 配 能 保 证 我 们 发 现 所 要 寻 找 的 窗 口 的 长
度。 例 如, 如 果 我 们 将“MicrosoftWord” 传 递 给 这 个 窗 口 函
数 时, 这 个 函 数 判 断 文 本 的 长 度 并 将 它 加1, 这 样 只 将
我 们 将 要 比 较 的 串 的 长 度 传 递 给 了GetWindowText 函 数, 该
函 数 在 顶 层 窗 口 之 间 循 环, 寻 找 以 该 字 符 串 开 头 的 窗
口。 如 果 我 们 发 现 了 一 个 匹 配, 就 设 置 结 束 循 环 标
志, 退 出 循 环 并 继 续 执 行。 如 果 我 们 没 有 发 现 匹 配,
将 在 循 环 到 我 们 自 己 的 应 用 窗 口 时 退 出 循 环。 这 时,
我 们 简 单 地 退 出 函 数。