android apk加载.so库和android apk调用C语言可执行程序

来源:互联网 发布:预测性数据分析 编辑:程序博客网 时间:2024/06/06 03:04

1.在Android中调用动态库文件(*.so)都是通过jni的方式,而且往往在apk或jar包中调用so文件时,都要将对应so文件打包进apk或jar包,工程目录下图:


以上方式的存在的问题:

  1、缺少灵活性比较类似静态加载了(不是静态加载),能加载的so文件绑定死了;

  2、但so文件很多或很大时,会导致对应的apk和jar包很大;

  3、不能动态的对so文件更新;

 

步骤:1,在项目根目录下建立文件夹libs/armeabi文件夹。最好文件夹名是armeabi,换了别的名字的话,试了一下,装上之后在data/data/程序/lib下是没有的

 2,so库放入libs/armeabi文件夹注意事项: 1,如果采用静态注册的方式请注意C文件中严格按照命名规则Java_packageName_className_method()的方式命名 2,Android项目中建立同上述命名规则中packageName中相同的包名,在此包名下建立同上述命名规则中className相同的类名 3,className声明native方法 4,程序中加载soSystem.loadLibrary(data/data/xxx.xxx.xxx/lib/xx.so)或者System.loadLibrary(xx)。这里的xx一定是方法名,即method(),不是类名或者文件名。


Android中加载so文件的提供的API:

void System.load(String pathName);

 但是有一点,pathName路径必须有执行权限,意思就是说我们不能加载SD卡上的SO,因为没有执行权限

说明:

  1、pathName:文件名+文件路劲;

  2、该方法调用成功后so文件中的导出函数都将插入的系统提供的一个映射表(类型Map);


看到以上对System.load(StringpathName);的函数说明可定有人会想到将so文件放到一个指定的目录然后再通过参数pathName直接引用该目录的路劲和对应的so文件问题不就解决了吗?

这里有个问题被忽略了,那就是System.load只能加载两个目录路劲下的so文件:

  1、/system/lib ;

  2、安装包的路劲,即:/data/data/<packagename>/…

2.apk调用C语言可执行程序

  APK中集成和使用 C 可执行程序

    怎样在APK中集成和使用一个 C 可执行程序呢?遵循下列步骤即可:

  1.     可以把可执行程序放在 assets 目录下,这样打包成 APK 时会自动打包
  2.     APK 运行时访问 assets 文件夹内的资源,释放可执行程序,添加可执行权限
  3.     使用 Runtime.exec() 启动可执行程序
private void copyXX() {String fileDir="data/data/XX";String fileName="XX";File file = new File(fileDir,fileName );    if (file.exists()) {        Log.i("log:copyXX", "copy XX");    } else {        Log.i("log:copyXX", "XX exits");                 // copy        try {            InputStream is = getAssets().open("XX");    // 获取数据库库文件输入流            FileOutputStream fos = new FileOutputStream(file);  // 定义输出流            byte[] bt = new byte[8192];            int len = -1;            while((len = is.read(bt)) != -1){                fos.write(bt, 0, len);            }            is.close();            fos.close();                     } catch (IOException e) {            e.printStackTrace();        }    }}
3.
Runtime runtime = Runtime.getRuntime();
//修改权限runtime.exec("chmod 777 data/data/apk程序/可执行程序");
runtime.exec("data/data/apk程序/./可执行程序+参数");




下面是runtime的操作资料,可以得到一些其他的东西,如可执行程序的权限:

Java编写应用时,有时需要在程序中调用另一个现成的可执行程序或系统命令,这时可以通过组合使用Java提供的Runtime类和Process类的方法实现。下面是一种比较典型的程序模式:   
...   

Process   process   =  Runtime.getRuntime().exec( ".//p.exe ");   
process.waitfor(   );   
...   
在上面的程序中,第一行的“.//p.exe”是要执行的程序名,Runtime.getRuntime()返回当前应用程序的Runtime对象,该对象的exec()方法指示Java虚拟机创建一个子进程执行指定的可执行程序,并返回与该子进程对应的Process对象实例。通过Process可以控制该子进程的执行或获取该子进程的信息。第二条语句的目的等待子进程完成再往下执行。   
但在windows平台上,如果处理不当,有时并不能得到预期的结果。下面是笔者在实际编程中总结的几种需要注意的情况:   
1
、执行DOS的内部命令   
如果要执行一条DOS内部命令,有两种方法。一种方法是把命令解释器包含在exec()的参数中。例如,执行dir命令,在NT上,   可写成exec( "cmd.exe   /c   dir"),在windows  95/98下,可写成“command.exe  /c   dir”,其中参数“/c”表示命令执行后关闭Dos立即关闭窗口。另一种方法是,把内部命令放在一个批命令my_dir.bat文件中,在Java程序中写成exec( "my_dir.bat ")。如果仅仅写成exec("dir ")Java虚拟机则会报运行时错误。前一种方法要保证程序的可移植性,需要在程序中读取运行的操作系统平台,以调用不同的命令解释器。后一种方法则不需要做更多的处理。   
2
、打开一个不可执行的文件   
打开一个不可执行的文件,但该文件存在关联的应用程序,则可以有两种方式。  以打开一个word文档a.doc文件为例,Java中可以有以下两种写法:   
exec( "start   .//a.doc ");  

exec( "   c://Program  Files//Microsoft   Office//office//winword.exe   .//a.doc ");  
显然,前一种方法更为简捷方便。  
3
、执行一个有标准输出的DOS可执行程序   
windows平台上,运行被调用程序的DOS窗口在程序执行完毕后往往并不会自动关闭,从而导致Java应用程序阻塞在waitfor(   )。导致该现象的一个可能的原因是,该可执行程序的标准输出比较多,而运行窗口的标准输出缓冲区不够大。解决的办法是,利用Java提供的Process类提供的方法让Java虚拟机截获被调用程序的DOS运行窗口的标准输出,在waitfor()命令之前读出窗口的标准输出缓冲区中的内容。一段典型的程序如下:   
...   

String   ls_1;   
Process   process   =  Runtime.getRuntime().exec( "cmd   /c   dir   //windows");   
BufferedReader   bufferedReader   =  new   BufferedReader(   /   
new   InputStreamReader(process.getInputStream());  
while   (  (ls_1=bufferedReader.readLine())   !=   null)   
System.out.println(ls_1);   
     
process.waitfor(   );   





0 0