linux so 调用 so linux有什么用
linuxso加载路径linuxso加载
SO文件是Linux系统下的动态链接库文件,它包含编译好的代码和数据,可以被其他程序在运行时调用。以下是关于SO文件和相关概念的澄清和润色:
1. SO文件与Windows下的.dll文件类似,都是用于提供程序函数库的文件。
2. SO文件通常是C或C++程序编译生成的动态链接库,与Java的.class文件(字节码)不同。
3.在Linux系统中,SO文件不能直接执行,它们被称为共享库,是为了在多个程序间共享代码而设计的。
4.使用SO文件的方法包括:
-编译动态库:通过命令行指令,将多个源文件编译为共享库文件,例如`gcc test_a.c test_b.c test_c.c-fPIC-shared-o libtest.so`。
-链接动态库:在程序编译时,通过`-L`标志指定库文件所在的目录,通过`-l`标志指定库的名称。例如,`gcc test.c-L.-ltest-o test`。
-运行程序时,系统会自动加载所需的动态库。
SO文件格式遵循ELF(Executable and Linkable Format)标准,这是Linux下可执行文件、共享库文件和目标文件的通用格式。ELF文件可以从链接视图和装载视图两个角度来理解:
-链接视图关注ELF文件在编译后的结构,包括多个section,每个section有不同的名称和权限。
-装载视图关注ELF文件在运行时的结构,包括多个segment,每个segment有不同的权限和名称。实际上,一个segment通常包含具有相同权限的多个section。
对于Ubuntu等Linux发行版,打包Qt程序通常涉及将依赖的Qt库一同打包,以便在不同发行版上兼容使用。而Android平台由于不包含glibc库和标准库链接器ld-linux.so.3,打包共享库的方法需要特别处理,但仍然可以通过在特定目录放置必需的库文件,并在程序中设置正确的路径来实现跨平台运行。
最后,由于Android基于Linux,它同样支持SO文件,使得C/C++编写的Native代码可以在Android设备上运行。为了提高安全性,越来越多的应用将核心代码编写为SO文件,供上层的Java代码调用。ELF文件头包含了文件的基本信息,如魔数、目标文件类型、体系结构、入口地址等,是理解和操作ELF文件的关键。
linux环境java如何调用so文件
用JNI实现
实例:
创建HelloWorld.java
class HelloWorld
{
private native void print();
public staticvoid main(String[] args)
{
new HelloWorld().print();
}
static
{
System.loadLibrary("HelloWorld");
}
}
注意print方法的声明,关键字native表明该方法是一个原生代码实现的。另外注意static代码段的System.loadLibrary调用,这段代码表示在程序加载的时候,自动加载libHelloWorld.so库。
编译HelloWorld.java
在命令行中运行如下命令:
javac HelloWorld.java
在当前文件夹编译生成HelloWorld.class。
生成HelloWorld.h
在命令行中运行如下命令:
javah-jni HelloWorld
在当前文件夹中会生成HelloWorld.h。打开HelloWorld.h将会发现如下代码:
/* DO NOT EDIT THIS FILE- it is machine generated*/
#include<jni.h>
/* Header for class HelloWorld*/
#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern"C"{
#endif
/*
* Class: HelloWorld
* Method: print
* Signature:()V
*/
JNIEXPORT void JNICALL Java_HelloWorld_print
(JNIEnv*, jobject);
#ifdef __cplusplus
}
#endif
#endif
该文件中包含了一个函数Java_HelloWorld_print的声明。这里面包含两个参数,非常重要,后面讲实现的时候会讲到。
实现HelloWorld.c
创建HelloWorld.c文件输入如下的代码:
#include<jni.h>
#include<stdio.h>
#include"HelloWorld.h"
JNIEXPORT void JNICALL
Java_HelloWorld_print(JNIEnv*env, jobject obj)
{
printf("Hello World!\n");
}
注意必须要包含jni.h头文件,该文件中定义了JNI用到的各种类型,宏定义等。
另外需要注意Java_HelloWorld_print的两个参数,本例比较简单,不需要用到这两个参数。但是这两个参数在JNI中非常重要。
env代表java虚拟机环境,Java传过来的参数和c有很大的不同,需要调用JVM提供的接口来转换成C类型的,就是通过调用env方法来完成转换的。
obj代表调用的对象,相当于c++的this。当c函数需要改变调用对象成员变量时,可以通过操作这个对象来完成。
编译生成libHelloWorld.so
在Linux下执行如下命令来完成编译工作:
cc-I/usr/lib/jvm/java-6-sun/include/linux/
-I/usr/lib/jvm/java-6-sun/include/
-fPIC-shared-o libHelloWorld.so HelloWorld.c
在当前目录生成libHelloWorld.so。注意一定需要包含Java的include目录(请根据自己系统环境设定),因为Helloworld.c中包含了jni.h。
另外一个值得注意的是在HelloWorld.java中我们LoadLibrary方法加载的是
“HelloWorld”,可我们生成的Library却是libHelloWorld。这是Linux的链接规定的,一个库的必须要是:lib+库
名+.so。链接的时候只需要提供库名就可以了。
运行Java程序HelloWorld
大功告成最后一步,验证前面的成果的时刻到了:
java HelloWorld
如果你这步发生问题,如果这步你收到java.lang.UnsatisfiedLinkError异常,可以通过如下方式指明共享库的路径:
java-Djava.library.path='.' HelloWorld
当然还有其他的方式可以指明路径请参考《在Linux平台下使用JNI》。
我们可以看到久违的“Hello world!”输出了。
Linux c调用so
实例代码(soTest.c):
1#include<stdio.h>
2#include<dlfcn.h>
3
4 int main(int argc, char*argv[]){
5 void* libm_handle= NULL;
6 float(*cosf_method)(float);
7 char*errorInfo;
8 float result;
9
10// dlopen函数还会自动解析共享库中的依赖项。这样,如果您打开了一个依赖于其他共享库的对象,它就会自动加载它们。
11//函数返回一个句柄,该句柄用于后续的 API调用
12 libm_handle= dlopen("libm.so", RTLD_LAZY);
13//如果返回 NULL句柄,表示无法找到对象文件,过程结束。否则的话,将会得到对象的一个句柄,可以进一步询问对象
14 if(!libm_handle){
15//如果返回 NULL句柄,通过dlerror方法可以取得无法访问对象的原因
16 printf("Open Error:%s.\n",dlerror());
17 return 0;
18}
19
20//使用 dlsym函数,尝试解析新打开的对象文件中的符号。您将会得到一个有效的指向该符号的指针,或者是得到一个 NULL并返回一个错误
21 cosf_method= dlsym(libm_handle,"cosf");
22 errorInfo= dlerror();//调用dlerror方法,返回错误信息的同时,内存中的错误信息被清空
23 if(errorInfo!= NULL){
24 printf("Dlsym Error:%s.\n",errorInfo);
25 return 0;
26}
27
28//执行“cosf”方法
29 result=(*cosf_method)(0.0);
30 printf("result=%f.\n",result);
31
32//调用 ELF对象中的目标函数后,通过调用 dlclose来关闭对它的访问
33 dlclose(libm_handle);
34
35 return 0;
36}
在这个例子中主要是调用了 math库(libm.so)中的“cosf”函数,dlopen函数的第二个参数表示加载库文件的模式,主要有两种:RTLD_LAZY暂缓决定,等有需要时再解出符号;RTLD_NOW立即决定,返回前解除所有未决定的符号。另外记得引用包含API的头文件“#include<dlfcn.h>”(^_^)。