Android 解决游戏发行切包资源索引冲突的问题

背景

游戏发行切包过程中,经常碰到渠道、研发、发行方,三方资源在合并过程中,资源ID冲突导致程序异常的问题,此类问题通过getIdentifier方式规避或者修改冲突资源ID的方式可以处理,但成本较高,本文旨在提出一种在切包过程中自动化处理资源冲突的解决方案

1、public.xml介绍

1、public.xml这个文件是哪来的?

该文件是apktool在反编译apk时,根据apk包中的resources.arsc文件生成。

没看过resource.arsc? (自己拖个apk到IDE看吧)

2、public.xml有什么作用

publc.xml是aapt在打包资源时用来固定资源id的,如果资源在public.xml中有对应的id了,那么打包资源时就用已经有的id。

3、public.xml中的id的格式

共四个字节32位,第一个字节代表PackgeID,第二个字节代表TypeID,后两个字节代表资源值

通常系统资源PackageID是01,而我们自己的资源PackageID是7f

TypeID,比如attr为01,string为02。但是并不固定,并不一定attr就是01。但是在public.xml中,同类型的该字节一定是一样的,否则回编译会失败。

2、R类介绍

R类这里有个知识点,library模块中生成的R类中的成员的值不是常量,不带final。app模块生成的R类的值是常量值。而常量值在java编译时会被优化,最终代码中输出的就是常量值,而不是R.id.xxx这样。而library的因为是变量,不会被优化,代码中会保留R.id.xxx

R类和public.xml的关系

从本质上讲,其实并没有啥关系。但是由于在代码中我们会使用R.id去查找资源,这就关联上了。如果都用getIdentifier的方式先获取id,那把R类删了也没事。

public.xml打包后对应的就是resources.arsc中的值,而资源值生成Java类,这个类就是R类。也就是说平时使用R类,就是用里面的索引值去到resources.arsc中找到对应资源位置,再去加载。

3、切包融合过程中R类和public.xml的处理

切包过程中,R类属于代码,采用直接覆盖的方式,但是由于我们生成的R类跟母包的R类其实值会是不同的。

下文中的cp指游戏研发方,即我们的SDK的接入方。

而public.xml是用的cp的,为什么用cp的?因为cp建立的是app工程,R类是常量值,如果我们把母包中public.xml中已有的值给改了,万一母包中用了,那就gg了

由于R类在library中使用的时候是个变量,保留了R.id.xxx这种形式,解决方法就有了,纠正R类中的值跟public.xml对应,这样就能继续愉快的使用R.id.xxx了。

我们的切包过程有几个步骤:

反编译母包(指接入我们SDK的乙方)====》合并渠道资源====》合并入新sdk的资源(跳过研发更新我们的sdk的过程哈)

1、在反编译母包的时候解析public.xml的值,存下来。

private void init() {
  List<Element> elements = mDocument.getRootElement().elements();
  for (Element element : elements) {
   String type = element.attribute(TYPE).getStringValue();
   String name = element.attribute(NAME).getStringValue();
   String id = element.attribute(ID).getStringValue();
   Map<String, String> typeMap = mTypeMap.get(type);
   if (typeMap == null) {
    typeMap = new HashMap<>();
    typeMap.put(name, id);
    mTypeMap.put(type, typeMap);
   } else {
    typeMap.put(name, id);
   }
  }
}

2、合并渠道资源的时候,将渠道资源中的public.xml(以channelPublic代指)合并到母包的public.xml(以matrixPublic代指)中

合并策略:

a、channelPublic中有,而matrixPublic中没有,增加到matrixPublic中

比如增加如下数据到matrixPublic中

<public type="attr" name="iconSrc" id="0x7f0200a8" />

如果该type在matrixPublic中已经存在:

首先要获取到attr在matrixPublic中的PackageId+TypeId。在一个public.xml文件中,同类型比如attr对应的PackageId+TypeId是不能变的,否则回编译失败。因此要添加数据时,数据的PackageId+TypeId需要纠正为matrixPublic的值。

其次资源值,不能和已有的资源值重复,正常情况下public.xml中的值是aapt生成的有序的,这里可以扫描matrixPublic中attr类型值的最大值,然后加一作为新加的iconSrc的id值

如果该type在matrixPublic中不存在(假设母包中matrixPublic中不存在attr类型)

首先要获取类型已经被占用的有哪些,即获取到matrixPublic中的TypeId,正常情况也是有序的,获取出最大的TypeId,加一作为新Type的起始值。赋值给iconSrc的id值

b、channelPublic中有,而matrixPublic中也有的,不需要处理,保留matrixPublic中的值不变

3、合并入新sdk的资源,在覆盖完R类,后开始纠正R类的值

扫描R类在PublicAndRHelper中

扫描覆盖完R类的smali代码中所有的R类,R$styleable类除外,因为styleable中保存的是一些数组的值,规则不同。

/**
  * 扫描代码中的R类
  * @return
  */
 private void scannerRClass(String path) {
  File smaliFilePath = new File(path);
  for (File file : smaliFilePath.listFiles()) {
   if (file.isDirectory()) {
    scannerRClass(file.getAbsolutePath());
   } else if(file.isFile()){
    if (file.getName().equals("R.smali") || file.getName().startsWith("R$")) {
     //此处过滤掉styleable文件
     if (!file.getName().endsWith("R$styleable.smali")) {
      mRClassFileList.add(file.getAbsolutePath());
     }
    }
   }
  }
 }

针对每一个R类调用纠正R类中方法,纠正R类值在RValueHelper类中

策略:匹配出要纠正的行,获取到type,name。在public.xml中找出对应的值,纠正。

注意这里的纠正不要用replace(oldValue,newValue)这种方式,要用替换行的方式,因为存在新值在R类中也存在后,后续替换出问题。比如a替换成b,b替换成c的情况最终R类中的a和b都被替换成了c

其次是styleable的处理,当扫描到的R是attr类型的时候,判断是否有styleable类型的存在,如果存在,则缓存下来attr中所做的纠正,用于纠正styleable。

public static void handle(String RFilePath, PublicXmlBean publicXmlBean) {
  File RFile = new File(RFilePath);
  String RStyleFilePath = "";
  Map<String, String> cacheMap = null;
  if (RFile.getName().endsWith("R$attr.smali")) {
   RStyleFilePath = RFilePath.replace("R$attr", "R$styleable");
   File RStyleAbleFile = new File(RStyleFilePath);
   //styleable存在,则把attr文件替换过的值缓存
   if (RStyleAbleFile.exists()) {
    cacheMap = new HashMap<>();
   }
  }
  String rFileContent = FileUtil.read(RFilePath);
  //找到RFile中是属性的每一行
  ArrayList<String> lines = FileUtil.readAllLines(RFilePath, ".field public static final");
  String regex = ".field public static final (.*):(.*) = (.*)";
  for (String line : lines) {
   Pattern pattern = Pattern.compile(regex);
   Matcher matcher = pattern.matcher(line);
   if (matcher.find()) {
    String type = RFile.getName().replace("R$", "").replace(".smali", "");
    String name = matcher.group(1);
    String resetValue = publicXmlBean.getValue(type, name);
    if (StringUtils.isEmpty(resetValue)) {
     resetValue = publicXmlBean.addValue(type, matcher.group(1));
    }
    //替换到文件内容中
    rFileContent = rFileContent.replace(line, ".field public static final " + name + ":" + matcher.group(2) + " = " + resetValue);
    if (cacheMap != null) {
     //换过的值缓存起来
     cacheMap.put(matcher.group(3), resetValue);
    }
   }
  }
  FileUtil.write(RFilePath, rFileContent);
  if (cacheMap != null) {
   //纠正R$styleable的值
   List<String> styleAbleLines = FileUtil.readAllLines(RStyleFilePath);
   BufferedWriter bw = null;
   try {
    bw = new BufferedWriter(new FileWriter(RStyleFilePath));
    for (String styleAbleLine : styleAbleLines) {
     for (String key : cacheMap.keySet()) {
      if (styleAbleLine.contains(key)) {
       styleAbleLine = styleAbleLine.replace(key, cacheMap.get(key));
      }
     }
     bw.write(styleAbleLine);
     bw.newLine();
    }
   } catch (IOException e) {
    e.printStackTrace();
   } finally {
    if (bw != null) {
     try {
      bw.close();
     } catch (IOException e) {
      bw = null;
     }
    }
   }
  }
 }

至此,纠正完了R类和public.xml的值

4、小结

游戏发行行业中,切包过程由于是多方代码资源的一个合并过程,经常出现资源冲突问题。本方案致力于优化切包过程,自动化地解决资源冲突问题。本方案已申请专利,并在我们实际业务中使用并稳定运行

以上就是Android 解决游戏发行切包资源索引冲突的问题的详细内容,更多关于Android 游戏发行切包资源索引冲突的资料请关注我们其它相关文章!

(0)

相关推荐

  • Android手机联系人快速索引(手机通讯录)

    最近需要实现一个手机通讯录的快速索引功能.根据姓名首字母快速索引功能.下面是一个手机联系人快速索引的效果,总体来说代码不算难,拼音转换的地方略有复杂.下面上源码:源码中有注释. 下面是效果图: MainActivity: import java.util.ArrayList; import java.util.Collections; import java.util.List; import android.app.Activity; import android.os.Bundle; imp

  • Android手机联系人带字母索引的快速查找

    喜欢另辟蹊径的我,在这里废话不多说了,直接上代码和图片了. 效果图如下: 第一步:MainActivity的代码如下: package net.loonggg.test; import java.util.ArrayList; import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.TreeSet; import android.os.Bundle; import and

  • 浅析Android位置权限以及数组寻找索引的坑

    一.Android 危险权限,来自官方文档的坑 Android开发者都知道,Android 6.0 之前,权限申请只需要在 AndroidManifest.xml 文件中声明就可以.Android 6.0 开始,权限申请发生了变化,危险权限需要在应用中动态申请,之前写过一篇 Android 动态申请危险权限的笔记,详情参考: Android 6.0 动态申请危险权限. 先截个图,看看Android官方的说明: 再看危险权限的分组情况: 意思是,对危险权限进行了分组,同一组中,只要有有一个权限被授

  • Android自定义View实现通讯录字母索引(仿微信通讯录)

    一.效果:我们看到很多软件的通讯录在右侧都有一个字母索引功能,像微信,小米通讯录,QQ,还有美团选择地区等等.这里我截了一张美团选择城市的图片来看看: 我们今天就来实现图片中右侧模块的索引功能,包括触摸显示以选中的索引字母.这里我的UI界面主要是参照微信的界面来实现,所以各位也可以对照微信来看看效果,什么都不说了,只有效果图最具有说服力! 二.分析: 我们看到这样的效果我们心理都回去琢磨,他是如何实现的: 首先,它肯定是通过自定义 View 来实现的,因为 Android 没有提供类似这样的控件

  • android仿微信联系人索引列表功能

    前言   因为自己在做的一个小软件里面需要用到从A-Z排序的ListView,所以自然而然的想到了微信的联系人,我想要的就是那样的效果.本来没打算自己去写,想要第三方写好的东西,搜了几个之后发现有的太复杂了,有的简单是简单,但是不符合我的要求,所以我就来个整合,把复杂性和简单性合二为一. 实现   先来看效果图吧: 要点分析   要实现这样的效果需要考虑下面的几个问题: 右边字母栏的绘制 点击效果的实现 汉字按A-Z的排序问题 正常的Item和字母分隔符的Item的实现   下面我们就解决这几个

  • android将搜索引擎设置为中国雅虎无法搜索问题解决方法

    该问题是由于yahoo的搜索接口改变导致,请修改 Donottranslate-all_search_engines.xml (x:\6575gb2\v2.12\alps\mediatek\source\frameworks\banyan\res\res\values)41753 8/11/2011 中的<string-array name="yahoo_cn" translatable="false">的定义为 复制代码 代码如下: <strin

  • android 左右滑动+索引图标实现方法与代码

    使用Gallery和ImageView实现android左右滑动+索引图标效果. 首先自定义Gallery实现一次只能滑动一个页面 复制代码 代码如下: public class MGalleryView extends Gallery{ public MGalleryView(Context context, AttributeSet attrs) { super(context, attrs); } //一次只能滑动一张图片注:一张图充满全屏 @Override public boolean

  • android仿微信通讯录搜索示例(匹配拼音,字母,索引位置)

    前言: 仿微信通讯录搜索功能,通过汉字或拼音首字母找到匹配的联系人并显示匹配的位置 一:先看效果图 字母索引 搜索匹配 二:功能分析 1:汉字转拼音 通讯录汉字转拼音(首个字符当考虑姓氏多音字), 现在转换拼音常见的有pinyin4j和tinypinyin, pinyin4j的功能强大,包含声调多音字,tinypinyin执行快占用内存少, 如果只是简单匹配通讯录,建议使用tinypinyin,用法也很简单这里不详细介绍 拼音类 public class CNPinyin <T extends

  • Android 实现带字母索引的侧边栏功能

    之前已经用自定义View做出如下这样一个效果了 这两天需要重新拿来使用,发现效果虽然做出来了,不过思路不太对,就重新参考写了一个,用法也更为简单了 首要的自然是需要继承View绘制出侧边栏,并向外提供一个监听字母索引变化的方法 /** * 作者:叶应是叶 * 时间:2017/8/20 11:38 * 描述: */ public class LetterIndexView extends View { public interface OnTouchingLetterChangedListener

  • Android ItemDecoration 实现分组索引列表的示例代码

    本文介绍了Android ItemDecoration 实现分组索引列表的示例代码,分享给大家.具体如下: 先来看看效果: 我们要实现的效果主要涉及三个部分: 分组 GroupHeader 分割线 SideBar 前两个部分涉及到一个ItemDecoration类,也是我们接下来的重点,该类是RecyclerView的一个抽象静态内部类,主要作用就是给RecyclerView的ItemView绘制额外的装饰效果,例如给RecyclerView添加分割线. 使用ItemDecoration时需要继

  • Android通用索引栏实现代码

    偶尔看到之前写过的代码,感觉好多东西几乎在很多项目中都要用到,虽然每个项目的需求和设计都不同,不过实现的效果都是一样的,可能只是数据格式和一些颜色等的细微差距.但是有的时候因为一个小改变,就要去重复的修改代码,麻烦不说,也容易导致新的问题和BUG. 就拿忽然想到的索引栏来说,几乎写过的项目中都用到了,比如城市选择.联系人等等.这些地方全都需要用到索引栏,但是用法都是一样的.翻看了几处之前写过的代码,发现每次用到索引栏,都要重新去写方法来处理数据或者对数据的索引进行提取这些,做法也都大同小异.于是

随机推荐