исключение c0000005 (нарушение прав доступа) при использовании JNI в C ++ - PullRequest
1 голос
/ 25 июля 2011

Я начал использовать JNI для вызова классов Java из C ++ несколько недель назад, и сегодня я столкнулся с особой ситуацией.Я новичок в C ++ (хотя знаком с Java), так что это может быть ошибка n00b.У меня есть этот класс в Java с именем IntArray.java, и я создал другой класс в C ++ с именем IntArrayProxy (разделенный на файл .h и .cpp) для доступа к его методам через JNI.У меня также есть другой исходный файл с именем IntArrayProxyTest.cpp, который проверяет методы IntArrayProxy.

В IntArrayProxy я использую элемент данных jobject * intArrayObject, который содержит экземпляр класса Java, и передаю его каждому методуКласс IntArrayProxy.Моя проблема заключается в том, что когда я использую его в качестве указателя (jobject *), после вставки (с помощью insert) некоторых целых чисел и изменения некоторых из них (с помощью setElement), когда я использую один и тот же метод size () дважды, исполняемый файл, который я создаю, вылетаетпредоставив мне исключение c0000005 (нарушение прав доступа).

Первое, что я заметил, было то, что вообще нет проблем, если я использую обычный задание (не задание *), а второе - возникновение исключениякогда я пытаюсь вызвать второй не пустой метод.insert () и setElement (int, int) оба void, поэтому я могу вызывать их столько раз, сколько захочу.Я пытался сделать это почти со всеми не пустыми методами, и каждый раз, когда я пытался вызвать два не пустых метода, возникало одно и то же исключение.

Я думал, что, возможно, указатель каким-то образом изменился, поэтому я попытался напечатать объект задания* в каждом методе, но он остался прежним.Второе объяснение, которое я нашел на форумах, было то, что, возможно, объект был разрушен, но я не знаю, как это проверить и почему это могло произойти.Я потратил весь день на поиск и отладку, но не повезло.

Я думаю, что это не имеет значения, но я использую последний (32-битный) компилятор minGW на Win7 (64-битной).Я также использую 32-битную jvm.dll.Я использую командную строку для ее компиляции ( g ++ -I "C: \ Program Files (x86) \ Java \ jdk1.6.0_26 \ include" -I "C: \ Program Files (x86) \ Java \ jdk1.6.0_26 \ include \ win32 "IntArrayProxy.cpp IntArrayProxyTest.cpp -L" C: \ Users \ jEOPARd \ Desktop \ Creta \ JNI samples "-ljvm -o IntArrayProxyTest.exe )

Надеюсь, кто-нибудьможете мне помочь !!

Спасибо заранее !!

Kostis

IntArray.java

package SortIntArray;

public class IntArray {

private int[] arrayOfInt;
private int cursor;
private static final int CAPACITY = 5;

public IntArray() {
    arrayOfInt = new int[CAPACITY];
    cursor = 0;
}

public void insert(int n) {
    if (isFull()) {
        System.out.println("Inserting in a full array!");
    } else {
        arrayOfInt[cursor++] = n;
    }
}

public int removeLast() {
    if (isEmpty()) {
        System.out.println("Removing from an empty array!");
        return -666;
    } else {
        return arrayOfInt[--cursor];
    }
}

private boolean isEmpty() {
    return cursor <= 0;
}

private boolean isFull() {
    return cursor >= CAPACITY;
}

public String toString() {
    if (isEmpty()) {
        return "Empty Array";
    }
    String s = Integer.toString(arrayOfInt[0]);
    for (int i = 1; i < cursor; i++) {
        s += ", " + Integer.toString(arrayOfInt[i]);
    }
    return s;
}

public int size() {
    return cursor;
}

public int getElement(int pos) {
    return arrayOfInt[pos];
}

public void setElement(int pos, int newElement) {
    arrayOfInt[pos] = newElement;
}
}

IntArrayProxy.h

#ifndef INTARRAYPROXY_H
#define INTARRAYPROXY_H

#include <jni.h>
using namespace std;

class IntArrayProxy {
JNIEnv *env;
jclass intArrayClass;
jobject *intArrayObject; //giati oxi pointer?
public:

IntArrayProxy(JNIEnv*);

void insert(int n);
int removeLast();
string toString();
int size();
int getElement(int);
void setElement(int pos, int newElement);
jobject *getIntArrayObject();
};


#endif  /* INTARRAYPROXY_H */

IntArrayProxy.cpp

#include <stdio.h>
#include <cstdlib>
#include <iostream>
using namespace std;

#include "IntArrayProxy.h"

IntArrayProxy::IntArrayProxy(JNIEnv *envir) {
env = envir;
intArrayClass = env -> FindClass("SortIntArray/IntArray");
if (intArrayClass == NULL) {
    cout << "--intArrayClass = NULL\n";
    exit(0);
}
jmethodID IntArrayConstructor = env->GetMethodID(intArrayClass, "<init>", "()V");
if (IntArrayConstructor == NULL) {
    cout << "--IntArrayConstructor = NULL";
    exit(0);
}
cout << "IntArrayProxy: Got constructor\n";
jobject obj = env -> NewObject(intArrayClass, IntArrayConstructor);
intArrayObject = &obj;     // I also can't assign intArrayObject directly at the above line, I don't know why (would be glad if you could tell me)
if (*intArrayObject == NULL) {
    cout << "--*intArrayObject = NULL";
    exit(0);
}
cout << "IntArrayProxy: Object created\n";
}

void IntArrayProxy::insert(int n) {
jmethodID insertID = env -> GetMethodID(intArrayClass, "insert", "(I)V");
if (insertID == NULL) {
    cout << "--insertID = NULL";
    exit(0);
}
env -> CallVoidMethod(*intArrayObject, insertID, (jint) n);
}

int IntArrayProxy::removeLast() {
jmethodID removeLastID = env -> GetMethodID(intArrayClass, "removeLast", "()I");
if (removeLastID == NULL) {
    cout << "--removeLastID = NULL";
    exit(0);
}
return (int) (env -> CallIntMethod(*intArrayObject, removeLastID));
}

string IntArrayProxy::toString() {
jmethodID toStringID = env -> GetMethodID(intArrayClass, "toString", "()Ljava/lang/String;");
if (toStringID == NULL) {
    cout << "--toStringID = NULL";
    exit(0);
}
jstring intArrayString = (jstring) env -> CallObjectMethod(*intArrayObject, toStringID);
string s = env -> GetStringUTFChars(intArrayString, NULL);
return s;
}

int IntArrayProxy::size(){
jmethodID sizeID = env -> GetMethodID(intArrayClass, "size", "()I");
if (sizeID == NULL) {
    cout << "--sizeID = NULL";
    exit(0);
}
return (int) (env -> CallIntMethod(*intArrayObject, sizeID));    
}

int IntArrayProxy::getElement(int pos) {
jmethodID getElementID = env -> GetMethodID(intArrayClass, "getElement", "(I)I");
if (getElementID == NULL) {
    cout << "--getElementID = NULL";
    exit(0);
}
return (int) env -> CallObjectMethod(*intArrayObject, getElementID, (jint) pos);
}

void IntArrayProxy::setElement(int pos, int newElement){
jmethodID setElementID = env -> GetMethodID(intArrayClass, "setElement", "(II)V");
if (setElementID == NULL) {
    cout << "--setElementID = NULL";
    exit(0);
}
env -> CallVoidMethod(*intArrayObject, setElementID, (jint) pos, (jint) newElement);    
}

jobject *IntArrayProxy::getIntArrayObject(){
return intArrayObject;
}

IntArrayProxyTest.cpp

#include <stdio.h>
#include <jni.h>
#include <cstdlib>
#include <iostream>
using namespace std;

#include "IntArrayProxy.h"

int main() {
cout << "--Starting..\n";
JavaVM *jvm; /* denotes a Java VM */
JNIEnv *env; /* pointer to native method interface */
JavaVMInitArgs vm_args; /* JDK/JRE 6 VM initialization arguments */
JavaVMOption* options = new JavaVMOption[1];
options[0].optionString = "-Djava.class.path=C:\\Users\\jEOPARd\\Desktop\\Creta\\JNI samples\\JNI tests\\build\\classes";
vm_args.version = JNI_VERSION_1_6;
vm_args.nOptions = 1;
vm_args.options = options;
vm_args.ignoreUnrecognized = false;
/* load and initialize a Java VM, return a JNI interface
 * pointer in env */
cout << "--Creating VM..\n";
JNI_CreateJavaVM(&jvm, (void **) &env, &vm_args);
cout << "--VM created successfully!!\n";
delete options;

cout << "--Finding IntArray class..\n";
IntArrayProxy *intArrayProxy = new IntArrayProxy(env);
if (env->ExceptionOccurred())
    env->ExceptionDescribe();


intArrayProxy -> insert(1);
intArrayProxy -> insert(10);
intArrayProxy -> insert(3);
intArrayProxy -> insert(88);
intArrayProxy -> insert(32);

intArrayProxy ->setElement(2, 5);
intArrayProxy ->setElement(3, 7);    

cout << "Size: " << intArrayProxy -> size() << endl;
cout << "Size: " << intArrayProxy -> size() << endl;

cout << "--Destroying VM..\n";
jvm->DestroyJavaVM();
cout << "--Done!!!\n";
return 0;
}

1 Ответ

2 голосов
/ 25 июля 2011

В прокси-конструкторе:

intArrayObject = & obj;

Вы берете адрес переменной в стеке. Когда конструктор завершает работу, адрес больше не действителен, поэтому происходит сбой.

intArrayObject (в заголовке) должен быть заданием, а не заданием *, и его различное использование должно быть соответственно изменено.

...