轻松实现动态获取Android手机CPU架构类型

2018-12-13 17:17:02

1、什么是动态库(.so文件)?

  .so文件是unix的动态连接库,是二进制文件,作用相当于windows下的.dll文件。

  他使用了C/C++代码编写的可以操作硬件比java更高级的

  底层代码,执行速度和效率比其他语言要高。

  在Android中调用动态库文件(*.so)都是通过jni的方式。


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

  void System.load(String pathName);


  说明:pathName:文件名+文件路径;

  void  System.loadLibrary("libraryName");


  说明:libraryName:动态库的名字,是原始动态库去掉前面的lib和.so后的名字,如:

  “libnative-lib.so”库,要加载是应写成“native-lib”,

  即System.loadLibrary("native-lib"),

  而不能是System.loadLibrary("libnative-lib.so");


2、为什么需要动态库?

  有时候原生的Java代码编写的程序不能满足需求时,如计算量很大,性能要求高,

  常见于游戏开发,我们就需要考虑使用C/C++开发程序,然后通过JNI的方式

  来调用。


3、加载动态库能做什么?

    1)、能直接操作硬件,如摄像头,音视频编码和解码;

    2)、更安全,so库是二进制的文件,不容易破解,人无法看懂,代码安全度更高。

    3)、执行效率高,运算速度快,C/C++编写的程序可以直接操作内存。

    4)、Android中提供的好多java接口调用的API底层都是通过JNI的方式来调用,

    如bitmap的压缩。


 5、Android开发为什么需要NDK开发,什么是CPU架构类型?

  有时候我们需要动态获取Android手机的cpu型号,这种需求不常见;

  Android系统中做app开发很多时候我们会集成三方的sdk,少不了因为适配而提供给不

  同架构cpu的动态库(.so文件),比方说分型类,即时通信类, Android系统在加载app

  中的动态库的时候,会检查对应的cpu架构的型号,然后在到对应的lib目录下加载对应

  的库,这是为什么,应为Android底层是用C/C++实现的,生成不同架构的cpu可以加载

  的库时使用的指令不一样(体现在ndk编译生成多个版本的.so文件),如:

-------------------------------------------------

  arm64-v8a

  armeabi-v7a

  armeabi

  mips

  mips64

  x86

  x86_64

-------------------------------------------------

  当某个手机加载的库不是当前手机cpu对应的指令编译出来的.so文件见时,程序就会异常

  ,所以为了保证程序的稳定性,必须保证动态库编译时使用的指令和当前手机cpu可支持的

  cpu执行对应,这样才不会造成程序异常的情况。


6、为甚需要动态获取CPU架构类型;

  当我们在集成三方sdk是,由于有多个版本的.so库,如果同时导入我们的工程,那么造成

  的后过就是我们生成的app的体积瞬间会暴涨,比如环信即时通信IM的动态库,我查看了一下

  ,每个架构的cpu对应的.so动态库总大小都是12M以上,那么可想而知,如果我们导入了3个

  不同架构的动态库,瞬间app就增加了30多M,还不算我们应用自己的大小,对有些用户来说

  在应用市场上下载app第一反应都会看app的大小,至少对我自己来说是这样,如果某些应用的

  app超过30M的样子我就不想下载了,所用app包的大小直接关系到用户的体验,除了支付宝,淘

  宝,京东必须的软件为,基本都会考虑是不是放弃下载。


  为了解决三方库对本应用包的大小营销,我已我们可以考虑把不懂版本的动态库压缩放到我们的服

  务器上,让后在用户运行我们app的时候动态的获取用户手机CPU的架构类型,然后从服务器下载对

  应的动态库到SDCar上,Android也提供了加载外部的动态库的API,这样的话我们app的大小就会减

  少二分之一以上,另一个好处就是当我们把对应的库压缩有,体积还会减少到原来库大小的1/3,这

  样的话我们单从加载三库增加的大小来讲,如果原来有3个架构库,库总大小为30M,单个大小为10M,

  使用压缩的方式就会减少的单个大小的1/3,即3M多不到4M的样子,姑且按4M来算吧,而如果全部打包

  进app,总大小会在30M,即使压缩过了还是比较大,如果采用压缩的话,就会是原来的30/4四舍五入

  与等于7,即大小减少到原来的1/7的大小,这样的话就大大减少了app的大小,同时也不影响程序的运

  行。


  一下我我分装的库,使用了ndk的方式,其他方式目前我还没发发现,检测cpu的代码是从Android底层

  源码中找出来的,然后经过自己封装,通过JNI的方式来完成调用:使用很方便,如下;


  1、导入不同架构的.so库:架构包括

-----------------------------------

    arm64-v8a

    armeabi-v7a

    armeabi

    mips

    mips64

    x86

    x86_64

-----------------------------------

  每个文件夹下都有一个libnative-lib.so文件。


2、导入调用so库的jar包,并添加到lib中:

    cpu.jar


  注意:需要在gradle中的Android标签下加入:


   // 设置Jni so文件路径

      sourceSets {

          main{

              jniLibs.srcDirs = ['libs']

          }

      }



3、调用:

    通过CPUFrameworkHelper类来完成调用,如:CPUFrameworkHelper.isArmCpu(),会返回一个

    boolean值



  注意:

  可以直接集成的库地址:

  项目下载地址在:https://github.com/pgyszhh/CPUTypeHelper/tree/master/mycpu

      里边有需要的各个版本的.so库和cpu.jar的包,直接导入就可以使用。



  库源码地址:

      编译动态库和JIN的源码地址为:https://github.com/pgyszhh/CPUTypeHelper下的app目录,该

      目录下是所有的源码,





  在CPUFrameworkHelper类中提供的方法有:


      public static native boolean isArmCpu();


      public static native boolean isArm7Compatible();


      public static native boolean isMipsCpu();


      public static native boolean isX86Cpu();


      public static native boolean isArm64Cpu();


      public static native boolean isMips64Cpu();

  

      public static native boolean isX86_64Cpu();

--------------------- 

作者:violetten 

来源:CSDN 

原文:https://blog.csdn.net/u010712703/article/details/71194881 

版权声明:本文为博主原创文章,转载请附上博文链接!

  • 2019-11-28 11:00:35

    Vue子组件调用父组件的方法

    下面有三种方法,我自己重点推荐第一种,毕竟这种简单粗暴好用好理解,不过这个有一个弊端,再组件嵌套组件的时候,尤其是用第三方组件里面调用自己的子组件的时候,其实已经是孙子组件了,这个时候就要parent.parent。。。。,这样就不好了,我们就得考虑其他方法了,具体怎么判断是父组件,还是爷爷组件,我会单独出一篇文章讲述。

  • 2019-11-29 13:04:47

    计算一个多边形的重心点坐标(准确版)

    在之前的 《如何判断一个多边形是否合法》 一文中有提到,用无人机规划飞行路线前,往往需要框选一个多边形的区域。 而在地图控件上显示这个多边形区域时,往往会遇到这样一个需求:需要把所要测绘的多边形区域移动到地图中心。 实现这个需求的基本思路就是:获取到多边形区域的重心点坐标,然后利用地图控件的 setCenter方法,就可以把地图的显示中心移动到多边形区域重心了。那么问题来了,如何求出一个多边形的重心点坐标呢?

  • 2019-11-29 13:06:27

    如何判断一个多边形是否合法

    利用无人机对一片区域进行测绘前,我们会先在地图上框选一个区域,然后再规划飞行的路线,而需要测绘的这片区域往往是一个多边形。在 MeshKit 中,我们加入了多边形区域的编辑功能,其中就涉及判断用户所编辑出来的多边形是否合法的问题。

  • 2019-11-29 13:47:22

    百度地图做电子围栏总结

    在地图上画出围栏,设置围栏信息后保存,生成围栏列表。全选时,地图视野可看到全部的围栏区域,单独勾选会调整地图视野到当前勾选的围栏。围栏区域的中心点要显示围栏名称。

  • 2019-11-29 13:50:29

    图片连接处出现白线

    block导致,只要父元素设置font-size:0或者设置img display: block; 便可。但是我设置了没有用,这条线不是所有的机型都有,而且页面滚动之后又消失,我琢磨半天,各种尝试,发现把图片高度减少(增加)1px就解决了。因为我们的项目是用postcss-px-to-viewport,我每张图片都是设置高度的,应该是数值转换出现偏差。

  • 2019-11-29 13:54:07

    粗略计算多边形中心点(并不是很准确,但简单好用)

    也是再做栅栏系统,搜索如何获取多边形中心点的问题上,发现了这个,简单易于理解,但是并不是特变准确,但也不影响使用。 后来发现了新的算法,并且百度地图也提供相应的api。 具体内容我写在了前面的文章,大家可以找一下。

  • 2019-11-29 14:20:38

    vue,vuthis.$parent算法

    由于组件嵌套,其实vue parent的位置也改变了,我们可以通过下面的图片,来看一下,parent到底什么哪一层