.c代码的java包装器

时间:2013-03-22 08:38:18

标签: java c java-native-interface swig

我有以下c代码:

test.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";
}

test.h

#ifndef TEST_H_
#define TEST_H_

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

#endif

的main.c

#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

GIVES:

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)

我做错了什么?

2 个答案:

答案 0 :(得分:2)

似乎你不明白jni是如何工作的,请查看本教程:

Calling C code from Java using JNI

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

native String print_hello();

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

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

答案 1 :(得分: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个文件
  • 确保您使用.so加载System.loadLibrary("test_app_wrap");库 - 您已经这样做了
  • 只需致电test_app.print_hello()
  • 即可

另外,建议阅读SWIG手册中的21 SWIG and Java部分。 预备基本C / C ++包装部分介绍了所有基础知识。

您收到的错误表明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();
    }
}

希望有所帮助:)