ndk調用so動態庫 編譯另外一個so庫

調用動態庫,先加web

#include<dlfcn.h>  //用於動態庫管理的系統頭文件 shell

(1)dlopen()
第一個參數:指定共享庫的名稱,將會在下面位置查找指定的共享庫.
ide

第二個參數:指定如何打開共享庫。
-RTLD_NOW:將共享庫中的全部函數加載到內存
-RTLD_LAZY:會推後共享庫中的函數的加載操做,直到調用dlsym()時方加載某函數函數

(2)dlsym()
調用dlsym時,利用dlopen()返回的共享庫的phandle以及函數名稱做爲參數,返回要加載函數的入口地址。spa

(3)dlerror()
該函數用於檢查調用共享庫的相關函數出現的錯誤。  返回是string字符串code

(4)dlclose() 
該函數用於關閉動態庫。orm

#include <string.h>  
#include <jni.h>  
#include <dlfcn.h>
jstring  
Java_org_crazyit_helloworld_MainActivity_stringFromJNI( JNIEnv* env,  
                                                  jobject thiz )  
{  
    return (*env)->NewStringUTF(env, "Hello world ");  
} 
//這樣寫,在動態加載時才能獲取到函數地址 
//第一次編譯hello.so的 plus代碼
/*


jint plus(jint a,jint b)
{
 return a+b;
}

 jint Java_org_crazyit_helloworld_MainActivity_plus(JNIEnv* env, jobject obj, jint a, jint b)  
{  
    return plus(a,b);
} 
*/
//第二次編譯加載代碼

 jint Java_org_crazyit_helloworld_MainActivity_test(JNIEnv* env, jobject obj, jint a, jint b)
{
  void*  filehandle = dlopen("/data/data/org.crazyit.helloworld/lib/libhello.so", RTLD_LAZY );
  jint sum = 123;
  // 打開原so文件
  if(filehandle != NULL)
  {
   sum = 345;
   int(* oldmethod)(int,int);
   oldmethod= (int(*)(int,int))dlsym(filehandle, "plus");
   //引入原so中的函數
   if(oldmethod != NULL)
   {
    sum = 6*oldmethod(a,b);
   }
  }
    return sum;
}

獲取方法的另外一種寫法對象

typedef void(* Tx_set_video_data_Func)(unsigned char *, int, int, int, int, int, int, int);
Tx_set_video_data_Func oldmethod= (Tx_set_video_data_Func)dlsym(filehandle, "tx_set_video_data");
//引入原so中的函數
if(oldmethod != NULL)
{
}

2,編譯代碼 Android.mk內存

加上動態庫的鏈接 LOCAL_LDLIBS := -L . -ldl 字符串

-l參數就是用來指定程序要連接的庫 .表明當前目錄下的庫

-ldl選項,表示生成的對象模塊須要使用共享庫

LOCAL_PATH := $(call my-dir)  
LOCAL_LDLIBS := -L . -ldl
  
include $(CLEAR_VARS)  
  
LOCAL_MODULE    := hello  
LOCAL_SRC_FILES := hello-jni.c  
  
include $(BUILD_SHARED_LIBRARY)

第一個plus的hello.so與test.so放在lib目錄下,運行MainActivity代碼

static {  
   System.loadLibrary("test");  
} 
//進行加法操做  
public native int test(int a,int b);

若是找不到要導入的庫:shell裏運行

export LD_LIBRARY_PATH=./lib:$LD_LIBRARY_PATH

NDK編譯錯誤

toolchains incompatible target libtx.a(dev.o)

編譯平臺不一樣致使的 如要把x86的 dev.o編譯到 arm平臺下,改變 Application.mk下的平臺選擇 x86

Application.mk 寫不一樣的平臺就會調用不一樣平臺下的toolchains 交叉編譯鏈

2,靜態庫 .a包含了全部.o文件    ar 進行 拆包     ar -x libdev.a  把全部的so解出來

靜態庫/動態庫中函數查看

1,nm -g libdev.so 顯示出 so所依賴的so庫和其中的一些函數方法

2,readelf -a libdev.so 顯示so的平臺等之類的信息

相關文章
相關標籤/搜索