欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

解析java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別

 更新時間:2021年05月11日 10:19:29   作者:gaussrieman123  
這篇文章主要介紹了java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下

背景

近期要將算法部署到一個機(jī)群的虛擬主機(jī)(Debian 9.1 gcc 6.3.0)上,采用的是Java + JNI + shared library的方式來完成底層算法能力的部署。

其中需要用到各種第三方庫,有從源碼編譯的,也有直接下載的so,包括OpenCV相關(guān)、TensorFlow相關(guān)、MKL以O(shè)penMP相關(guān)的動態(tài)庫。

遇到一個問題,libmklml_intel.so 這個庫只能放在 LD_LIBRARY_PATH中進(jìn)行加載,而不能通過java.library.path完成加載,所以有必要搞清楚這兩個路徑究竟有什么區(qū)別。

java.library.path

官方文檔的定義是:List of paths to search when loading libraries
從定義我們可以發(fā)現(xiàn),首先是一個list,也就是說可以包括多個地址,然后這些地址是用來幫助jvm搜索需要加載的庫文件的。

設(shè)置java.library.path

最簡單的辦法就是在啟動jvm前通過java -Djava.library.path=path-to-your-libs設(shè)置這個全局變量。

作用

那么這個地址具體是如何被使用的呢?
當(dāng)我們調(diào)用System.loadLibrary(libname)時,會調(diào)用Runtime.loadLibary,然后調(diào)用java/lang/ClassLoader.loadLibrary。在ClassLoader.loadLibrary中,系統(tǒng)屬性java.library.path將會被獲取,并用來生成需要加載的庫的絕對路徑,然后將這個絕對路徑傳給本地方法來調(diào)用dlopen/dlsym并最終加載這個庫。
如果加載失敗,會根據(jù)實(shí)際情況返回三個異常值:

SecurityException − if a security manager exists and its checkLink method doesn't allow loading of the specified dynamic library
UnsatisfiedLinkError − if the library does not exist
NullPointerException − if libname is null

可以參考OpenJDK的倉庫:

static void loadLibrary(Class fromClass, String name,
                            boolean isAbsolute) {
        ClassLoader loader =
            (fromClass == null) ? null : fromClass.getClassLoader();
        if (sys_paths == null) {
            usr_paths = initializePath("java.library.path");
            sys_paths = initializePath("sun.boot.library.path");
        }
        if (isAbsolute) {
            if (loadLibrary0(fromClass, new File(name))) {
                return;
            }
            throw new UnsatisfiedLinkError("Can't load library: " + name);
        }
        if (loader != null) {
            String libfilename = loader.findLibrary(name);
            if (libfilename != null) {
                File libfile = new File(libfilename);
                if (!libfile.isAbsolute()) {
                    throw new UnsatisfiedLinkError(
    "ClassLoader.findLibrary failed to return an absolute path: " + libfilename);
                }
                if (loadLibrary0(fromClass, libfile)) {
                    return;
                }
                throw new UnsatisfiedLinkError("Can't load " + libfilename);
            }
        }
        for (int i = 0 ; i < sys_paths.length ; i++) {
            File libfile = new File(sys_paths[i], System.mapLibraryName(name));
            if (loadLibrary0(fromClass, libfile)) {
                return;
            }
        }
        if (loader != null) {
            for (int i = 0 ; i < usr_paths.length ; i++) {
                File libfile = new File(usr_paths[i],
                                        System.mapLibraryName(name));
                if (loadLibrary0(fromClass, libfile)) {
                    return;
                }
            }
        }
        // Oops, it failed
        throw new UnsatisfiedLinkError("no " + name + " in java.library.path");
    }

LD_LIBRARY_PATH

為了搞清楚這個變量的作用,我們先說明一下Unix系統(tǒng)是如何加載動態(tài)庫的,然后自然就明白為什么要有LD_LIBRARY_PATH以及如何使用了。

動態(tài)庫如何加載?

在基于GNU glibc的系統(tǒng)上,包括所有的linux系統(tǒng),啟動一個ELF格式的二進(jìn)制可執(zhí)行文件會自動調(diào)用加載器加載必要的動態(tài)鏈接庫,一個最簡單的可執(zhí)行文件一般也會包含一些系統(tǒng)的動態(tài)庫比如libc.so等。在Linux系統(tǒng)中,這個加載器叫做/lib/ld-linux.so.X,這個X指的是加載器的版本號。加載器然后查找并加載所需的動態(tài)庫。

加載器在什么路徑中搜索和加載動態(tài)庫呢——/etc/ld.so.conf,這個文件會包括/etc/ld.so.conf.d/*.conf這些文件夾中所有的.conf文件,而具體的動態(tài)庫搜索路徑,就包含在每個.conf文件中,比如/etc/ld.so.conf.d/libc.conf,它是libc的默認(rèn)的搜索路徑/usr/local/lib,這也是為什么我們不需要顯示聲明使用系統(tǒng)庫卻能自動完成加載的原因,也是為什么不同的系統(tǒng)編出來的庫無法通用的可見原因之一,因為不同系統(tǒng)的/usr/local/lib目錄下的動態(tài)庫并不一致。

如果每次啟動都去查找所有的目錄,那樣顯然是比較笨的做法,所以使用/etc/ld.so.cache來緩存路徑,并通過ldconfig來更新這個緩存路徑,有興趣的可以自行查看一下這個緩存文件。實(shí)際上,這個緩存路徑也很長了,基本上包含了系統(tǒng)可能存放動態(tài)庫的路徑。

為什么有LD_LIBRARY_PATH?

上面我們說到可以通過cache和ldconfig來簡化搜索和加載動態(tài)庫的流程,但是還有兩個問題沒有考慮到,一是還沒有將編出來的庫放到系統(tǒng)目錄中去,二是依賴庫數(shù)量很少,不需要經(jīng)過這么復(fù)雜的查找。

LD_LIBRARY_PATH就是用來滿足這個需要,它也指定一個搜索路徑,且ld-linux.so會優(yōu)先在這個路徑下搜索需要的動態(tài)庫,如果沒找到,再去ld.so.conf中指定的目錄尋找。

使用

export LD_LIBRARY_PATH=paths-to-libs

需要注意的一點(diǎn)是,多個目錄是通過:隔開的

區(qū)別

前面分別介紹了java.library.path 和 LD_LIBRARY_PATH,都是為了加載所需的動態(tài)庫,有什么區(qū)別呢?

前者是在java環(huán)境中調(diào)用,在jvm啟動前設(shè)置生效;后者也是在啟動前,但是是在Unix環(huán)境中使用前者是通過修改property來設(shè)置路徑;后者是直接增加了ld-linux.so的搜索路徑對于JNI直接調(diào)用的庫,最好使用前者,對于有多重依賴關(guān)系的庫,最好使用LD_LIBRARY_PATH

參考

HowTo: How to configure library path for JNI dependent libraries
https://zauner.nllk.net/post/0013-jni-and-the-java-library-path/
https://docs.oracle.com/javase/8/docs/api/java/lang/System.html#getProperties–
https://www.tutorialspoint.com/java/lang/runtime_loadlibrary.htm
https://stackoverflow.com/questions/27945268/difference-between-using-java-library-path-and-ld-library-path
Linux關(guān)于動態(tài)庫的文檔

到此這篇關(guān)于解析java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別的文章就介紹到這了,更多相關(guān)java.library.path和LD_LIBRARY_PATH內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 剖析Java中阻塞隊列的實(shí)現(xiàn)原理及應(yīng)用場景

    剖析Java中阻塞隊列的實(shí)現(xiàn)原理及應(yīng)用場景

    這篇文章主要介紹了剖析Java中阻塞隊列的實(shí)現(xiàn)原理及應(yīng)用場景,這里也對阻塞和非阻塞隊列的不同之處進(jìn)行了對比,需要的朋友可以參考下
    2015-12-12
  • Spring?BeanPostProcessor后處理器源碼解析

    Spring?BeanPostProcessor后處理器源碼解析

    這篇文章主要介紹了Spring?BeanPostProcessor后處理器源碼解析,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2023-09-09
  • Java自動化測試中多數(shù)據(jù)源的切換(實(shí)例講解)

    Java自動化測試中多數(shù)據(jù)源的切換(實(shí)例講解)

    下面小編就為大家?guī)硪黄狫ava自動化測試中多數(shù)據(jù)源的切換(實(shí)例講解)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-10-10
  • Java面試重點(diǎn)中的重點(diǎn)之Elasticsearch核心原理

    Java面試重點(diǎn)中的重點(diǎn)之Elasticsearch核心原理

    ElasticSearch是一個基于Lucene的搜索引擎,是用Java語言開發(fā)的,能夠達(dá)到實(shí)時搜索,穩(wěn)定,可靠,快速,安裝使用方便,作為Apache許可條款下的開放源碼發(fā)布,是一種流行的企業(yè)級搜索引擎,是最受歡迎的企業(yè)搜索引擎
    2022-01-01
  • Spring?自定義propertyEditor的示例代碼

    Spring?自定義propertyEditor的示例代碼

    這篇文章主要介紹了Spring?自定義propertyEditor的示例代碼,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-12-12
  • Java實(shí)現(xiàn)對象復(fù)制的方法實(shí)例

    Java實(shí)現(xiàn)對象復(fù)制的方法實(shí)例

    這篇文章主要介紹了Java實(shí)現(xiàn)對象復(fù)制的方法實(shí)例,深復(fù)制:復(fù)制出來的對象中的變量(包括基本類型和字符串)和原來的對象的值都相同,引用對象也會指向復(fù)制出來的對象,需要的朋友可以參考下
    2023-08-08
  • Java如何從List中刪除元素的正確用法舉例

    Java如何從List中刪除元素的正確用法舉例

    這篇文章主要給大家介紹了關(guān)于Java如何從List中刪除元素的正確用法, 列表List是Java中的一種數(shù)據(jù)結(jié)構(gòu),存放按照添加順序的元素,是個有序的集合,需要的朋友可以參考下
    2023-07-07
  • java實(shí)現(xiàn)學(xué)生成績錄入系統(tǒng)

    java實(shí)現(xiàn)學(xué)生成績錄入系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)學(xué)生成績錄入系統(tǒng),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • java使用poi讀取excel內(nèi)容方法實(shí)例

    java使用poi讀取excel內(nèi)容方法實(shí)例

    本文介紹java使用poi讀取excel內(nèi)容的實(shí)例,大家參考使用吧
    2014-01-01
  • idea插件之mybatis log plugin控制臺sql的問題

    idea插件之mybatis log plugin控制臺sql的問題

    這篇文章主要介紹了idea插件之mybatis log plugin控制臺sql,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-09-09

最新評論