当前位置 : 主页 > 手机开发 > android >

AndroidNDK开发中SO包大小压缩方法详解

来源:互联网 收集:自由互联 发布时间:2023-02-01
目录 背景 1.STL的使用方式 2.不使用Exception和RTTI RTTI Exception 3.使用 gc-sections去除没有用到的函数 4.去除冗余代码 5.设置编译器的优化flag 6.设置编译器的 Visibility Feature 7.设置编译器的St
目录
  • 背景
  • 1.STL的使用方式
  • 2.不使用Exception和RTTI
    • RTTI
    • Exception
  • 3.使用 gc-sections去除没有用到的函数
    • 4.去除冗余代码
      • 5.设置编译器的优化flag
        • 6.设置编译器的 Visibility Feature
          • 7.设置编译器的Strip选项
            • 8.去除C++代码中的iostream相关代码
              • 总结

                背景

                这周在做Yoga包的压缩工作。Yoga本身是用BUCK脚本编译的,而最终编译出几个包大小大总共约为7M,不能满足项目中对于APK大小的限制,因此需要对它进行压缩。

                这里先将Yoga编译脚本用CMAKE重新改写,以便可以在android studio中直接使用并输出一个AAR的包。后面又对它进行了压缩,最终将Yoga包的大小压缩到200多KB。

                下面整理了一些可以用于减少NDK开发中Android SO包大小的方法:

                1.STL的使用方式

                对于C++的library,引用方式有2种:

                • 静态方式(static)
                • 动态方式(shared)

                其中,静态方式在编译时会将用到的相关代码直接复制到目的文件中;而动态方式则会将相关的代码打成so文件,以便多次引用。由于编译器在编译时并不能知道所有被引用的地方,所以同时会打入了很多不相关的代码。

                所以,如果项目中引用library的函数较多时,用动态方式可以避免多次拷贝,节省空间。相反,则直接使用静态方式会更节省空间。

                NDK开发中,可以通过gradle的设置来配置:

                defaultConfig{
                  externalNativeBuild{
                    cmake{
                      // gnustl_shared 动态
                      arguments "-DANDROID_STL=gnustl_static" 
                    }
                  }
                }

                在Yoga中,项目里的stl使用较少时,安卓运行时使用static的方式,而不是shared,所以这里采用static的方式。在采取了这种方式后,包的大小从2.7M缩减到了2M。

                2.不使用Exception和RTTI

                C++的exception和RTTI功能在NDK中默认是关闭的,但是可以通过配置打开的。

                Android.mk:

                APP_CPPFLAGS += -fexceptions -frtti
                

                CMake:

                set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fexceptions -frtti")
                

                Exception和RTTI会显著的增加包的体积,所以非必须的时候,没有必要使用。

                RTTI

                通过RTTI,能够通过基类的指针或引用来检索其所指对象的实际类型,即运行时获取对象的实际类型。C++通过下面两个操作符提供RTTI。

                (1)typeid:返回指针或引用所指对象的实际类型。

                (2)dynamic_cast:将基类类型的指针或引用安全的转换为派生类型的指针或引用。

                在yoga中,RTTI的选项是默认打开的,而代码中其实并没有用到相关的功能,这里可以直接关闭。

                Exception

                使用C++的exception会增加包的大小,而目前JNI对C++的exception的支持是有bug的,比如下面这段代码就会引起程序的crash(对于低版本的android NDK)。

                因此要在程序中引入exception要自己实现相关逻辑,yoga就是这么做的,这个又增加了一些包体大小。对于开发者来说,exception可以帮助快速定位问题,而对于使用者并不是那么重要,这里可以去掉。

                try {
                    ...
                } catch (std::exception& e) {
                    env->ThrowNew(env->FindClass("java/lang/Exception"), "Error occured");
                }

                在yoga中,在关闭RTTI和Exception功能并把exception相关的代码都去掉后,包的大小从2M缩减到的1.8M。

                3.使用 gc-sections去除没有用到的函数

                去除未使用的代码显然可以减少包体的大小,而在NDK的开发中,并不需要手动的来做这一点。可以开启编译器的gc-sections选项,让编译器自动的帮你做到这一点。

                编译器可以配置自动去除未使用的函数和变量,以下是配置方式:

                CMake:

                # 去除未使用函数与变量
                set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -ffunction-sections -fdata-sections")
                set(CMAKE_CXX_FLAGS "${CMAKE_C_FLAGS}")
                # 设置去除未使用代码的链接flag
                SET_TARGET_PROPERTIES(yoga PROPERTIES LINK_FLAGS "-Wl,--gc-sections")
                

                Android.mk:

                LOCAL_CPPFLAGS += -ffunction-sections -fdata-sections
                LOCAL_CFLAGS += -ffunction-sections -fdata-sections 
                LOCAL_LDFLAGS += -Wl,--gc-sections
                

                4.去除冗余代码

                在NDK中,链接器还有一个选项 “-icf = safe”,可以用于去除代码中的冗余代码。但是要注意的是,这个选项也有可能去除定义好的inline函数,这里必须要做好权衡。

                下面是配置方式:

                CMake:

                SET_TARGET_PROPERTIES(yoga PROPERTIES LINK_FLAGS "-Wl,--gc-sections,--icf=safe")
                

                Android.mk:

                LOCAL_LDFLAGS += -Wl,--gc-sections,--icf=safe
                

                5.设置编译器的优化flag

                编译器有个优化flag可以设置,分别是-Os(体积最小),-O3(性能最优)等。这里将编译器的优化flag设置为-Os,以便减少体积。

                CMake:

                set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Os")
                set(CMAKE_CXX_FLAGS "${CMAKE_C_FLAGS}")
                

                Android.mk

                LOCAL_CPPFLAGS += -Os
                LOCAL_CFLAGS += -Os

                在采用了3,4,5这几种方式后,Yoga包的大小从1.8M减少到了1.7M。这里减少的比较少是因为Yoga在这方面已经做的挺好了,其他的库可能会更有效。

                6.设置编译器的 Visibility Feature

                还有个减少包体大小的方法,就是设置编译器的visibility feature。

                Visibility Feature就是用来控制在哪些函数可以在符号表中被输入,由于C++并不是完全面向对象的,非类的方法并没有public这种修饰符,因此,要用Visibility Feature来控制哪些函数可以被外部调用。

                而JNI提供了一个宏-JNIEXPORT来控制这点。所以只要对函数加上这个宏,像这样:

                // JNIEXPORT就是控制可见的宏
                // JNICALL在NDK这里没有什么意义,只是个标识宏
                JNIEXPORT void JNICALL Java_ClassName_MethodName(JNIEnv *env, jobject obj, jstring javaString)
                

                然后在编译器的FLAGS选项开启 -fvisibility = hidden 就可以。这样,不仅可以控制函数的可见性,并且可以减少包体的大小。

                CMake:

                set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fvisibility=hidden")
                set(CMAKE_CXX_FLAGS "${CMAKE_C_FLAGS}")
                

                7.设置编译器的Strip选项

                我在把Yoga库编译成AAR包的过程中发现,它的体积明显会大于最后打包进APK的大小,这点非常不合理,但是无法找到原因。

                最终搜索到这是谷歌NDK的一个bug,在打AAR包的过程中,无论是debug版本还是release版本,NDK toolchain不会自动的把方便调试的C++ 符号表(Symbol Table)中数据删除,而只会在打APK包的时候进行这一操作。这就导致了打成的AAR包中的SO体积明显偏大。

                找到原因后这个问题就很好解决了,可以手动的在链接选项中加入 strip参数,配置如下所示:

                SET_TARGET_PROPERTIES(yoga PROPERTIES LINK_FLAGS "-Wl,--gc-sections,--icf=safe,-s")
                

                强制进行strip操作后,将Yoga包的体积从1.7M成功减少到了282KB。

                8.去除C++代码中的iostream相关代码

                使用STL中的iostream相关库会明显的增加包的体积,而NDK本身是有预编译库(android/log.h)可以代替这一功能的,在Yoga这里,用log的函数代替了iostream中的所有函数,如:

                //代替所有的iostream库里函数
                //cout << obj->toString() << endl;
                __android_log_print(ANDROID_LOG_VERBOSE,"Yoga","Node is: %s",obj->toString().c_str());
                

                在做完代替之后,yoga包的体积从282KB减少到了218KB。

                总结

                在做完这一系列工作后,最终成功的压缩了Yoga包的体积,从几M到最后输出一个218KB的AAR包提供使用。以上几种方法并不局限于Yoga包的缩减。在NDK开发中,要缩减SO包的体积都可以按照这几种方式尝试一下。

                以上就是Android NDK 开发中 SO 包大小压缩方法详解的详细内容,更多关于Android NDK开发SO包压缩的资料请关注自由互联其它相关文章!

                上一篇:ViewPager实现轮播图引导页
                下一篇:没有了
                网友评论