【发布时间】:2011-03-04 12:49:37
【问题描述】:
我正在尝试将 Java 中的 C++ API 与 JNA 一起使用。此 API 使用回调来处理会话事件。
我找到的关于如何使用 JNA 注册回调的唯一资源是 this,它处理 C 回调,我真的不知道如何将它扩展到 C++ 非静态回调。
编辑:我刚刚找到this resource,我认为“重访回调”一章可能会有所帮助。
回调的所有函数指针都存储在以下sp_session_callbacks结构中:
/**
* Session callbacks
*
* Registered when you create a session.
* If some callbacks should not be of interest, set them to NULL.
*/
typedef struct sp_session_callbacks {
void (__stdcall *logged_in)(sp_session *session, sp_error error);
void (__stdcall *logged_out)(sp_session *session);
void (__stdcall *connection_error)(sp_session *session, sp_error error);
void (__stdcall *message_to_user)(sp_session *session, const char *message);
// Other callbacks function pointers
} sp_session_callbacks;
我为描述这种结构而创建的 Java 类如下:
public class sp_session_callbacks extends Structure{
public Function logged_in;
public Function logged_out;
public Function connection_error;
public Function message_to_user;
}
您认为在这种情况下用 com.sun.jna.Function 对象表示函数指针是否有意义?
每个会话都由一个 sp_session 对象表示,它是一个 C++ 不透明结构。我在初始化 sp_session_callbacks 对象时确实有一个句柄。
这是我的主要课程中的代码 sn-p:
JLibspotify lib = (JLibspotify)Native.loadLibrary("libspotify", JLibspotify.class);
sp_session_config cfg = new sp_session_config();
/* Some cfg config here */
sp_session_callbacks sessCallbacks = new sp_session_callbacks(); // Handle on my sp_session_callbacks object
cfg.callbacks = sessCallbacks;
PointerByReference sessionPbr = new PointerByReference();
int errorId = lib.sessionCreate(cfg, sessionPbr);
sp_session session = new sp_session(sessionPbr.getValue()); // handle on my sp_session object
我应该如何注册这些回调函数以在它们被触发时在 Java 端实际执行某些操作?
谢谢!
编辑
使用回调而不是函数的新代码:
public class sp_session_callbacks extends Structure{
public LoggedIn logged_in;
/* Other callbacks... */
}
public interface LoggedIn extends StdCallCallback {
public void logged_in(sp_session session, int error);
}
主类:
JLibspotify lib = (JLibspotify)Native.loadLibrary("libspotify", JLibspotify.class);
sp_session_config cfg = new sp_session_config();
/* Some cfg config here */
sp_session_callbacks sessCallbacks = new sp_session_callbacks(); // Handle on my sp_session_callbacks object
LoggedIn loggedInCallback = new LoggedIn(){
public void logged_in(sp_session session, int error){
System.out.println("It works");
}
};
sessCallbacks.logged_in = loggedInCallback;
/* Setting all the other callbacks to null */
cfg.callbacks = sessCallbacks;
PointerByReference sessionPbr = new PointerByReference();
int errorId = lib.sessionCreate(cfg, sessionPbr);
sp_session session = new sp_session(sessionPbr.getValue()); // handle on my sp_session object
当 cfg.logged_in 未设置为 null 而是设置为 LoggedIn 实例时,sessionCreate() 调用将引发 JRE 致命错误 (EXCEPTION_ACCES_VIOLATION 0x0000005)。奇怪的是,logged_in 和 connection-error 这两个回调的签名相同,而且当设置了 cfg.connection_error 时,它不会抛出任何东西。
【问题讨论】:
-
您的示例中的所有函数指针都是 c,没有 c++ 功能可以与 jna 混淆。
-
@josefx 我认为将元素作为结构的非静态字段(这是一个面向对象的特性)这一事实是 C++ 的典型特征。
-
您使用的结构是标准 c 的一部分。我不明白您对非静态字段的意思,只有静态字段的结构将毫无用处。(再说一次,我主要是 java 程序员,所以我可能会混淆静态的含义)
-
@josefx 好的。我也是一名 Java 程序员,对 C/C++ 了解不多,这就是我问的原因。我的意思是非静态字段,是一个可以为结构的 2 个不同实例保存不同值/对象的字段。我虽然这个概念是由 OOP 带来的,所以我认为在不是 OO 语言的 C 中是不可能的,但我没有把握。
标签: java c++ pointers callback jna