我用Ubuntu 10.10

我就是这么做的.

Hello.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.class分和Hello.h分.

Hello.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;
}

现在我觉得我搞砸了.我当时inspired

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

但当我try 用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选项生成的,不是吗?

Update #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)

Update #2

推荐答案

本机库可以由具有有效名称的loadLibrary加载.举个例子,libXXXX.所以对于linux家族,你的hellolib.所以应该改名为libhello.所以

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

你好.cpp:

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

using namespace std;

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

你好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"

  1. 使用编译jni.o

g++-I"/opt/java/include"

在第2步中,我们使用g++来编译它.这非常重要.你能看到How to mix C and 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!

Linux相关问答推荐

通过shell 脚本基于AM/PM对日期时间字段进行排序

如何在 shell 脚本中自定义 SFTP 详细输出

std::chrono::time_zone 在不同操作系统上不可用

在 bash 脚本中保持两个进程处于活动状态(并在死亡时重生它们)

如何使用 sed 和 date 命令替换和格式化字符串中的日期?

替换前 3 个字符范围内的所有整数

删除一行和它之前的一行

c++进程状态中的+是什么意思

使用 sed linux 命令和 i sed 命令进行 preprend 时的反向引用

为什么我的 Docker 进程不断在我的 Raspberry Pi 上重新启动?

提交SLURM作业(job)时出现nohup问题

从 Ansible 中的 shell 命令输出中提取特定数据

我在哪里放置第三方库来设置 C++ Linux 开发环境?

用于判断进程是否正在运行并对结果采取行动的 Linux 脚本

从 .war 文件外部化 Tomcat webapp 配置

使用 shell 脚本获取 MAC 地址

加密/解密在两个不同的 openssl 版本之间不能很好地工作

为什么两次使用 grep 时没有显示输出?

如何从 shell 确定当前的 CPU 利用率?

事件驱动和异步有什么区别?在 epoll 和 AIO 之间?