0

我在 windows 上做了一个 Opencv 的应用程序,现在我正在使用 JNI 将此代码转换为 Android,但我遇到了一些问题。具体来说,我的本机代码什么也不做。

这是我定义本机方法的 Java 类:

package com.example.telo3;

import org.opencv.core.Mat;

public class Process {

    static {
        System.loadLibrary("nativo");
    }

    public Process(){

        dir=inicializar_nativo();
    }

    public void Procesar(Mat framedetect, Mat framedraw){

        procesar_nativo(dir,framedetect.getNativeObjAddr(),framedraw.getNativeObjAddr());
    }


    private long dir;
    private static native long inicializar_nativo();
    private static native void procesar_nativo(long thiz, long framedetect, long framedraw);

}

这是我的 JNI 代码:

#include "nativo.h"
#include <opencv2/objdetect/objdetect.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/imgproc/imgproc.hpp>
#include "opencv2/video/tracking.hpp"

#include <iostream>
#include <stdio.h>
#include "FaceDetector.h"
#include "Draw.h"
#include "Almacena.h"
#include "Runnable.h"



using namespace std;
using namespace cv;

#include <android/log.h>

#define LOG_TAG "NATIVO"
#define LOGD(...) ((void)__android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__))

struct variables {
    Almacena almacena;
    Draw draw;
    FaceDetector face_detector;
};

JNIEXPORT jlong JNICALL Java_com_example_telo3_Process_inicializar_1nativo(
        JNIEnv *, jobject) {

    long dir = (long) new variables();

    return (dir);


}

JNIEXPORT void JNICALL Java_com_example_telo3_Process_procesar_1nativo(JNIEnv *,
        jobject, jlong dir, jlong framedetect, jlong framedraw) {



Mat* telo =(Mat*)framedetect;
Mat* telo2= (Mat*)framedraw;

((variables*)dir)->almacena = ((variables*)dir)->face_detector.Detect(*telo);


 //almacena = face_detector.Detect(frame_gray);


 ((variables*)dir)->draw.Dibujar(*telo2,((variables*)dir)->almacena);


 //frame_capturado = draw.Dibujar(frame_capturado, almacena);



if( (((variables*)dir)->almacena.get_faces()).size() ==0){

    LOGD("no detecto caras");
}


}

我认为我正确使用了 Jni,但函数 Detect 无法正常工作,因为当我使用它时,如果返回 0。

4

1 回答 1

0

framedetect0吗?我在此基础上制作了一个测试应用程序,它运行良好(也就是说,jlong​​ 被转换为和从它转换得很好,所以这不应该是问题)。

尝试捕捉错误ndk-gdb以更好地理解问题。将它附加到进程可能会出现问题,但如果它立即崩溃,在这种情况下,我喜欢在崩溃之前将断点放在 java 端,使用调试器使执行暂停,附加ndk-gdb并继续让 java 进程继续。

我还建议使用reinterpret_cast<jlong>andreinterpret_cast<variables*>而不是 C 风格的强制转换,并将该强制转换保存到单独的变量中以避免一直强制转换!更干净的代码!

于 2013-05-13T10:53:42.673 回答