linux so 加载 linux权限不够

大家好,今天来为大家分享linux so 加载的一些知识点,和linux权限不够的问题解析,大家要是都明白,那么可以忽略,如果不太清楚的话可以看看本篇文章,相信很大概率可以解决您的问题,接下来我们就一起来看看吧!

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系统重so库加载环境变量设置方法

Linux环境配置,动态库,静态库,profile,bashrc,source,ldconfig

环境配置第一:export PATH

修改办法:export PATH=$PATH:/home/test/bin

生效操作:立即生效

生效期限:当前终端有效,窗口关闭后无效

生效范围:仅对当前用户有效

所属级别:用户级别

注意事项:不要忘了加上原来的配置,即$PATH部分,避免覆盖原来配置

第二:vim~/.bashrc

修改办法:修改~/.bashrc,在最后一行加上:export PATH=$PATH:/home/test/bin

生效操作:使用相同的用户打开新的终端时生效,或者手动source~/.bashrc生效

生效期限:永久有效

生效范围:仅对当前用户有效

所属级别:用户级别

注意事项:如果有后续的环境变量加载文件覆盖了PATH定义,则可能不生效

第三:vim~/.bash_profile

修改办法:修改~/.bash_profile,在最后一行加上:export PATH=$PATH:/home/test/bin

生效操作:使用相同的用户打开新的终端时生效,或者手动source~/.bash_profile生效

生效期限:永久有效

生效范围:仅对当前用户有效

所属级别:用户级别

注意事项:如果有后续的环境变量加载文件覆盖了PATH定义,则可能不生效

第四:vim/etc/bashrc

修改办法:修改/etc/bashrc,在最后一行加上:export PATH=$PATH:/home/test/bin

生效操作:新开终端生效,或者手动source/etc/bashrc生效

生效期限:永久有效

生效范围:对所有用户有效

所属级别:系统级别

注意事项:修改系统配置,需要管理员权限(如root)或者对该文件的写入权限

第五:vim/etc/profile

修改办法:修改/etc/profile,在最后一行加上:export PATH=$PATH:/home/test/bin

生效操作:新开终端生效,或者手动source/etc/profile生效

生效期限:永久有效

生效范围:对所有用户有效

所属级别:系统级别

注意事项:修改系统配置,需要管理员权限(如root)或者对该文件的写入权限

第六:vim/etc/environment

修改办法:修改/etc/environment,在最后一行加上:export PATH=$PATH:/home/test/bin

生效操作:新开终端生效,或者手动source/etc/environment生效

生效期限:永久有效

生效范围:对所有用户有效

所属级别:系统级别

注意事项:修改系统配置,需要管理员权限(如root)或者对该文件的写入权限

用户级别环境变量定义文件:~/.bashrc、~/.profile(部分系统为:~/.bash_profile)

系统级别环境变量定义文件:/etc/bashrc、/etc/profile(部分系统为:/etc/bash_profile)、/etc/environment

注意:更改系统环境配置后,需要执行source命令使其立即生效

加载原理分析:系统环境变量->用户自定义环境变量

Linux加载环境变量的顺序如下:

1,/etc/environment

2,/etc/profile(部分系统为:/etc/bash_profile)

3,/etc/bashrc

4,/etc/profile.d/xxx.sh

5,~/.profile(部分系统为:~/.bash_profile)

7,~/.bashrc

什么是/etc/profile.d/?

当用户登录Linux系统或使用su命令切换到另一个用户时,首先要确保执行的启动脚本就是/etc/profile。注意:只有Login shell启动时才会运行/etc/profile这个脚本,而Non-login shell不会调用这个脚本

在/etc/profile.d目录中存放的是一些应用程序所需的启动脚本,其中包括了颜色、语言、less、vim及which等命令的一些附加设置。这些脚本文件之所以能够被自动执行,是因为在/etc/profile中使用一个for循环语句来调用这些脚本,而这些脚本文件是用来设置一些变量和运行一些初始化过程的

也就是说,Login shell启动时会运行/etc/profile,/etc/profile再调用/etc/profile.d/的脚本进行设置

/etc/profile和/etc/profile.d区别?

1、两个文件都是设置环境变量文件的,/etc/profile是永久性的环境变量,是全局变量,/etc/profile.d/设置所有用户生效

2、/etc/profile.d/高度解耦,比/etc/profile好维护,不想要什么变量直接删除/etc/profile.d/下对应的脚本即可

3、/etc/profile和/etc/profile.d同样是登录(login)级别的变量,当用户重新登录shell时会触发,所以效果一致

如何配置函数库路径?

为了将动态函数库载入内存(高速缓存),首先在/etc/ld.so.conf文件中列出动态函数库所在的目录,然后使用ldconfig命令将/etc/ld.so.conf中的数据载入高速缓存。

具体操作包括创建/编辑/etc/ld.so.conf.d/目录下的配置文件,添加函数库路径,执行ldconfig命令更新缓存。

如何修改ld.so.conf.d和/etc/profile.d的配置?

修改ld.so.conf.d和/etc/profile.d目录下的配置文件,可以动态添加、删除环境变量配置,方便管理而不影响系统整体配置。

在配置系统环境或函数库后,确保执行source命令以立即生效,以确保新设置在当前终端或新打开的终端中可用。

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文件的关键。

阅读剩余
THE END