在Mono Droid中创建数组并将其作为参数传递给JNI

时间:2013-03-01 16:25:38

标签: c# android arrays mono java-native-interface

在单声道Droid项目中,我有一个.java文件,其中包含以下类的定义:

class A
{
    String str;
    String[] arr;

    public A(string str, string[] arr) 
    {
        this.str = str;
        this.arr = arr;
    }
}

以及需要使用JNI环境从C#调用的方法foo(A[])

在C#文件中,我有一个C#类的相应镜像定义

class B
{
    string str;
    string[] arr;
    ...
}

在一段C#代码中,我准备了B[]数组,现在想要复制一个新数组中的所有对象,然后传递给该方法。为此我写了一个方法B.ConvertArrayToJValue,它返回JValue

public class B
{
    string str;
    string[] arr;

    public B(string str, string[] arr) 
    {
        this.str = str;
        this.arr = arr;
    }

    public JValue ToJavaObject()
    {
        IntPtr classA = JNIEnv.FindClass("com/packagename/A");
        IntPtr method_ctor = JNIEnv.GetMethodID(classA, "<init>", "(Ljava/lang/String;[Ljava/lang/String;)V");
        Java.Lang.String.[] tempValues = new Java.Lang.String[this.arr.Length];
        for (int i = 0; i < this.arr.Length; ++i)
        {
            tempValues[i] = new Java.Lang.String(this.arr[i]);
        }

        JValue strArray = new JValue(Java.Lang.Object.FromArray<Java.Lang.String>(tempValues));
        IntPtr result = JNIEnv.NewObject(classA, method_ctor, new JValue[]{new JValue(new Java.Lang.String(this.str)), strArray});
        return new JValue(new Java.Lang.Object(result, JniHandleOwnership.DoNotTransfer));
    }

    public static JValue ConvertArrayToJValue(B[] arr)
    {
        JValue[] tempCopy = new JValue[arr.Length];
        for (int i = 0; i < arr.Length; ++i)
        {
            tempCopy[i] = arr[i].ToJavaObject();
        }
        //FIXME: doesn't properly convert arrays! 
        // Compiles but throws an exception in runtime: "Can't convert JValue to IJavaObject"
        return new JValue(Java.Lang.Object.FromArray<JValue>(tempCopy));
    }

}

以及C#代码中的其他地方:

b[] = new B[]{ ... };

IntPtr method_foo = JNIEnv.GetStaticMethodID(classC, "foo", "([Lcom/packagename/A;)V");
JNIEnv.CallStaticVoidMethod(classC, method_foo, new JValue[] { B.ConvertArrayToJValue(b) });

这不起作用。字符串数组似乎是正确生成的(至少它看起来像C#字符串被转换为Java字符串(?))但是当我尝试创建类型为B的自定义对象数组时,我得到一个错误:

Can't convert JValue to IJavaObject

谢谢!

2 个答案:

答案 0 :(得分:0)

为什么要在JValue结构中包装返回值?

我认为只需从方法IJavaObject返回ToJavaObject即可。

那是:

public IJavaObject ToJavaObject()
{
    // Snip
    return new Java.Lang.Object(result, JniHandleOwnership.DoNotTransfer);
}

public static IJavaObject ConvertArrayToJavaObject(B[] arr)
{
    // Snip
    return Java.Lang.Object.FromArray<IJavaObject>(tempCopy);
}

答案 1 :(得分:0)

我从一位Xamarin开发人员那里获得了以下工作解决方案。理解JNI如何工作实际上有很多帮助:

http://forums.xamarin.com/discussion/1930/creating-arrays-of-custom-java-objects-in-jni