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

2024-11-01 08:41:00
admin
原创
40
摘要:问题描述:我正在使用Ubuntu 10.10所以我就是这么做的。你好.java:class Hello { public native void sayHello(); static { System.loadLibrary("hellolib&qu...

问题描述:

我正在使用Ubuntu 10.10

所以我就是这么做的。

你好.java

class Hello {
        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

你好.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++ins 而gcc不是 。但是使用该方法仍然有问题load。我似乎无法告诉它正确的路径。


解决方案 1:

本机库可以通过 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

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

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

解决方案 2:

我的代码终于可以运行了。这是 hello.java

public class hello {
  public native void sayHello(int length) ;
  public static void main (String args[]) {
    String str = "I am a good boy" ;
    hello h = new hello () ;
    h.sayHello (str.length() ) ;
  }
  static {
    System.loadLibrary ( "hello" ) ;
  }
}

您应该将其编译为:

$ javac hello.java 

要创建.h 文件,您应该运行以下命令:

$ javah -jni hello

这是hello.h

JNIEXPORT void JNICALL Java_hello_sayHello
(JNIEnv *, jobject, jint);

这里是hello.c

#include<stdio.h>
#include<jni.h>
#include "hello.h" 

JNIEXPORT void JNICALL Java_hello_sayHello
  (JNIEnv *env, jobject object, jint len) {
  printf ( "
Length is %d", len ); }

要编译它并创建共享库,我们必须运行以下命令:

$ gcc -I/usr/lib/jvm/java-6-openjdk/include -o libhello.so -shared hello.c

然后最后运行这个:

$ java -Djava.library.path=. hello

解决方案 3:

这抱怨 C++ 符号不可用。我似乎记得,当我过去一直做 JNI 东西时,在 C++ 库中链接时会出现问题,我们总是坚持使用普通的 C

如果您将代码更改为标准 C(并重命名文件):

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

JNIEXPORT void JNICALL Java_Hello_sayHello (JNIEnv *env, jobject obj) {
        printf("Hello World");
        return;
}

并编译它

gcc -I/usr/lib/jvm/java-6-openjdk/include  -o libhellolib.so -shared Hello.c

有用

java -Djava.library.path=`pwd` Hello
Hello World
相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   601  
  华为IPD与传统研发模式的8大差异在快速变化的商业环境中,产品研发模式的选择直接决定了企业的市场响应速度和竞争力。华为作为全球领先的通信技术解决方案供应商,其成功在很大程度上得益于对产品研发模式的持续创新。华为引入并深度定制的集成产品开发(IPD)体系,相较于传统的研发模式,展现出了显著的差异和优势。本文将详细探讨华为...
IPD流程是谁发明的   7  
  如何通过IPD流程缩短产品上市时间?在快速变化的市场环境中,产品上市时间成为企业竞争力的关键因素之一。集成产品开发(IPD, Integrated Product Development)作为一种先进的产品研发管理方法,通过其结构化的流程设计和跨部门协作机制,显著缩短了产品上市时间,提高了市场响应速度。本文将深入探讨如...
华为IPD流程   9  
  在项目管理领域,IPD(Integrated Product Development,集成产品开发)流程图是连接创意、设计与市场成功的桥梁。它不仅是一个视觉工具,更是一种战略思维方式的体现,帮助团队高效协同,确保产品按时、按质、按量推向市场。尽管IPD流程图可能初看之下显得错综复杂,但只需掌握几个关键点,你便能轻松驾驭...
IPD开发流程管理   8  
  在项目管理领域,集成产品开发(IPD)流程被视为提升产品上市速度、增强团队协作与创新能力的重要工具。然而,尽管IPD流程拥有诸多优势,其实施过程中仍可能遭遇多种挑战,导致项目失败。本文旨在深入探讨八个常见的IPD流程失败原因,并提出相应的解决方法,以帮助项目管理者规避风险,确保项目成功。缺乏明确的项目目标与战略对齐IP...
IPD流程图   8  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用