linux下 java JNI调用动态链接库

JNI就是在java中调用非java代码,如果看到native声明的方法,肯定是有jni调用的,其实现是由其他语言写的程序,常用的是c,c++。
1 首先创建一个简单的java类:

public class Hello
{
 static
 {
  try
  {
//此处即为本地方法所在链接库名
   System.loadLibrary("hello");
  }
  catch(UnsatisfiedLinkError e)
  {
   System.err.println( "Cannot load hello library:\n " +
                                e.toString() );
  }
 }
 public Hello()
 {
 }
//声明的本地方法
  public native void SayHello(String strName);
} 


2 生成 Hello.h

javac Hello.java生成class文件
javah Hello生成hello.h

这个头文件就是根据了jni的规则来生成了native方法在c/c++的声明。下一步就需要新建c/c++文件来实现这里面的方法

3 在与Hello.h相同的路径下创建一个CPP文件Hello.cpp
#include "Hello.h"
JNIEXPORT void JNICALL Java_Hello_SayHello  (JNIEnv * env, jobject arg, jstring instring)
{
    const char *str = env->GetStringUTFChars( instring, JNI_FALSE );
    printf("Hello,%s\n",str);
    env->ReleaseStringUTFChars( instring, str );
    return;
}

4.编译生成共享库

a. 编译命令,生成Hello.o
g++ -I /usr/lib/jvm/java-6-sun-1.6.0.03/include -I /usr/lib/jvm/java-6-sun-1.6.0.03/include/linux -fPIC -c Hello.cpp
b.生成动态库文件,libhello.so.1.0
g++ -shared -Wl,-soname,libhello.so.1 -o libhello.so.1.0 Hello.o

接下来将生成的共享库拷贝为标准文件名

cp libhello.so.1.0 libhello.so

最后通知动态链接程序此共享文件的路径。

export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH

这里用export 加入共享文件的路径,有时候会有点问题,比如:环境变量不会马上更新等等。

还有一个办法,就是直接将libhello.so拷贝到 /usr/lib 或者/lib 等系统库目录下


5.编写一个简单的Java程序来测试我们的本地方法。
import Hello;
import java.util.*;
public class ToSay
{
	public static void main(String argv[])
	{
		ToSay say = new ToSay();
	}
	public ToSay()
	{
		Hello h = new Hello();
		//调用本地方法向John问好
		h.SayHello("John");			
	}
}

用javac编译ToSay.java,生成ToSay.class
向执行普通Java程序一样使用java ToSay,我们会看到在屏幕上出现Hello,John

参考: http://watershitter.iteye.com/blog/477615
http://longrenrex.blog.sohu.com/93931899.html

你可能感兴趣的:(java,c,linux,C#,jni)