动态使用共享库函数

dll_main   

 

环境介绍

续上节代码

目录结构:

 

android.mk如下:

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := demo

LOCAL_SRC_FILES := mod1.cpp mod2.cpp mod3.cpp

include $(BUILD_SHARED_LIBRARY)

include $(CLEAR_VARS)

LOCAL_MODULE    := Hello

LOCAL_SRC_FILES := Hello.cpp

include $(BUILD_EXECUTABLE)

 

Hello.cpp

#include <stdio.h>

#include <dlfcn.h>

typedef void (*FUNTYPE)();

int main(int argc, char* argv[])

{

    //加载共享库

    void *handle = dlopen("/data/local/tmp/libdemo.so", RTLD_NOW);

    if (handle == NULL)

    {

        puts(dlerror());

        return 0;

    }

    printf("handle=%p\n", handle);

    //获取导出函数

    FUNTYPE pfnMod = (FUNTYPE)dlsym(handle, "_Z4mod1v");

    if (pfnMod != NULL)

    {

        printf("address=%p\n", pfnMod);

        pfnMod();

    }

    pfnMod = (FUNTYPE)dlsym(handle, "_Z4mod2v");

    if (pfnMod != NULL)

    {

        printf("address=%p\n", pfnMod);

        pfnMod();

    }

    pfnMod = (FUNTYPE)dlsym(handle, "_Z4mod3v");

    if (pfnMod != NULL)

    {

        printf("address=%p\n", pfnMod);

        pfnMod();

    }

    pfnMod = (FUNTYPE)dlsym(handle, "mod4");

    if (pfnMod == NULL)

    {

        puts(dlerror());

    }

    dlclose(handle);

    return 0;

}

 

_Z4mod2v 是C++的名称粉碎

函数名可以用readelf来进行查看:

该工具在: E:\Android\android-ndk-r10b\toolchains\x86-4.6\prebuilt\windows-x86_64\bin\i686-linux-android-readelf.exe

 

 

类似于Windows动态调用dll的思想

dlopen打开一个so文件

dlsym根据函数名拿到函数指针

 

编译后使用makefile执行

在工程目录根下新建makefile:

MODALE_NAME := Hello

# x86 path

X86_TOOLS_PATH :=E:\Android\android-ndk-r10b\toolchains\x86-4.6\prebuilt\windows-x86_64\bin

X86_GDB_PATH := $(X86_TOOLS_PATH)\i686-linux-android-gdb.exe

X86_GDB_SERVER := E:\Android\android-ndk-r10b\prebuilt\android-x86\gdbserver\gdbserver

# arm-linux-androideabi-4.6 path

arm_tools_path :=E:\Android\android-ndk-r10b\toolchains\arm-linux-androideabi-4.6\prebuilt\windows-x86_64\bin

arm_4_6_path := $(arm_tools_path)\arm-linux-androideabi-gdb.exe

arm_gdb_server :=E:\Android\android-ndk-r10b\prebuilt\android-arm\gdbserver\gdbserver

run_arm:

    adb push .\libs\armeabi-v7a\$(MODALE_NAME) /data/local/tmp

    adb shell chmod 755 /data/local/tmp/$(MODALE_NAME)

    adb shell /data/local/tmp/$(MODALE_NAME)

run_x86:

    adb push .\libs\x86\$(MODALE_NAME) /data/local/tmp

    adb shell chmod 755 /data/local/tmp/$(MODALE_NAME)

    adb shell /data/local/tmp/$(MODALE_NAME)

run_x86_share:

    adb push .\libs\x86\$(MODALE_NAME) /data/local/tmp

    adb push .\libs\x86\libdemo.so /data/local/tmp

    adb shell chmod 755 /data/local/tmp/$(MODALE_NAME)

    adb shell /data/local/tmp/$(MODALE_NAME)

debug_x86:

    adb forward tcp:12345 tcp:12345

    adb push $(X86_GDB_SERVER) /data/local/tmp

    adb shell chmod 777 /data/local/tmp/gdbserver

    adb push .\obj\local\x86\$(MODALE_NAME) /data/local/tmp

    adb shell chmod 777 /data/local/tmp/$(MODALE_NAME)

    adb shell /data/local/tmp/gdbserver :12345  /data/local/tmp/$(MODALE_NAME)

client_x86:

    $(X86_GDB_PATH) .\obj\local\x86\$(MODALE_NAME)

# 1. target remote localhost:12345

# 2. gdb.setup  

 

 

make run_x86_share 即可成功执行

mod4 会提示找不到

 

 

DLL_MAIN

修改mod1.cpp

#include <stdio.h>

// 初始化函数

void   _init()

{

  printf("_init\r\n");

}

// so卸载函数

void   _fini()

{

  printf("_fini\r\n");

}

// 新版本初始化函数

void __attribute__((constructor))  OnLoad()

{

  printf("OnLoad\r\n");

}

void __attribute__((destructor))  UnLoad()

{

  printf("UnLoad\r\n");

}

void __attribute__((constructor))  OnLoad2()

{

  printf("OnLoad2\r\n");

}

void __attribute__((destructor))  UnLoad2()

{

  printf("UnLoad2\r\n");

}

//隐藏函数

void __attribute__((visibility("hidden")))  mod1()

{

  printf("mod1\r\n");

}

 

运行效果:

说明:

1. _init函数比构造函数来的早

2. hidden后在用dlsym函数无法找到

 

总结

1.相关函数

dlopen()函数打开一个共享库

dlsym()函数在库中搜索一个符号

dlclose() 函数光比之前dlopen打开的库

dlerror() 函数返回一个错误消息的字符串

 

2.隐藏函数

Void __attribute__ ((visibility("hidden"))) fun() {}

 

3. so构造析构(会在so加载和卸载的时候调用)

   void  __attribute__ ((constructor)) Load()

   void  __attribute__ ((destructor)) UnLoad()

 

4. _init()和_fini()函数

会在so加载和卸载时调用

 

 

 

ndk学习9: 动态使用共享库的更多相关文章

  1. Android NDK 交叉编译C++代码生成.so共享库详细步骤

    Android NDK 交叉编译C++代码生成.so共享库详细步骤 Android NDK 调用c++ stl 模板库(修改android.mk文件) 1  在需要调用模板库的文件前包含头文件:   ...

  2. Android JNI和NDK学习(03)--动态方式实现JNI(转)

    本文转自:http://www.cnblogs.com/skywang12345/archive/2013/05/23/3092491.html 前面总结了静态实现JNI的方法,本文介绍如何动态实现J ...

  3. Linux学习笔记——如何使用共享库交叉编译

    0.前言     在较为复杂的项目中会利用到交叉编译得到的共享库(*.so文件).在这样的情况下便会产生下面疑问,比如:     [1]交叉编译时的共享库是否须要放置于目标板中,假设须要放置在哪个文件 ...

  4. Android NDK生成共享库和静态库

    Date: 2014-03-14 Title: Compile Android Native Binary And Library Published: true Type: post Tags: A ...

  5. Android JNI和NDK学习(04)--NDK调试方法(转)

    本文转自:http://www.cnblogs.com/skywang12345/archive/2013/05/23/3092812.html 本文主要介绍在ndk中添加log的方法.然后,我们就可 ...

  6. ndk学习8: 编译动态库

    目录: 手工编译动态库 ndk-build编译动态库(Eclipse环境)   手工编译静态库 老规矩还是先手工操作,知其然并知其所以然   需要用到的核心命令: gcc -g -c -fpic -W ...

  7. 《CMake实践》笔记三:构建静态库(.a) 与 动态库(.so) 及 如何使用外部共享库和头文件

    <CMake实践>笔记一:PROJECT/MESSAGE/ADD_EXECUTABLE <CMake实践>笔记二:INSTALL/CMAKE_INSTALL_PREFIX &l ...

  8. ndk学习20: jni之OnLoad动态注册函数

    一.原理 当在系统中调用System.loadLibrary函数时,该函数会找到对应的动态库, 然后首先试图找到"JNI_OnLoad"函数,如果该函数存在,则调用它 JNI_On ...

  9. Linux共享库、静态库、动态库详解

    1. 介绍 使用GNU的工具我们如何在Linux下创建自己的程序函数库?一个“程序函数库”简单的说就是一个文件包含了一些编译好的代码和数据,这些编译好的代码和数据可以在事后供其他的程序使用.程序函数库 ...

随机推荐

  1. Haproxy安装配置及日志输出问题

    简介: 软件负载均衡一般通过两种方式来实现:基于操作系统的软负载实现和基于第三方应用的软负载实现.LVS就是基于Linux操作系统实现的一种软负载,HAProxy就是开源的并且基于第三应用实现的软负载 ...

  2. JSP的原理

    一.什么是JSP JSP全称是Java Server Pages,它和servle技术一样,都是SUN公司定义的一种用于开发动态web资源的技术. JSP这门技术的最大特点在于,写JSP就行html, ...

  3. verilog阻塞与非阻塞的初步理解(二)

    将阻塞模块改为下述代码: module blocking(clk,a,b,c); :] a; input clk; :] b,c; :] b,c; always @(posedge clk) begi ...

  4. java中InvocationHandler 用于实现代理。

    以下的内容部分参考了网络上的内容,在此对原作者表示感谢! Java中动态代理的实现,关键就是这两个东西:Proxy.InvocationHandler,下面从InvocationHandler接口中的 ...

  5. AndroidStudio-使用Translations Editor

    前言 如果你的App支持多语言,你需要正确的管理你的翻译字符串资源.Android Studio提供了翻译编辑器使更容易的查看和管理翻译资源. 关于翻译编辑器 翻译资源存储工程的多个目录下的多个XML ...

  6. JavaScript内置对象与原型继承

    (一)   理解JavaScript类定义 1>关于内置对象理解 console.log(Date.prototype.__proto__===Object.prototype    //tru ...

  7. linux下文件结束符

    linux下文件结束符,我试过了所有的linux,发现其文件的结束符都是以0a即LF结束的,这个是操作系统规定的,windows下是\r\n符结束,希望可以帮助大家. -------------转:来 ...

  8. Vue 入门指南 JS

    Vue 入门指南 章节导航 英文:http://vuejs.org/guide/index.html 介绍 vue.js 是用来构建web应用接口的一个库 技术上,Vue.js 重点集中在MVVM模式 ...

  9. php开发中怎么获取服务端MAC地址?

    MAC(Media Access Control或者Medium Access Control)地址,意译为媒体访问控制,或称为物理地址.硬件地址,用来定义网络设备的位置.在php中如何获取MAC(M ...

  10. [译]git fetch

    git fetch从远程仓储导入commit到你的本地仓储. 这些fetch到的commit是做为一个远程分支存储在你本地的. 这样你可以在集成这些commit到你的项目前先看看都有些什么修改. 用法 ...