将一个byte []从JNI返回给Java

时间:2011-05-16 08:14:32

标签: java android java-native-interface android-ndk

事实上,我有这个工作,只是不正确。我有一个回调(发布在上一个问题:Unable to get JNIEnv* value in arbitrary context),现在调用Java层中的回调...唯一的问题是回调返回的数据类型。在我的Java代码中,在调试时,我看到回调传递给Java的类型是class [B类型,而不是我期望的类型byte[]

Java回调唯一能做的就是将它放在队列中,但是当我需要处理该队列时会出现问题。

Java中的回调:

public void enqueueAudio(byte[] audioData){
    if(audioData != null){
        mWriteQueue.offer(audioData);
    }
}

处理队列:

private void writeToFile(String file){
    int totalNumOfBytes = 0;
    byte[] dataFromQueue = new byte[0];
    byte[] temp;
    for(byte[] data : mWriteQueue){
        temp = dataFromQueue;
        dataFromQueue = new byte[temp.length + data.length];
        System.arraycopy(temp, 0, dataFromQueue, 0, temp.length);
        System.arraycopy(data, 0, dataFromQueue, temp.length, data.length);
        totalNumOfBytes += data.length;
    }
    // Write the total byte[] to the specified file.
    mFileHandler.write(file, dataFromQueue);
    updateUI("Number of bytes written to " + file + " : " + totalNumOfBytes + "\n");
}

正如您所看到的,我假设Queue填充了byte[],而不是class [B,导致类强制转换异常......

那么,本机代码返回class [B而不是byte[]是否正确?

为了完整起见,这是C中的方法:

void recorderCallback(SLAndroidSimpleBufferQueueItf bq, void *context){
    SLresult result;
    JNIEnv* env;
    jbyteArray data;
    (*javaVM)->AttachCurrentThread(javaVM, &env, NULL);
    if(env == NULL){
        LOG_ERROR("Could not get JNIEnv*");
        return;
    }
    data = (*env)->NewByteArray(env, MAX_PACKET_SIZE);
    if(data == NULL){
        LOG_ERROR("No memory could be allocated for buffer");
        return;
    }
    (*env)->SetByteArrayRegion(env, data, 0, MAX_PACKET_SIZE, recorderBuffer);
    (*env)->CallByteMethodA(env, javaObject, javaCallbackMID, data);
    (*env)->DeleteLocalRef(env, data);
    result = (*bq)->Enqueue(bq, recorderBuffer,
                            RECORDER_FRAMES * sizeof(jbyte));
    checkError(result, "Unable to enqueue new buffer");
    (*javaVM)->DetachCurrentThread(javaVM);
}

1 个答案:

答案 0 :(得分:3)

  

事实上,我有这个工作,只是没有   正确。

这是一个矛盾。

  (*env)->CallByteMethodA(env, javaObject, javaCallbackMID, data);

这不是ByteMethod。这是一个VoidMethod。