近日我作为一名投产支持人员参与到项目投产中,在经历了一夜紧张而疲惫的投产和验证后,终于迎来了系统正式上线使用的一天,随着使用者逐渐增加,群里随之而来的是各种问题和操作上的反馈,群消息让人应接不暇,作为支持人员的我,立马打起精神,开始查问题,为客户解答,一扫熬了一夜的疲态,废话不多说,直接进入正题。
反馈中有一个问题,登录到我们自己的c/s应用(简称:A)后,接着要嵌入另外一个c/s系统(简称:B),我们采用C语言的窗体操作,将一个窗体嵌入到另一个里,但是投产后发现部分终端A登录后,B虽然调用并打开了,但没有嵌入到A里去,导致用户使用很不方便,虽说不影响正常功能,但是A作为一个前端系统,这样的问题还是很影响的。
这类系统集成我在其他地方也做过,于是领导就让我处理这个问题,先熟悉这个项目的集成代码,发现在嵌入代码中,有一个嵌入延时处理时间,设置的是3秒,在测试环境下,这个3秒没有问题,但投产后,面对各类环境以及网络延迟等,明显3秒设置是不合适的,将这个时间调大,虽然可以解决一部分问题,但毕竟不是长久之计,难免会有B系统加载超过这个延迟时间,于是我就在想,如何能判断这个B加载完成后,再进行嵌入,对于这类窗体的操作,最好的方法是用windows自带的功能来处理,而我们的系统是采用Java语言开发的,对于windows的操作就需要JNI来做,这里我想到了SWT,在SWT中提供了丰富的windows调用。
SWT版本为:swt_3.7.0
首先说下思路:
1、去掉延时时间
2、增加根据窗口标题或者ClassName查找B系统窗口句柄
B系统是一个类似于eclipse的应用,窗口的ClassName是SWT_Window0,利用Spy++发现存在多个SWT_Window0,于是我想到通过windows方法将SWT_Window0相关的窗口全部查出,然后遍历,根据窗口标题锁定B窗口句柄。
查阅了一遍Windows的窗口查找和遍历,发现有几个方法
拿着这几个方法进行验证时发现SWT没有EnumWindows和FindWindowEx,无奈只有FindWindow方法,于是针对FindWindow方法开始研究,这个方法呢,有两个参数,第一个参数是ClassName,第二个参数是窗口名称,研究后发现,这个方法无法做到返回所有相关的句柄,窗口名称不支持模糊查询(因为B系统的窗口标题每次登陆可能都不一样),className又存在多个,无法准确获取,最后只能放弃FindWindow尝试。
最后剩下GetWindow,紧急在网上冲了一会儿浪,最后在这个网址里获得了启发,连接如下
https://zhidao.baidu.com/question/560484903012685964.html
废话不多说,上代码
/**
* 模糊查找
* 根据关键字获取对应窗口句柄
* @param key 窗口标题
* @return
*/
public int getHWND(String key) {
int desktopWindow = OS.GetDesktopWindow();
int hwnd = OS.GetWindow(desktopWindow, OS.GW_CHILD);
String windowName = null;
int windowTextLen = 200;
TCHAR winNameTChar = null;
try {
while(hwnd != 0) {
winNameTChar = new TCHAR(0, windowTextLen);
OS.GetWindowText(hwnd, winNameTChar, windowTextLen);
windowName = winNameTChar.toString().trim();
if(windowName.startsWith(key)) {
break;
}
hwnd = OS.GetWindow(hwnd, OS.GW_HWNDNEXT);
}
} catch (Throwable e) {
e.printStackTrace();
}
return hwnd;
}
如果返回值为0,利用循环加延时,可以多次调用,直到返回值大于0,但也不能无限循环调用,适当增加重试次数或者超时时间,保证程序的健壮性。
EnumWindows
枚举屏幕上的所有的顶层窗口,轮流地将这些窗口的句柄传递给一个应用程序定义的回调函数。EnumWindows会一直进行下去,直到枚举完所有的顶层窗口,或者回调函数返回了FALSE. ↩︎FindWindow
检索处理顶级窗口的类名和窗口名称匹配指定的字符串。这个函数不搜索子窗口. ↩︎FindWindowEx
该函数获得一个窗口的句柄,该窗口的类名和窗口名与给定的字符串相匹配。这个函数查找子窗口,从排在给定的子窗口后面的下一个子窗口开始。在查找时不区分大小写. ↩︎GetWindow
该函数返回与指定窗口有特定关系(如Z序或所有者)的窗口句柄,如果与指定窗口有特定关系的窗口不存在,则返回值为NULL。 ↩︎