0

私は現在、Tox チャット ライブラリの JNI ラッパーを作成しています。多くのことは、コールバックを通じて処理されます (メッセージの受信、友達リクエストなど)。これらを処理するために、私はすでに機能するものを実装しました(今まで、なぜ機能しなくなったのかわかりません):

メイン クラスでは、次のようにコールバックを設定できるメソッドを定義します。

    /**
 * Native call to tox_callback_friendrequest
 * 
 * @param messengerPointer
 *            pointer to the internal messenger struct
 * @param callback
 *            the callback to set for receiving friend requests
 */
private native void tox_onfriendrequest(long messengerPointer,
        OnFriendRequestCallback callback);

/**
 * Method used to set a callback method for receiving friend requests. Any
 * time a friend request is received on this Tox instance, the
 * {@link OnFriendRequestCallback#execute(String, String)} method will be
 * executed.
 * 
 * @param callback
 *            the callback to set for receiving friend requests
 * @throws ToxException
 *             if the instance has been killed
 */
public void setOnFriendRequestCallback(OnFriendRequestCallback callback)
        throws ToxException {
    lock.lock();
    try {
        checkPointer();

        tox_onfriendrequest(this.messengerPointer, callback);
    } finally {
        lock.unlock();
    }
}

パブリック Java メソッドは、次のネイティブ コードを呼び出すだけです。


JNIEXPORT void JNICALL Java_im_tox_jtoxcore_JTox_tox_1onfriendrequest(
        JNIEnv * env, jobject obj, jlong messenger, jobject callback) {
    tox_jni_globals_t *_messenger = (tox_jni_globals_t *) messenger;
    if (_messenger->frqc) {
        if (_messenger->frqc->jobj) {
            (*env)->DeleteGlobalRef(env, _messenger->frqc->jobj);
        }
        free(_messenger->frqc);
    }

    friendrequest_callback_t *data = malloc(sizeof(friendrequest_callback_t));
    data->env = env;
    data->jobj = (*env)->NewGlobalRef(env, callback);
    (*env)->DeleteLocalRef(env, callback);
    _messenger->frqc = data;
    tox_callback_friendrequest(_messenger->tox, (void *) callback_friendrequest,
            data);
}

static void callback_friendrequest(uint8_t *pubkey, uint8_t *message,
        uint16_t length, void *ptr) {
    friendrequest_callback_t *data = ptr;

    jclass clazz = (*data->env)->GetObjectClass(data->env, data->jobj);
    jmethodID meth = (*data->env)->GetMethodID(data->env, clazz, "execute",
            "(Ljava/lang/String;[B)V");

    char buf[ADDR_SIZE_HEX] = { 0 };
    addr_to_hex(pubkey, buf);
    jstring _pubkey = (*data->env)->NewStringUTF(data->env, buf);
    jbyteArray _message = (*data->env)->NewByteArray(data->env, length);
    (*data->env)->SetByteArrayRegion(data->env, _message, 0, length, message);

    printf("definitely happening"); //printf debugging. This IS being printed.
    fflush(stdout);
    (*data->env)->CallVoidMethod(data->env, data->jobj, meth, _pubkey, message); //this does not seem to be called
}

このコードはかなり複雑ですが、次のように動作します。まず、このアクションにコールバックが設定されているかどうかを確認し、設定されている場合は削除します。次に、受信した Callback オブジェクトへの新しいグローバル参照を作成し、その参照と JNIEnv ポインターをメッセンジャー構造体に保存します。最後に、Tox-API によって提供されるコールバック メソッドを呼び出し、コールバックを登録します。static void callback_friendrequest

opaque void pointerptrに、JNIEnv と Callback オブジェクトを格納します。コールバック クラスは次のようになります。

package im.tox.jtoxcore.test;

import im.tox.jtoxcore.JTox;
import im.tox.jtoxcore.ToxException;
import im.tox.jtoxcore.callbacks.OnFriendRequestCallback;

public class TestOnFriendRequestCallback extends OnFriendRequestCallback {

    public TestOnFriendRequestCallback(JTox jtox) {
        super(jtox);
    }

    @Override
    public void execute(String publicKey, byte[] message) {
        String msg;
        if (message == null) {
            msg = "No mesage";
        } else {
            try {
                msg = JTox.getByteString(message);
            } catch (ToxException e) {
                msg = "ERROR: Unable to get message";
            }
        }
        System.out.println("We received a friend request from " + publicKey
                + " with the following message: " + msg);

        try {
            jtox.confirmRequest(publicKey);
            System.out.println("confirmed!");
        } catch (ToxException e) {
            System.out.println("Unable to confirm friend request");
        }
    }
}

今、別のクライアントを介して友達リクエストを送信すると、コールバックを登録した後、「間違いなく起こっている」と出力されます。これは、私が導入したデバッグ ステートメントです。CallVoidMetod 呼び出しは実行されません。これは、FriendRequest を受信したことも確認されたことも明らかに出力されていないためです。

私にとって、これはコールバックが正常に登録されたことを意味しますが、何らかの理由で Java メソッドが呼び出されません。

完全なコードは GitHub で入手できます: https://github.com/Tox/jToxcore

4

1 に答える 1

1

これが失敗する理由は、data という名前のオブジェクト内に Java env ポインターを格納しているためだと思われます。env ポインターはこの方法でキャッシュすることはできず、コールバックが呼び出されたときに無効になる可能性があります。これに対処する方法の詳細については、この質問を参照してください。

于 2013-09-19T17:09:30.317 に答える