APK 瘦身

news2024/11/13 11:28:04

APK 瘦身的主要原因是考虑应用的下载转化率和留存率,应用太大了,用户可能就不下载了。再者,因为手机空间问题,用户有可能会卸载一些占用空间比较大的应用,所以,应用的大小也会影响留存率。

1 APK 的结构

包含以下目录:

  • assets/:包含了应用的资源,这些资源能够通过 AssetManager 对象获得;
  • lib/:包含了针对处理器层面的被编译的代码。这个目录针对每个平台类型都有一个子目录,比如 armeabi,armeabi-v7a,arm64-v8a,x86,x86_64 和 mips;
  • res/:包含了没被编译到 resources.arsc 的资源;
  • META-INF/:包含 CERT.SF 和 CERT.RSA 签名文件,也包含了 MANIFEST.MF 文件;

包含以下文件:

  • classes.dex:包含了能被 Dalvik/Art 虚拟机理解的 .dex 文件格式的类;
  • resources.arsc:包含了被编译的资源。该文件包含了 res/values 目录的所有配置的 xml 内容。打包工具将 xml 内容编译成二机制形式并压缩。这些内容包含了语言字符串和 styles,还包含了那些内容虽然不直接存储在 resources.arsc 文件中,但是给定了该内容的路径,比如布局文件和图片,所以又叫资源映射表;
  • AndroidManifest.xml:包含了主要的 Android 配置文件。这个文件列出了应用名称、版本、访问权限、引用的库文件。该文件使用二进制的 .xml 格式存储;

主要是针对 lib/ 、res/ 和 classes.dex 进行瘦身:

APK 文件

2 图片优化

2.1 一套图片资源

图片优化主要是针对多套图片资源的问题。

Android 常见分辨率

  • ldpi:320 x 240
  • mdpi:480 x 320
  • hdpi: 800 x 480
  • xhdpi:1280 x 720
  • xxhdpi:1920 x 1080

因为 APP 在加载图片时会优先加载对应分辨率文件夹下的图片,如果在对应的分辨率文件夹下没有所要的图片,就会找高分辨率文件夹下的图片。那么,如果我们把所有的图片都放在高分辨率的文件夹下是不是就可以了呢?并不是,这样做会导致低分辨率手机加载图片时会消耗更多的内存。

2017年后,Android 手机一般大小在 5 寸以上,分辨率至少是 720p,1080p,所以对应的 dpi(Dots Per Inch 每英寸点) 分别为:

  • 1280 x 720,5 寸的手机,对应的 dpi 大约是 300dpi;
  • 1080 x 1920,5 寸的手机,对应的 dpi 大约是 440dpi;

对于绝大多数 APP 来说,只需要取一套设计图就足够了。鉴于现在分辨率的趋势,建议取 720p 的资源,放在 xhdpi 目录。相对于多套资源,只使用 720P 的一套资源,在视觉上差别不大,很多大公司的产品也是如此,但却能显著的减少资源占用大小,身边也能减轻设计师的出图工作量了。

注意,这里不是说把不是 xhdpi 的目录都删除,而是强调保留一套设计资源就够了。

2.2 使用 TinyPNG 有损压缩

TinyPNG 工具只支持上传 PNG 图片到官网上压缩,然后下载保存,在保持 alpha 通道的情况下对 PNG 的压缩可以达到 1/3 之内,而且用肉眼基本上分辨不出压缩的损失。

TinyPNG 的官方网站:http://tinypng.com/

2.3 使用 jpg 格式

如果对于非透明的大图,jpg 将会比 png 的大小有显著的优势,虽然不是绝对的,但是通常会减小一般都不止。

在启动页,活动页等之类的大图展示区采用 jpg 将是非常明智的选择。

2.4 使用 webp 格式

webp 支持透明度,压缩比 jpg 更高且显示效果却不输 jpg,官方评测 quality 参数等于 75 均衡最佳。

相对于 jpg、png,webp 作为一种新的图片格式,限于 android 的支持情况暂时还没用在手机端广泛引用起来。从 Android 4.0 开始原生支持,但不支持包含透明度,直到 Android 4.2.1+ 才支持显示含透明度的 webp,使用的时候要特别注意。

官方介绍:https://developers.google.com/speed/webp/docs/precompiled

2.5 缩小大图

如果经过上述步骤之后,你的工程里面还有一些大图,考虑是否有必要维持这样的大尺寸,是否能适当的缩小。事实上,由于设计师出图的原因,我们拿到的很多图片完全可以适当的缩小而对视觉影响是极小的。

2.6 覆盖第三库里的大图

有些第三方库里引用了一些大图但是实际上并不会被我们用到,就可以考虑用 1x1 的透明图片覆盖。这样会导致 drawable 文件下包含了一些莫名其妙的名称的 1x1 图片…

2.7 SVG

对于位图图像来说,其的大小是固定的,分辨率低,占的内存空间小,分辨率高,占的内存空间大。

SVG,SVG 意为可缩放矢量图形(Scalable Vectors Graphics),使用 XML 格式定义图像,适用简单的小图标。

首先在 res 包目录下点击 New —> Vector Asset:

添加 svg 图片

进入 Configure Vector Asset 窗口:

添加 svg 类型的图片

选择 Local file (SVG, PSD),选择要导入的 svg 图片:

添加 svg 类型的图片

点击 Next —> Finish:

添加 svg 类型的图片

导入成功:

svg 导入成功

使用:

使用 svg

在 apk 包的 drawable 目录下是 .xml 文件而不是 .png:

svg 图片所在位置

使用 svg 可以不用考虑图片的分辨率、大小、颜色等信息。.xml 的性能比 .png 更好,占用内存更少,转换成机器码的效率更高。

如果 SVG 文件包含不受支持的功能,将在 Vector Asset Studio 的底部显示一个错误提示。不支持的功能:

  • 滤镜效果:不支持投影,模糊和颜色矩阵等效果;
  • 文本:建议使用其他工具将文本转换为形状;

svg 文件出错

Tint 着色器:可以直接在 xml 文件中修改矢量图的颜色,但是并不建议直接修改,我们一般用 Tint 着色器去修改矢量图的颜色:

<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:color="#00ff00" android:state_pressed="true" />
    <item android:color="@android:color/black" />
</selector>

<ImageView
    android:id="@+id/iv"
    android:layout_width="60dp"
    android:layout_height="60dp"
    android:clickable="true"
    android:focusable="true"
    android:src="@drawable/fanhui"
    app:tint="@color/selector_back_color" />

使用 .svg 格式的图片,转换成 vector 格式的文件,不同大小,不同颜色的图片只需要一个文件就可以了实现了。

矢量图是由点与线组成,和位图不一样,它再放大也能保持清晰度,而且使用矢量图比位图设计方案能节约 30~40% 的空间,现在谷歌一直在强调扁平化方式,矢量图可很好的契合该设计理念。

优势

  • 占用内存空间小
  • 无限拉伸不会出现锯齿,可以照顾不同尺寸的机型;
  • Android Studio 自带很多资源,减小 UI 的工作量;

劣势

  • 只支持 5.0 及以上的系统;
  • 与位图相比多了一层计算,需消耗更多的性能;
  • 不支持 .9 图;
  • 不适合表现真实照片和复杂图形,一般使用在简单的 icon 和动画上;

3 动态库打包配置

在 Android 开发中,so 库是不可或缺的。so 库指的是动态链接库,也就是在运行时载入内存的库文件。在 Android 应用程序中使用 so 库可以大大降低内存的使用,提高系统的性能。so 库文件的产生主要有两种方式:

  • 使用 C/C++ 编写的 Native 代码,将 Native 的 C/C++ 代码编译为 so 文件,通过 Java Native Interface(JNI) 的方式使用 so 文件;
  • 使用第三方的 so 库文件,例如 FFmpeg、OpenCV 等开源库;

so 库是由 ndk 编译出来的动态库,是 C/C++ 写的,所以不是跨平台的,即每一个平台需要使用对应的 so 库。

ABI 是应用程序二进制接口简称(Application Binary Interface),定义了二进制文件(尤其是 .so 文件)如何运行在相应的系统平台上,从使用的指令集,内存对齐到可用的系统函数库。

在 Android 系统上,每一个 CPU 架构对应一个 ABI:arm64-v8a,armeabi-v7a,armeabi,x86_64,x86,mips64,mips。现在我们一般只需要配置 arm64-v8a。

  • arm64-v8a:第 8 代 64 位 ARM 处理器。目前,国内的应用生态正在向 64 位架构过渡,移动芯片平台也将逐步弱化对 32 位应用的支持(兼容 armeabi-v7a、armeabi);
  • armeabi-v7a:第 7 代及以上的 AMR 处理器(兼容 armeabi);
  • armeabi:第 5、6 代的 ARM 处理起,早期的手机用的比较多;
  • x86_64:64 位的平板;
  • X86:平板、模拟器用的比较多;
  • mips64/mips:极少用于手机可以忽略;

在 build.gradle 构建脚本中,配置 ndk 编译的动态库 CPU 架构类型:

android {
    defaultConfig {
        ndk {
        		abiFilters "armeabi-v7a", "arm64-v8a", "x86", "x86_64"
        }
    }
}

按照以上配置,打包时会将 4 种 CPU 架构的动态库都配置到 APK 中。事实上,绝大多数的应用都不需要配置全架构的动态库,arm64-v8a 架构的 CPU 可以向下兼容:

android {
    defaultConfig {
        ndk {
        		abiFilters "arm64-v8a"
        }
    }
}

4 移除无用资源

Android Studio 给我们提供了一键移除所有无用的资源,如下所示:

移除无用资源

移除无用资源

但是这种方式不建议使用,因为如果某资源仅存在动态获取资源 id 的方式,那么这个资源会被认为没有使用过,从而会直接被删除。

动态获取的方式:getResources().getIdentifier(“name”, “defType”, getPackageName())

另外一种方式是通过 Analyze Code 手动移除无用资源:

Analyze Code

Run Inspection by Name

搜索 unused resources:

unused resources

选择搜索范围:

选择搜索范围

无用资源:

无用文件

5 代码压缩-混淆

  • 长类名改为短类名;
  • 长方法名改为短方法名;
  • 变量名改变;
  • 删除无用代码;
  • 缺点:启用后编译较慢,混淆可能导致编译出错或其他 bug。debug 模式下最好不要做混淆,因为做了混淆之后方法名、类名都会发生变化,不容易定位 bug。
buildTypes {
  
  	debug {
        minifyEnabled false // 不做混淆
        proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
    }
  
    release {
        minifyEnabled true // 做混淆
          shrinkResources true
        proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
    }
}

minifyEnabled 用来开启删除无用代码,比如没有引用到的代码。

shrinkResources 用来开启删除无用资源,也就是没有被引用的文件(drawable、layout),实际上并没有删除,只是保留文件名,但是没有内容。但是需要知道资源是否被引用,需要配合 minifyEnabled 来使用,只有当两者都为 true 的时候才能真正的删除无效代码和无用资源。

资源压缩只与代码压缩协同工作。

默认情况未启用严格模式(严格模式是指清除掉资源本身,非严格模式指的是只清除资源内容),如需启动则设置 shrinkMode,创建 keep.xml,如下:

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:android="http://schemas.android.com/tools"
           tools:shrinkMode="strict" >

如果开发者想要特定保留或者必须移除的资源,可以进行自定义配置:

  • tools:keep:指定要保留的资源,如果有多个资源保留需要用逗号隔开;
  • Tools:discard:指定要移除的资源,如果有多个资源移除需要用逗号隔开;
<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:tools="http://schemas.android.com/tools"
    tools:discard="@color/selector"
    tools:keep="@layout/l_used*_c,@layout/l_used_a,@layout/l_used_b*"
    tools:shrinkMode="strict">
		
  	<!-- 该设置支持 * 通配符设置 -->
    <!-- shrinkMode="strict": 该模式只保留在代码或者资源文件中明确引用的资源-->
    <!-- shrinkMode="safe": 该模式会保留所有明确引用的资源以及可能被 Resources.getIdentifier() 动态引用的资源-->

</resources>

6 动态加载 so 库

Java 加载 so 库的方式有两种:

  • 静态加载:System.loadLibrary(String libname),参数为 so 库名称,位于 APK 的 lib 的目录下。采用此种方式是要将 so 库打包进 apk 或者 jar 包中的;
  • 动态加载:System.load(String filePath),参数为 so 库在磁盘中的完整路径,可以加载磁盘中的 so 库文件;
//加载的是 jni_mix.so,注意的是这边只需要传入 "jni_mix"
static {
  	System.loadLibrary("jni_mix");
}
//传入的是 so 文件完整的绝对路径
System.load("/data/data/[packagename]/lib/jni_mix.so")

静态加载会导致 apk 包比较大,所以采用动态加载 so 库的形式,也就是从网络上下载,放入本地数据目录下。这样做的好处是不仅减小了 apk 的大小,而且可以随时使用最新的依赖库,这也是动态加载的最多的用途之一。

用动态加载技术,编译前不把 so 文件放入 jniLibs 目录(原因很多,比如想减少安装包的大小),自然打包生成的安装包也不包含该 so 库。接着在手机上安装这个 apk 并启动 APP,如果 APP 的运行不涉及到 jni 方法的调用,那就当 so 不存在;如果 APP 打开某个页面,而该页面又需要调用 jni 方法,则 APP 自动到指定地址下载需要的 so 文件,然后保存到用户目录,并从用户目录加载该 so,最后再调用 jni 方法。

步骤一:下载 so;

步骤二:拷贝 so 至私有(data)目录;

步骤三:通过绝对路径加载 so;

7 插件化压缩包

通过插桩式来实现加载插件,AssetManager 加载资源和 java 文件

插件化开发就是将整个 APP 拆分成很多模块,每个模块都是一个 APK,最终打包的时候将宿主 APK 和插件 APK 分开打包,插件 APK 通过动态下发到宿主 APK。

插件可以放到服务器加载。

8 删除无用的语言资源

在 resources.arsc 文件下可以查看支持的语言:

语言资源

大部分应用其实并不需要支持几十种语言的国际化支持。比如只支持中文、英文:

android {
		defaultConfig {
				resConfigs "zh", "en" // 支持中文、英文
		}
}

去除无用资源之后:

去除无用资源之后

9 shape 背景、着色、在线化素材、避免重复库

9.1 使用 shape 背景

特别是在扁平化盛行的当下,很多纯色的渐变的圆角的图片都可以使用 shape 实现,代码灵活可控,省去了大量的背景图片。

9.2 使用着色方案

相信你的功能里面也有很多 selector 文件,也有很多相似的图片只是颜色不同,通过着色方案我们能大大减轻这样的工作量,减少这样的文件。

借助 android support 库乐意实现一个全版本兼容的着色方案,参考代码 DrawableLess.java

9.3 在线化素材库

如果你的 APP 支持素材库(比如聊天表情库)的话,考虑在线加载模式,因为往往素材库都有不小的体积。

这一步需要开发者实现在线加载,一方面增加代码的复杂度,一方面提高了 APP 的流量消耗,建议酌情选择。

避免重复库

避免重复库看上去是理所当然的,但是秘密总是藏得很深,一定要当心你引用的第三方库又引用了哪个第三方库,这就很容易出现重复的库了,比如使用了两个图片加载库:Glide 和 Picasso。

通过查看 exploded-aar 目录和 External Libraries 或者反编译生成的 APK,尽量避免重复的库的大小,减小 APP 大小。

10 清除冗余代码

版本迭代过程中,因为删减功能经常有冗余代码和第三方库留下,这或多或少都会增加包体,这种情况下没有捷径,只能每个文件查找,这是苦力活。还有要查看第三方库有没有可能精简,比如谷歌分基础、广告和分析包,网络库、supportv4 等,这个就具体情况具体分析了,不多阐述。

11 使用微信资源压缩打包工具

微信资源压缩打包工具通过短资源名称,采用 7zip 对 APP 进行极致压缩实现减小 APP 的目标,效果非常好,强烈推荐。

建议开启 7zip,注意白名单的配置,否则会导致有些资源找不到,官方已经发布 AndResGuard 到 gradle 中了,非常方便。

12 Provided 编译

对于一些库是按照需要动态加载的,可能在某些版本并不需要,但是代码又不方便去除否则编译不过。

对于 provided 可以保证代码编译通过,但是实际打包中并不引用第三方库,实现了控制 APP 大小的目标。

但是也同时就需要开发者自己判断不引用这个第三方库时就不要执行到相关的代码,避免 APP 崩溃。

参考

【Android】浅谈APP的瘦身之路
Android SO库的详细阐述
安卓APK安装包arm64-v8a、armeabi-v7a、x86、x86_64有何区别?如何选择?
【转】android中的armeabi、armeabi-v7a、arm64-v8a及x86等
【Android 安装包优化】动态库打包配置 ( “armeabi-v7a“, “arm64-v8a“, “x86“, “x86_64“ APK 打包 CPU 指令集配置 | NDK 完整配置参考 )

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1354519.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

台灯哪个品牌比较护眼?2024学生考研台灯推荐

在近几年&#xff0c;儿童青少年近视率非常高。很多家长认为孩子近视的原因是没有养成正确的用眼习惯&#xff0c;例如经常趴桌子写作业、眯眼看书等&#xff0c;但实际上这些坏习惯是因为没有合适的光线而导致的。所以安排一盏合适的台灯给孩子学习是非常重要的。但是市面上护…

Pruning Papers

[ICML 2020] Rigging the Lottery: Making All Tickets Winners 整个训练过程中mask是动态的&#xff0c;有drop和grow两步&#xff0c;drop是根据权重绝对值的大小丢弃&#xff0c;grow是根据剩下激活的权重中梯度绝对值生长没有先prune再finetune/retrain的两阶段过程 Laye…

el-table 展开行表格,展开的内容高度可以变化时,导致的固定列错位的问题

问题描述 一个可展开的表格&#xff08;列设置了type“expand”&#xff09;&#xff0c;并且展开后的内容高度可以变化&#xff0c;会导致后面所有行的固定列错位&#xff0c;图如下&#xff0c;展示行中是一个树形表格&#xff0c;默认不展示子级&#xff0c;点击树形表格的…

【金猿CIO展】现代咨询CIO崔恩博:数字化转型,CIO不仅要懂技术和业务,更要“懂人”...

‍ 崔恩博 本文由现代咨询CIO崔恩博撰写并投递参与“数据猿年度金猿策划活动——2023大数据产业年度优秀CIO榜单及奖项”评选。 大数据产业创新服务媒体 ——聚焦数据 改变商业 最近几年&#xff0c;大数据行业的发展备受关注&#xff0c;尤其是2019年以后&#xff0c;随着企业…

异步优势演员-评论家算法 A3C

异步优势演员-评论家算法 A3C 异步优势演员-评论家算法 A3C网络结构并行步骤 异步优势演员-评论家算法 A3C A3C 在 A2C 基础上&#xff0c;增加了并行训练&#xff08;异步&#xff09;来提高效率。 网络结构 A2C&#xff1a; A3C&#xff1a; 在这两张图之间&#xff0c;…

Spring事务传播问题 — PROPAGATION_REQUIRES_NEW

一、描述 Spring遇到嵌套事务时&#xff0c;当被嵌套的事务被定义为“PROPAGATION_REQUIRES_NEW”时&#xff0c; 内层Service的方法被调用时&#xff0c;外层方法的事务被挂起&#xff1b; 内层事务相对于外层事务是完全独立的&#xff0c;有独立的隔离性等等。 二、实验 但实…

判断一个给定的数组是否是Fortran连续的np.isfortran()

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 判断一个给定的数组 是否是Fortran连续的 np.isfortran() [太阳]选择题 以下代码的输出结果中正确的是? import numpy as np A np.array([[1, 2], [3, 4]]) B np.array([[1, 2], [3, 4]]…

ETLCloud X 明道云实现无缝数据连接

明道云作为一款云端协作工具&#xff0c;为企业提供高效的沟通、协作和数据分析服务。它可以实现企业内部沟通和协作的高效性和一体化&#xff0c;并提供数据分析功能&#xff0c;让企业能够更好地理解业务和决策。 一、传统方式同步数据的痛点 传统方式同步数据需要手动进行…

Navicat(数据库可视化软件)安装教程以及连接MYSQL

Navicat安装教程以及连接MYSQL Navicat&#xff08;数据库可视化软件&#xff09;安装流程安装MySQLnavicat连接mysql数据库 Navicat&#xff08;数据库可视化软件&#xff09; Navicat 是一款专门为 MySQL 设计的可视化数据库 GUI 管理工具&#xff0c;我们可以在自己的计算机…

网络安全—部署CA证书服务器

文章目录 网络拓扑安装步骤安装证书系统安装从属证书服务器 申请与颁发申请证书CA颁发证书 使用windows Server 2003环境 网络拓扑 两台服务器在同一网段即可&#xff0c;即能够互相ping通。 安装步骤 安装证书系统 首先我们对计算机名进行确认&#xff0c;安装了证书系统后我…

低代码平台开发 - 编辑器拓展

设计器&#xff08;编辑器&#xff09;这边内容比较杂&#xff0c;我们这次挑两个讲&#xff0c;一个是自定义出码&#xff0c;一个是新版本引擎中 array-setter 存在的问题这期和之前的文章关联性不大&#xff0c;可以直接在阿里的低代码引擎初始化的目录下进行&#xff0c;如…

实现文本 内容展开 / 收起

<template><el-table :data"tableData" style"width: 100%" height"250"><el-table-columnfixedprop"date"label"日期"width"150"></el-table-column><el-table-columnprop"name…

geemap学习笔记040:GEE中样本点选择操作流程

前言 geemap中目前有一个bug&#xff0c;就是在选择样本点的时候不合理&#xff0c;选完一类样本之后&#xff0c;没法继续选择下一类&#xff0c;并且没法在线进行编辑和修改。因此目前就只能结合在线版的GEE进行样本选择&#xff0c;本节就详细的介绍一下GEE中样本点的选择过…

HCIA-Datacom题库(自己整理分类的)——其他网络协议【完】

&#xff08;一&#xff09;单选 下列属于链路状态协议的是? Direct static FTP OSPF 解析&#xff1a; FTP&#xff1a;文件传输协议 OSPF&#xff1a;链路状态路由协议 如下图所示的网络主机A通过Telnet登录到路由器A然后在远程的界面通过FTP获取路由器的配置文件&…

multipath 内核接口及框架介绍

文章目录 1 云主机使用网络存储 io 流程2 multipath 介绍 1 云主机使用网络存储 io 流程 对于一个云服务环境&#xff0c;大致会有网络节点&#xff0c;存储节点&#xff0c;计算节点&#xff0c;控制节点&#xff0c;其中虚拟云主机在计算节点工作&#xff0c;而虚拟云主机&a…

数据结构——二叉树四种遍历的实现

目录 一、树的概念 1、树的定义 1&#xff09;树 2&#xff09;空树 3&#xff09;子树 2、结点的定义 1&#xff09;根结点 2&#xff09;叶子结点 3&#xff09;内部结点 3、结点间关系 1&#xff09;孩子结点 2&#xff09;父结点 3&#xff09;兄弟结点 4、树…

人工智能在金融领域的应用存在的4大挑战

金融服务供应商应该有计划地应对AI面临的难题 金融行业投资人工智能热潮带来有关数据安全和透明度的新问题。由于数据管理实践随着新的 AI 解决方案的引入而不断发展&#xff0c;应对这些新问题以及金融服务领域 AI 面临的其他挑战尤为重要。各组织必须认识到可能面临以下挑战…

网络安全—IPSec安全策略

文章目录 网络拓扑添加策略ESP添加筛选器添加筛选器的操作另一台主机设置 AH 使用Windows Server 2003系统 网络拓扑 client1 IP 192.168.17.105client2 IP 192.168.17.106 只要保证两个主机在同一网段接口&#xff0c;即互相ping通即可完成策略的实现 下面的所有通讯都只是…

Vue 模板编译原理解析

Vue 模板编译原理解析 模板编译整体流程 首先我们看一下什么是编译&#xff1f; 所谓编译&#xff08;Compile&#xff09;&#xff0c;指的是将语言 A 翻译成语言 B&#xff0c;语言 A 就被称之为源码&#xff08;source code&#xff09;&#xff0c;语言 B 就被称之为目标…

半导体设备系列:半导体制造产能扩张,设备零部件需求旺盛

近年来国内半导体制造产能不断扩张&#xff0c;半导体设备厂商加速成长。我们认为下游发展将拉动上游本地化配套需求&#xff0c;半导体设备零部件迎来高增长阶段。 摘要 半导体设备零部件包含密封圈、EFEM、射频电源、静电吸盘、硅电极、真空泵、气体流量计、喷淋头等产品&a…