当前位置: 首页 > 面试题库 >

如何在Linux上为JNI应用程序编译动态库?

羊舌子瑜
2023-03-14
问题内容

我正在使用 Ubuntu 10.10

这就是我所做的。

Hello.java

class Hello {
        html" target="_blank">public native void sayHello();

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

        public static void main(String[] args){
                Hello h = new Hello();
                h.sayHello();
        }
}

然后我运行以下命令:

dierre@cox:~/Scrivania/provajni$ javac Hello.java

dierre@cox:~/Scrivania/provajni$ javah -jni Hello

我已获得Hello.classHello.h

您好

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class Hello */

#ifndef _Included_Hello
#define _Included_Hello
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     Hello
 * Method:    sayHello
 * Signature: ()V
 */
JNIEXPORT void JNICALL Java_Hello_sayHello
  (JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

然后我创建了 Hello.cpp

#include <jni.h>
#include "Hello.h"
#include  <iostream>

using namespace std;

JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) {
        cout << "Hello World!" << endl;
        return;
}

现在我认为我搞砸了。我从本指南(“

编译动态或共享库”
部分)中
得到了
启发
__

dierre@cox:~/Scrivania/provajni$ gcc -I"/usr/lib/jvm/java-6-sun/include" -I"/usr/lib/jvm/java-6-sun/include/linux" -o hellolib.so -shared -Wl,-soname,hello.so Hello.cpp -static -lc

生成文件 hellolib.so

但是当我尝试运行它时,java Hello出现以下错误:

Exception in thread "main" java.lang.UnsatisfiedLinkError: no hellolib in java.library.path
 at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1734)
 at java.lang.Runtime.loadLibrary0(Runtime.java:823)
 at java.lang.System.loadLibrary(System.java:1028)
 at Hello.<clinit>(Hello.java:4)
Could not find the main class: Hello.  Program will exit.

我什至尝试了这个:

  LD_LIBRARY_PATH=`pwd`
  export LD_LIBRARY_PATH

没有结果。

我知道我在做一些非常愚蠢的事情,但我不知道这是什么。动态库是使用-shared选项生成的,不是吗?

更新#1

我试着static { System.load("/home/dierre/Scrivania/provajni/hellolib.so"); }看看是否可行,但现在:

Exception in thread "main" java.lang.UnsatisfiedLinkError: /home/dierre/Scrivania/provajni/hello.so: /home/dierre/Scrivania/provajni/hello.so: undefined symbol: _ZSt4cout
    at java.lang.ClassLoader$NativeLibrary.load(Native Method)
    at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1803)
    at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1699)
    at java.lang.Runtime.load0(Runtime.java:770)
    at java.lang.System.load(System.java:1003)
    at Hello.<clinit>(Hello.java:4)

更新#2 好了,要解决 更新#1 问题,显然我必须使用g++insted of
gccload虽然仍然无法使用该方法。我似乎无法告诉它正确的道路。


问题答案:

可以通过loadLibrary使用有效名称加载本机库。例如,对于Linux家族的lib XXXX
.so,您的hellolib.so应该重命名为libhello.so。顺便说一下,我使用jni开发Java,我将实现和本地接口(.c或.cpp)分开。

static {
    System.loadLibrary("hello"); // will load libhello.so
}

实现标头(HelloImpl.h):

#ifndef _HELLO_IMPL_H
#define _HELLO_IMPL_H

#ifdef __cplusplus
        extern "C" {
#endif

        void sayHello ();

#ifdef __cplusplus
        }
#endif

#endif

HelloImpl.cpp:

#include "HelloImpl.h"
#include  <iostream>

using namespace std;

void sayHello () {
    cout << "Hello World!" << endl;
    return;
}

Hello.c(我更喜欢在c中编译jni):

#include <jni.h>
#include "Hello.h"
#include "HelloImpl.h"

JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) {
    sayHello();
    return;
}

最后,我们可以按以下步骤进行编译:

  1. 编译obj(生成HelloImpl.o)

g ++ -c -I“ / opt / java / include” -I“ / opt / java / include / linux”
HelloImpl.cpp

  1. 用.o编译jni

g ++ -I“ / opt / java / include” -I“ / opt / java / include / linux” -o
libhello.so -shared -Wl,-soname,hello.so Hello.c HelloImpl.o -static -lc

在第2步中,我们使用g ++进行编译。这个非常重要。您可以看到如何混合使用C和C
++

编译后,可以检查使用nm命名的函数:

$ nm libhello.so |grep say
00000708 T Java_Hello_sayHello
00000784 t _GLOBAL__I_sayHello
00000718 T sayHello

有一个标记为T的Java_Hello_sayHello。它应完全等于您的本机方法名称。如果一切正常。您可以运行它:

$ java -Djava.library.path=. Hello
Hello World!


 类似资料:
  • 问题内容: 我正在Windows 10计算机(使用Visual Studio 2015更新3 + Microsoft .NET Core 1.0.1 VS 2015工具预览2)上开发.NET Core应用程序,该应用程序应在Ubuntu 16计算机上发布。为此,我必须将源代码移动到终端机并在那里进行编译,以使其运行。例如,我无法在Windows上编译代码并在Linux上运行它。问题:有什么方法可以

  • 问题内容: 我想安装rJava,但没有用。当我在控制台中键入R CMD javareconf时,出现以下错误: 任何想法如何解决? 问题答案: 该文件是JDK安装的一部分。您可能没有正确安装Java JDK。从此处下载Oracle Java ,或使用软件包管理器。 如果您已经安装了Java JDK,则将JAVA_HOME环境变量设置为指向JDK目录。

  • 在Linux上如何使用gcc编译c程序 如何安装gcc 编辑c代码 编辑代码 运行代码

  • 我想通知您,其中一个雇主已经创建了一个基于java的程序,通过I2C通信与Raspberry PI。 他在C++中创建了一些端口方法(open,write,close),并使用JNI库与基于java的程序建立了一个桥,因此,在本例中,我们可以使用C++原生库(JNI)桥通过java与嵌入式卡进行通信。 在本例中,只要禁用C++JN️library命令,我的java(包括javafx)程序就可以在r

  • 所以我有一个使用unity的android游戏,我在dll中寻找脚本,并确切地知道我应该在C#代码中更改什么,我的意思是我几乎完全反向工程了这个应用程序。下一步该怎么办?顺从它?但怎么做?

  • 以下是错误: 扑动医生输出: []Flutter(Channel stable,v1.7.8+Hotfix.4,在Linux上,locale en_IN)•Flutter版本1.7.8+Hotfix.4,在/home/sagar/development/Flutter•Framework修订版20E59316B8(两周前),2019-07-18 20:04:33-0700•Engine修订版fee