0

0

获取 Windows 上所有可见窗口的完整指南(基于 JNA)

霞舞

霞舞

发布时间:2026-01-06 14:33:00

|

523人浏览过

|

来源于php中文网

原创

获取 Windows 上所有可见窗口的完整指南(基于 JNA)

本文详解如何使用 jna 高效获取当前系统中所有真正对用户可见的窗口(非仅 `ws_visible` 样式),涵盖 `windowutils.getallwindows()` 的正确用法、可见性判断的深层逻辑、z-order 与遮挡检测策略,并提供生产级实践建议。

在 Java 中开发桌面覆盖层(overlay)时,实时跟踪目标窗口的位置与可见状态是核心需求。你已正确选用 JNA 与 Windows API 交互,但需注意:User32.IsWindowVisible(HWND) 仅检查窗口及其祖先是否设置了 WS_VISIBLE 样式,而非用户实际能否看到该窗口——即使窗口被完全遮挡、最小化或坐标为 (0,0,0,0),该函数仍可能返回 true。

✅ 正确获取“用户可见”窗口的方法

JNA 官方平台库(jna-platform)已封装更实用的接口:

import com.sun.jna.platform.WindowUtils;
import com.sun.jna.platform.DesktopWindow;

// 获取所有「逻辑可见」窗口(即 IsWindowVisible == true)
List allVisibleWindows = WindowUtils.getAllWindows(true);

// 获取全部窗口(含隐藏/不可见)
List allWindows = WindowUtils.getAllWindows(false);

每个 DesktopWindow 实例包含关键信息:

  • hwnd: 原生窗口句柄(可用于后续 GetWindowRect、IsIconic 等调用)
  • title: 窗口标题(UTF-8 安全,已自动处理空指针)
  • locAndSize: java.awt.Rectangle,表示屏幕坐标系下的位置与尺寸(x, y, width, height)
  • filePath: 可选,进程可执行文件路径(需管理员权限,慎用)

⚠️ 注意:getAllWindows(true) 仍不等价于“用户当前可见”。它仅过滤掉 IsWindowVisible == false 的窗口(如任务栏、系统托盘、隐藏工具窗口),但无法识别以下情况:

蛙蛙写作——超级AI智能写作助手
蛙蛙写作——超级AI智能写作助手

蛙蛙写作辅助AI写文,帮助获取创意灵感,提供拆书、小说转剧本、视频生成等功能,是一款功能全面的AI智能写作工具。

下载
  • 窗口被其他窗口完全覆盖
  • 窗口已最小化(IsIconic(hwnd) == true)
  • 窗口位于屏幕外(如多显示器环境中的负坐标)

? 进阶:判断“用户实际可见性”

要实现 overlay 的智能显隐,需结合 Z-order 与几何分析。推荐分两步:

1. 排除明显不可见状态

import com.sun.jna.platform.win32.User32;
import com.sun.jna.platform.win32.WinDef.HWND;

public static boolean isTrulyVisible(HWND hwnd) {
    if (hwnd == null || hwnd.equals(User32.INSTANCE.GetDesktopWindow())) 
        return false;

    // 检查是否最小化或无有效矩形
    Rectangle rect = getWindowRect(hwnd); // 自定义封装,调用 GetWindowRect
    if (rect.width <= 0 || rect.height <= 0 || 
        User32.INSTANCE.IsIconic(hwnd)) 
        return false;

    // 检查 WS_VISIBLE 样式(基础过滤)
    if (!User32.INSTANCE.IsWindowVisible(hwnd)) 
        return false;

    return true;
}

2. Z-order 遮挡检测(轻量级方案)

利用 GetWindow(hwnd, GW_HWNDPREV) 或 EnumWindows + GetWindowPlacement 获取窗口层级,再对比 locAndSize 重叠区域。不建议逐像素检测(性能差),推荐角点检测法:

public static boolean isPartiallyVisible(HWND hwnd, List topWindows) {
    Rectangle target = getWindowRect(hwnd);
    if (target == null) return false;

    // 只检查 Z-order 在其上方的窗口(按 EnumWindows 逆序或 OSHI 的 order 字段)
    for (DesktopWindow top : topWindows) {
        if (top.getHWnd().equals(hwnd)) continue;
        if (top.getOrder() <= getZOrder(hwnd)) continue; // 仅比目标层级高的窗口

        Rectangle topRect = top.getLocAndSize();
        if (target.intersects(topRect)) {
            // 计算重叠面积占比,若 >95% 被覆盖则视为不可见
            Rectangle overlap = target.intersection(topRect);
            double coverage = (double) overlap.width * overlap.height 
                            / (target.width * target.height);
            if (coverage > 0.95) return false;
        }
    }
    return true;
}
? 提示:OSHI 库(new SystemInfo().getOperatingSystem().getDesktopWindows())返回的 DesktopWindow 已含 order 字段(Z-order 降序),且跨平台兼容,适合生产环境直接集成。

⚙️ 性能优化建议

  • 采样频率控制:避免高频轮询(如
  • 缓存窗口句柄:对目标窗口 FindWindow 后缓存 HWND,避免重复字符串匹配开销。
  • 增量更新:首次全量枚举后,后续仅监听 EVENT_OBJECT_CREATE/DESTROY(通过 AccessibleObjectFromEvent 或 SetWinEventHook),减少 EnumWindows 调用。
  • 线程安全:getAllWindows() 是线程安全的,但结果列表需在 Swing/AWT 线程中更新 UI(使用 SwingUtilities.invokeLater)。

✅ 总结

方法 是否满足“用户可见” 适用场景 备注
User32.IsWindowVisible() ❌ 仅样式可见 快速排除隐藏窗口 易误判(遮挡/最小化)
WindowUtils.getAllWindows(true) ⚠️ 半满足 批量获取候选窗口 需二次过滤最小化与遮挡
OSHI.getDesktopWindows() ✅ 推荐 生产级应用 含 Z-order、进程 ID、跨平台
自定义 Z-order + 几何分析 ✅ 精确控制 Overlay 智能显隐 平衡精度与性能

最终,一个健壮的 overlay 跟踪逻辑应组合:句柄缓存 → 样式可见性初筛 → 最小化状态校验 → Z-order 层级排序 → 关键角点重叠判定。这样既规避了 IsWindowVisible 的语义陷阱,又在性能与准确性间取得最佳平衡。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

829

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

733

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

733

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

Golang 分布式缓存与高可用架构
Golang 分布式缓存与高可用架构

本专题系统讲解 Golang 在分布式缓存与高可用系统中的应用,涵盖缓存设计原理、Redis/Etcd集成、数据一致性与过期策略、分布式锁、缓存穿透/雪崩/击穿解决方案,以及高可用架构设计。通过实战案例,帮助开发者掌握 如何使用 Go 构建稳定、高性能的分布式缓存系统,提升大型系统的响应速度与可靠性。

27

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.4万人学习

C# 教程
C# 教程

共94课时 | 6.3万人学习

Java 教程
Java 教程

共578课时 | 43.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号