【问题标题】:How do I modify a window class without a window handle?如何修改没有窗口句柄的窗口类?
【发布时间】:2025-11-30 22:10:01
【问题描述】:

我目前正在尝试使用 JNI 将一些 WIN32API 函数转换为 Java。其中一个功能是RegisterClassEx。通常,我认为您会为注册的每个窗口类指定不同的回调函数,但由于我也将回调转换为 Java,这不起作用。

所以目前的计划是将jobject(定义为_jobject*)附加到窗口类并在回调中使用它。问题是:您只能使用 HWND 更改附加到窗口类的数据。据我所知,MSDN 文档没有指定可以仅使用窗口类 ATOM 或名称修改窗口类的函数。

因此我的问题是:有没有办法更改窗口类(使用类似SetClassLongPtr),而不必使用有效的 HWND?

Java 方面(我最终会添加一个公共函数来完成我真正需要做的事情):

public class c_winjni implements i_jnisystem {
    public interface i_wnd_proc {
        public int wnd_proc(long il_hwnd, int im_message, long im_wparam, long im_lparam);
    }

    private class c_wndclassex {
        public int im_style = 0;
        public i_wnd_proc ds_wnd_proc = null;
        public int im_cls_extra = 0;
        public int im_wnd_extra = 0;
        public long il_instance = 0L;
        public long il_icon = 0L;
        public long il_small_icon = 0L;
        public long il_cursor = 0L;
        public long il_background = 0L;
        public String str_menu_name = null;
        public String str_class_name = null;
    }

    private static native short registerClassEx(c_wndclassex ds_wcx, int[] imr_error);
}

C++ 方面

LRESULT CALLBACK default_window_callback_proc(HWND ds_hwnd, UINT im_message, WPARAM im_w_param, LPARAM im_l_param) {
    return DefWindowProc(ds_hwnd, im_message, im_w_param, im_l_param);
}

/*
 * Class:     c_winjni
 * Method:    registerClassEx
 * Signature: (Lc_winjni/c_wndclassex;[I)S
 */
JNIEXPORT_EX jshort JNICALL Java_c_1winjni_registerClassEx
    (JNIEnv *ads_env, jclass /*jds_class*/, jobject jds_wcx, jintArray jimr_error)
JNI_CPPEXCEPTION_TRAP_BEGIN {
    c_jnienv jds_env(ads_env);

    jint *aim_error = NULL;
    if (jimr_error && jds_env.get_array_length(jimr_error) > 0) {
        aim_error = jds_env.get_array_elements(jimr_error, NULL);
    }

    WNDCLASSEX ds_wcx;
    ds_wcx.cbSize = sizeof(WNDCLASSEX);
    ds_wcx.style = jds_env.get_int_field(jds_wcx, "im_style");

    // Imagine I'm checking whether field ds_wnd_proc in object jds_wcx is null.
    // If it is, use the default callback (as shown below).
    // If it isn't, set ds_wcx.lpfnWndProc to some other callback that reads
    // custom class data and calls a Java function of the object attached to the window class.
    ds_wcx.lpfnWndProc = default_window_callback_proc;

    ds_wcx.cbClsExtra = jds_env.get_int_field(jds_wcx, "im_cls_extra") + sizeof(LONG_PTR);
    ds_wcx.cbWndExtra = jds_env.get_int_field(jds_wcx, "im_wnd_extra");
    ds_wcx.hInstance = (HINSTANCE) jds_env.get_long_field(jds_wcx, "il_instance");
    ds_wcx.hIcon = (HICON) jds_env.get_long_field(jds_wcx, "il_icon");
    ds_wcx.hIconSm = (HICON) jds_env.get_long_field(jds_wcx, "il_small_icon");
    ds_wcx.hCursor = (HCURSOR) jds_env.get_long_field(jds_wcx, "il_cursor");
    ds_wcx.hbrBackground = (HBRUSH) jds_env.get_long_field(jds_wcx, "il_background");
    ct_jstring<TCHAR, 256> str_menu_name(ads_env, (jstring) jds_env.get_string_field(jds_wcx, "str_menu_name"));
    ds_wcx.lpszMenuName = str_menu_name.get_data();
    ct_jstring<TCHAR, 256> str_class_name(ads_env, (jstring) jds_env.get_string_field(jds_wcx, "str_class_name"));
    ds_wcx.lpszClassName = str_class_name.get_data();

    jshort result = RegisterClassEx(&ds_wcx);
    if (result == NULL && aim_error) {
        *aim_error = GetLastError();
    }

    // commit changes and invalidate pointer
    if (aim_error) {
        jds_env.release_array_elements(jimr_error, aim_error, 0);
    }
    return result;
} JNI_CPPEXCEPTION_TRAP_END2(ads_env, 0)

【问题讨论】:

  • 您要做的是实现我所知道的每个好的 OOP 包装器到 Win32 使用的解决方案。但是您需要负责拨打RegisterClassEx。您需要在cbWndExtra 中保留一些字节来存储您的对象引用。然后您可以将公共窗口过程路由到该对象。
  • 也许我没有说清楚我想要实现的目标:我想使用 JNI 将 RegisterClassEx 映射到一些 Java 函数。 WNDCLASSEX 包含一个 WndProc 成员,因此 per class 有一个回调函数,而不是每个窗口都有一个回调函数(这就是 cbWndExtra 的用途)。所以我使用cbClsExtra 将自定义数据添加到窗口类,只有在没有该类的窗口的情况下无法更改 per class 数据(就像试图访问应该是静态成员,只是为了找出它的正常字段,保证每个实例都相同)。
  • 为什么要将回调关联到类?这对我来说毫无意义。你想要一个实例方法。
  • @David 如果我正在创建一个抽象,是的,但我真的只对在 C++ 和 Java 之间进行转换感兴趣。

标签: c++ winapi


【解决方案1】:

您的问题的简单答案是只能使用SetClassLongPtr 修改额外的类内存。您需要一个有效的窗口句柄。

【讨论】:

    【解决方案2】:

    有趣的问题... WNDCLASSEX 是一个类,而 HWND 是一个窗口句柄。您可能将第一个视为 OO 类,第二个视为“指针”或“引用”对象(实例)。 然而要修改这个类,你似乎必须通过那个类的一个实例......奇怪:)我可以想象这是最常见的场景(有 HWND)(顺便说一句,为什么你没有 HWND? )

    一个想法:使用 ATOM 创建一个窗口(隐藏)并使用返回的 HWND 作为SetClassLongPtr 的“参考”是否可以接受?

    【讨论】:

    • 据我所知,这会起作用。对于应该相当容易做的事情来说,这似乎是一个很大的开销。
    • @Deaod 同意了,但是为什么你一开始就没有 HWND?怎么可能只有一个 ATOM 类?
    • 我正在尝试将 WIN32API 函数尽可能直接映射到 JNI 函数,以使 c++ 层尽可能平坦。据我所知,您需要一个类才能使用它创建一个窗口。
    • @Deaod 我不明白的是为什么您要使用通用函数注册一个窗口类,然后您需要更改它(或窗口类的另一部分)。你不能推迟改变直到你真正有一个窗口?还是不能有 1 个类 - 1 个窗口 - 1 个 wndproc 关系? (查看您的 API 以及您希望如何使用它可能会有所帮助!)
    • 我不会在以后尝试更改某些内容。在创建第一个窗口之前,我正在尝试正确设置课程。我将在我的问题中添加更多代码。
    【解决方案3】:

    要获得注册类 ATOM 的有效 HWND,您可以使用FindWindow function

    通过将NULL 传递给lpWindowName 参数,您将获得与该类匹配的第一个窗口的句柄。当然,如果没有窗口实例存在,您将失败。但在此,最好假设相应的窗口类尚未注册。

    【讨论】: