4

我有以下c代码:

测试.c

#include <stdio.h>
#include <math.h>

int add (int a, int b)
{
    a=4;
    b=4;
    return a+b;
}

int add_pointer (int *a, int *b)
{
    printf("values of a,b: %d,%d \n",*a,*b);
return ((*a)+(*b));
}

char* print_hello()
{
    return "hello_world";
}

测试.h

#ifndef TEST_H_
#define TEST_H_

int add(int a, int b);
int add_pointer (int *a, int *b);
char *print_hello();

#endif

主程序

#include "test.h"
#include <stdio.h>
#include <math.h>

int main()
{
    int a,b,c,d;
    char* rez;

    a=5;
    b=2;

    //int *r=&a;
    c= add(a,b);
    d=add_pointer(&a,&b);
    rez=print_hello();

    printf("sum is: %d and : %s %d \n",c,rez,d);
    return 0;
}

test_app.i

%module test_app

%{
#include "test.h"
%}

%include "test.h"

我想为这个 .c 代码创建一个 java 包装器。我想稍后在 android 演示中使用这个包装器。

我做了以下事情:

$: swig -java test_app.i

给:

test_app_wrapper.c
test_app.java
test_appJNI.java
SWIGTYPE_p_int.java 

$: gcc -fpic -c test.c test_app_wrap.c -I /usr/lib/jvm/java-7-openjdk-amd64/include -I /usr/lib/jvm/java-7-openjdk-amd64/include/linux
$: gcc -shared test.o test_app_wrap.o -o libtest_app_wrap.so

更新:

HelloWorld.java

      public class HelloWorld {
    native String print_hello(); /* (1) */
    static {
        System.loadLibrary("test_app_wrap"); /* (2) */
    }
    static public void main(String argv[]) {
        HelloWorld helloWorld = new HelloWorld();
        helloWorld.print_hello(); /* (3) */
    }
}

运行时:

$: javac HelloWorld.java 
$ java HelloWorld

我有:

Exception in thread "main" java.lang.UnsatisfiedLinkError: no test_app_wrap in java.library.path
    at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1856)
    at java.lang.Runtime.loadLibrary0(Runtime.java:845)
    at java.lang.System.loadLibrary(System.java:1084)
    at HelloWorld.<clinit>(HelloWorld.java:4)

我究竟做错了什么?

4

2 回答 2

2

看来您不了解 jni 的工作原理,请查看本教程:

使用 JNI 从 Java 调用 C 代码

test_app_wrap不存在,要使用 JNI,您必须为 C 函数分配一个特定名称,然后使用本机方法创建一个 Java 类来调用它们,即

native String print_hello();

并且,在 Java 类中,加载本机库。

然后您将创建一个 YourClass Java 对象并调用本print_hello()机方法

于 2013-03-22T08:45:25.130 回答
1

当您执行swig -java test_app.i时,它会创建一些 Java 粘合类,您必须将它们包含在您的 Java 项目中。生成的主界面命名test_app.java如下:

public class test_app {
  public static int add(int a, int b) {
    return test_appJNI.add(a, b);
  }

  public static int add_pointer(SWIGTYPE_p_int a, SWIGTYPE_p_int b) {
    return test_appJNI.add_pointer(SWIGTYPE_p_int.getCPtr(a), SWIGTYPE_p_int.getCPtr(b));
  }

  public static String print_hello() {
    return test_appJNI.print_hello();
  }

}

如您所见,它将所有调用委托给test_appJNI,这也是自动生成的,并充当本机代码的粘合剂(使用native方法):

public class test_appJNI {
  public final static native int add(int jarg1, int jarg2);
  public final static native int add_pointer(long jarg1, long jarg2);
  public final static native String print_hello();
}

所以,你应该:

  • 在项目中包含两个生成.java的文件
  • 确保您加载.soSystem.loadLibrary("test_app_wrap");- 您已经这样做了
  • 只需调用test_app.print_hello()

另外,推荐阅读: SWIG 手册中的21 SWIG 和 Java部分。Preliminaries和A tour of basic C / C++ wrapping部分很好地解释了所有基础知识。

您收到的错误表明 JVM 无法加载该.so库。我的第一个问题是它是否正确编译和链接。如果你确定你有libtest_app_wrap.so,它可能不在 JVM 寻找它的路径上(检查这个 - http://www.chilkatsoft.com/p/p_499.asp)。例如,对我来说,有必要添加-Djava.library.path=.到 Java 命令行:

java -Djava.library.path=. HelloWorld

供参考 - 我已经修改了您的HelloWorld.java文件:

public class HelloWorld {
    static {
        System.loadLibrary("test_app_wrap"); 
    }
    static public void main(String argv[]) {
        test_app.print_hello();
    }
}

希望有帮助:)

于 2013-03-22T08:52:50.723 回答