Google Play渠道超过100M?尝试APK分包!

第一次安装完app后,需要将obb文件进行解压并将解压后的文件存储到我们定义的文件夹里(可以是data/data/包名/files/也可以是内置存储下自定义的项目文件夹)。要想解压obb文件,第一步是获取obb文件的本地路径,具体代码如下:

public static String getObbFilePath(Context context) {
try {
return Environment.getExternalStorageDirectory().getAbsolutePath()

  • “/Android/obb/”
  • context.getPackageName()
  • File.separator
  • “main.”
  • context.getPackageManager().getPackageInfo(context.getPackageName(), 0).versionCode
  • “.”
  • context.getPackageName()
  • “.obb”;
    } catch (PackageManager.NameNotFoundException e) {
    e.printStackTrace();
    return null;
    }
    }

拿到obb文件路径后,可以开始进行解压了:

public static void unZipObb(Context context) {
String obbFilePath = getObbFilePath(context);
if (obbFilePath == null) {
return;
} else {
File obbFile = new File(obbFilePath);
if (!obbFile.exists()) {
//下载obb文件
} else {
File outputFolder = new File(“yourOutputFilePath”);
if (!outputFolder.exists()) {
//目录未创建 没有解压过
outputFolder.mkdirs();
unZip(obbFile, outputFolder.getAbsolutePath());
} else {
//目录已创建 判断是否解压过
if (outputFolder.listFiles() == null) {
//解压过的文件被删除
unZip(obbFile, outputFolder.getAbsolutePath());
}else {
//此处可添加文件对比逻辑
}
}
}
}
}

谷歌官方有提供解压obb文件的库供开发者使用,叫做APK Expansion Zip Library,感兴趣的小伙伴可以在一下路径下查看。

/extras/google/google_market_apk_expansion/zip_file/

笔者不推荐使用该库,原因是这个库已经编写了有一些年头了,当时编译的sdk版本比较低,有一些兼容性的bug需要开发者修改代码后才能使用。所以这里使用的upzip方法是用最普通的ZipInputStream和FileOutputStream解压zip包的方式来实现的:

//这里没有添加解压密码逻辑,小伙伴们可以自己修改添加以下
public static void unzip(File zipFile, String outPathString) throws IOException {
FileUtils.createDirectoryIfNeeded(outPathString);
ZipInputStream inZip = new ZipInputStream(new FileInputStream(zipFile));
ZipEntry zipEntry;
String szName;
while ((zipEntry = inZip.getNextEntry()) != null) {
szName = zipEntry.getName();
if (zipEntry.isDirectory()) {
szName = szName.substring(0, szName.length() - 1);
File folder = new File(outPathString + File.separator + szName);
folder.mkdirs();
} else {
File file = new File(outPathString + File.separator + szName);
FileUtils.createDirectoryIfNeeded(file.getParent());
file.createNewFile();
FileOutputStream out = new FileOutputStream(file);
int len;
byte[] buffer = new byte[1024];
while ((len = inZip.read(buffer)) != -1) {
out.write(buffer, 0, len);
out.flush();
}
out.close();
}
}
inZip.close();
}

public static String createDirectoryIfNeeded(String folderPath) {
File folder = new File(folderPath);
if (!folder.exists() || !folder.isDirectory()) {
folder.mkdirs();
}
return folderPath;
}

解压完成后,就可以通过输出文件的路径来访问到我们需要访问的大容量资源了,文件的读取在这里就不展开了。

  • 下载obb

从Google Play下载和安装App有一定概率会下载到不包含obb文件的apk,或者obb文件被人为删除了。这种情况下,需要开发者到谷歌提供的下载地址处下载相应的obb文件。可是要怎么获取到下载地址呢,这里使用了官方的Downloader Library

这个库可以通过Android Sdk Manager下载到,打开manager后勾上Google Play Licensing Library packageGoogle Play APK Expansion Library package点下载即可。可是在我兴高采烈准备大干一场的时候,发现它竟然编译不过[捂脸]。这个库和上面说的APK Expansion Zip Library一样,由于年代久远又年久失修,基本不能使用了。折腾了一些时间后,魔改了一个版本,才终于可以使用。 这里提供一个编译好的jar包google_apk_expand_helper。具体代码如下:

//随机byte数组,随便填就好
private static final byte[] salt = new byte[]{18, 22, -31, -11, -54, 18, -101, -32, 43, 2, -8, -4, 9, 5, -106, -17, 33, 44, 3, 1};

private static final String TAG = “Obb”;

public static void getObbUrl(Context context, String publicKey) {
final APKExpansionPolicy aep = new APKExpansionPolicy(
context,
new AESObfuscator(salt,
context.getPackageName(),
Settings.Secure.getString(context.getContentResolver(), Settings.Secure.ANDROID_ID)
));
aep.resetPolicy();

final LicenseChecker checker = new LicenseChecker(context, aep, publicKey);

checker.checkAccess(new LicenseCheckerCallback() {
@Override
public void allow(int reason) {
Log.i(TAG, “allow:” + reason);
if (aep.getExpansionURLCount() > 0) {
//这里就是获取到的地址
String url = aep.getExpansionURL(0);
}
}

@Override
public void dontAllow(int reason) {
Log.i(TAG, “dontAllow:” + reason);
}

@Override
public void applicationError(int errorCode) {
Log.i(TAG, “applicationError:” + errorCode);
}
});
}

上述方法中需要提供参数publicKey,这个publicKey可以在GooglePlayConsole中找到。

  • 小结

掌握了上述的方法我们就已经完成了Apk分包的主要流程了,以下内容将举例说明如果通过配置gradle文件进行多渠道打包,如何在每次打包的时候自动将大容量资源文件压缩成obb等。

多渠道与自动化

  • 例子

假设我们现在需要发布一个超过100M的安装包到GooglePlay以及应用宝,对于GooglePlay来说,我们需要生成小于100M的apk文件和obb文件,而对于应用宝来说,只需要生成一个完整的apk即可。

那么问题来了,我们不可能说在打包GooglePlay的时候将资源文件手动移除并修改资源引用的相关逻辑,然后再在打包应用宝的时候将他们放回来,这样做会大大增加开发者的工作量并且增大出错的可能性。那有没有办法在单个工程项目下既能打包GooglePlay的包又可以打包应用宝的包呢?答案是有的,build.gradle中的sourceSets就可以解决这样的问题。

  • 利用sourceSets隔离渠道资源和资源引用代码

假设我们有一个splash.mp4文件,在应用宝中渠道包中,它被放在了res/raw/目录下。而在googlePlay渠道包中,它被放置在obb文件里,我们可以这么处理。

首先,在src目录下创建两个新的目录googlePlay和tencent,并在他们的目录下新建java,res和assest文件夹。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

在app级别的build.gradle文件中添加GooglePlay和应用宝的渠道信息:

android {
flavorDimensions “default”
productFlavors {

GooglePlay { dimension “default” }
Tencent { dimension “default” }

/** 在AndroidManifest.xml中加入

**/

productFlavors.all { flavor ->
flavor.manifestPlaceholders = [CHANNEL_NAME: name]
}

}
}

紧接其后添加sourceSets配置,指定不同渠道的资源和代码地址,其中main为共有资源和代码,其余的为对应渠道包的资源和代码:

sourceSets {
main {
java.srcDirs = [‘src/main/java’]
assets.srcDirs = [‘src/main/assets’]
res.srcDirs = [‘src/main/res’]
}
GooglePlay {
java.srcDirs = [‘src/googlePlay/java’]
res.srcDirs = [‘src/googlePlay/res’]
assets.srcDirs = [‘src/googlePlay/assets’]
}
Tencent {
java.srcDirs = [‘src/tencent/java’]
res.srcDirs = [‘src/tencent/res’]
assets.srcDirs = [‘src/tencent/assets’]
}
}

将splash.mp4放到tencent/res/raw/文件夹下,并为不同渠道的java文件夹新建包名文件夹以及ResourcesHelper.java,完成后的目录结构如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

有两点需要注意的地方:

一是java包下必须创建包名文件夹,否则会无法引用到项目下的类。该例子中就是com.example.obbtest包。

二是AndroidStudio中可以通过左下角的Build Variants窗口选择当前需要编译的渠道包类型,当选择GooglePlay时会发现tencent下的java文件失效了。所以,如果需要修改某渠道下的java文件,请先通过Build Variants切换到指定渠道。

最后,针对不同渠道的ResourcesHelper.java采用不同的资源获取方式:

GooglePlay版本:

public class ResourcesHelper {
public static void playSplashVideoResource(VideoView videoView){
String filePath = ObbHelper.getCurrentObbFileFolder()+“raw/”+“splash.mp4”;
videoView.setVideoPath(filePath);
}
}

tencent版本:

public class ResourcesHelper {
public static void playSplashVideoResource(VideoView videoView) {
int resource = R.raw.splash;
String uri = “android.resource://” + videoView.getContext().getApplicationContext().getPackageName() + “/” + resource;
videoView.setVideoURI(Uri.parse(uri));
}
}

通过sourceSets隔离渠道资源和资源引用代码在这里就完成了,针对更加复杂的场景,就需要小伙伴根据实际情况进行扩展和修改了。下面我们来看一下如何在构建时自动将资源打包成obb文件。

  • 构建时生成obb文件

要在构建时生成obb文件就必须通过添加gradle脚本来实现。我们先在项目目录下新建一个脚本文件flavour.gradle。

然后,要想打包obb文件,就必须知道现在构建的是哪个渠道的包,那要怎么拿到现在的渠道呢,请看代码:

def String getCurrentFlavor() {
Gradle gradle = getGradle()
String tskReqStr = gradle.getStartParameter().getTaskRequests().toString()

Pattern pattern;

if (tskReqStr.contains(“assemble”))
pattern = Pattern.compile(“assemble(\w+)(Release|Debug)”)
else
pattern = Pattern.compile(“generate(\w+)(Release|Debug)”)

Matcher matcher = pattern.matcher(tskReqStr)

if (matcher.find())
return matcher.group(1).toLowerCase()
else {
println “NO MATCH FOUND”
return “”
}
}

我们知道obb的本质就是zip文件,所以只要在flavour.gradle中添加压缩文件的方法,就可以达到生成obb的效果了。由于笔者的Groovy语言不精通,所以这里使用java代码来解决,在flavour.gradle中添加:

import java.util.regex.Matcher
import java.util.regex.Pattern
import java.util.zip.ZipEntry
import java.util.zip.ZipOutputStream

ext {
zipObb = this.&zipObb
getCurrentFlavor = this.&getCurrentFlavor
}

//外部压缩方法入口,参数是所有需要压缩文件的目录以及输出路径,同样没有添加压缩密码逻辑,小伙伴们需要的自己添加吧
def static zipObb(File[] fs, String zipFilePath) {
if (fs == null) {
throw new NullPointerException(“fs == null”);
}
ZipOutputStream zos = null;
try {
zos = new ZipOutputStream(new BufferedOutputStream(new FileOutputStream(zipFilePath)));
for (File file : fs) {
if (file == null || !file.exists()) {
continue;
}
compress(file, zos, file.getName());
}
zos.flush();
} catch (Exception e) {
e.printStackTrace();
} finally {
if(zos != null){
try {
zos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}

//内部递归压缩方法
def static compress(File sourceFile, ZipOutputStream zos, String name) throws Exception {
byte[] buf = new byte[2048];
if (sourceFile.isFile()) {
// 向zip输出流中添加一个zip实体,构造器中name为zip实体的文件的名字
zos.putNextEntry(new ZipEntry(name));
// copy文件到zip输出流中
int len;
FileInputStream inputStream = new FileInputStream(sourceFile);

Android高级架构师

由于篇幅问题,我呢也将自己当前所在技术领域的各项知识点、工具、框架等汇总成一份技术路线图,还有一些架构进阶视频、全套学习PDF文件、面试文档、源码笔记。

  • 330页PDF Android学习核心笔记(内含上面8大板块)

  • Android学习的系统对应视频

  • Android进阶的系统对应学习资料

  • Android BAT部分大厂面试题(有解析)

好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
dlJU-1715766817034)]

  • Android学习的系统对应视频

  • Android进阶的系统对应学习资料

[外链图片转存中…(img-11JDI58l-1715766817036)]

  • Android BAT部分大厂面试题(有解析)

[外链图片转存中…(img-5fzht2Hy-1715766817038)]

好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 5
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值