Android 應(yīng)用間資源覆蓋

該方案基于android 9.x驗(yàn)證

問題描述

客戶想要我們的主應(yīng)用MainApp,但是想通過值改變資源應(yīng)用ResApp疾宏,達(dá)到改變應(yīng)用多語言翻譯的需求

也就是,我們提供給客戶MainApp的apk文件,和ResApp的源碼文件,客戶通過改變ResApp狼忱,來控制MainApp的顯示膨疏。

解決方案 1

  1. 將兩個apk shared到同一個進(jìn)程中去
//主應(yīng)用 MainApp
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.jdf.res.main"
    android:sharedUserId="com.jdf.res">

//資源應(yīng)用 ResApp
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.jdf.res.resapp"
    android:sharedUserId="com.jdf.res">
  1. 將兩個應(yīng)用新增同樣名稱,不同值的資源字段
//主應(yīng)用 MainApp-默認(rèn)資源
<resources>
    <string name="app_name">MainApp</string>
    <string name="test1">Main App test1</string>
    <string name="test2">Main App test2</string>
    <string name="test3">Main App test3</string>
    <string name="test4">Main App test4</string>
    <string name="test5">Main App test5</string>
    <string name="test6">Main App test6</string>

</resources>

//資源應(yīng)用 ResApp--默認(rèn)資源
<resources>
    <string name="app_name">ResApp</string>
    <string name="test1">Res app test1</string>
    <string name="test2">Res app test2</string>
    <string name="test3">Res app test3</string>
    <string name="test4">Res app test4</string>
    <string name="test5">Res app test5</string>
    <string name="test6">Res app test6</string>
</resources>

//資源應(yīng)用 ResApp--中文資源
<resources>
    <string name="app_name">資源應(yīng)用</string>
    <string name="test1">資源應(yīng)用 test1</string>
    <string name="test2">資源應(yīng)用 test2</string>
    <string name="test3">資源應(yīng)用 test3</string>
    <string name="test4">資源應(yīng)用 test4</string>
    <string name="test5">資源應(yīng)用 test5</string>
    <string name="test6">資源應(yīng)用 test6</string>
</resources>

  1. 然后在主應(yīng)用钻弄,覆蓋resource值為資源應(yīng)用的resource
   //主應(yīng)用 MainApp
    @Override
    public Resources getResources() {
        Context context = null;
        try {
            context = createPackageContext("com.jdf.res.resapp", Context.CONTEXT_IGNORE_SECURITY);
        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }

        if (context != null) {
            return context.getResources();
        } else {
            return super.getResources();
        }
    }

通過以上方式覆蓋后佃却,我們再訪問資源時,訪問的就是資源應(yīng)用的

Log.d("res", "Main app res string:" + getString(R.string.test1));
Log.d("res", "Main app res string:" + getString(R.string.test2));
Log.d("res", "Main app res string:" + getString(R.string.test3));
Log.d("res", "Main app res string:" + getString(R.string.test4));
Log.d("res", "Main app res string:" + getString(R.string.test5));
Log.d("res", "Main app res string:" + getString(R.string.test6));
  1. 測試結(jié)果
2010-01-01 08:37:02.153 8481-8481/com.jdf.res.main D/res: Main app res string:資源應(yīng)用 test1
2010-01-01 08:37:02.156 8481-8481/com.jdf.res.main D/res: Main app res string:資源應(yīng)用 test2
2010-01-01 08:37:02.157 8481-8481/com.jdf.res.main D/res: Main app res string:資源應(yīng)用 test3
2010-01-01 08:37:02.159 8481-8481/com.jdf.res.main D/res: Main app res string:資源應(yīng)用 test4
2010-01-01 08:37:02.161 8481-8481/com.jdf.res.main D/res: Main app res string:資源應(yīng)用 test5
2010-01-01 08:37:02.162 8481-8481/com.jdf.res.main D/res: Main app res string:資源應(yīng)用 test6

為什么能夠訪問到資源應(yīng)用的字符串

原因是通過 android:sharedUserId將兩個應(yīng)用share到同一個進(jìn)程窘俺,兩個apk的同一個資源屬性的資源id是相同的饲帅;然后通過覆蓋resource,就能夠通過資源id瘤泪,訪問到到另外一個應(yīng)用的資源了

通過反編譯灶泵,我們看下,兩個應(yīng)用的对途,新增的字符串屬性的資源id編號

    //主應(yīng)用的資源id
    /* renamed from: com.jdf.res.main.R$string */
    public static final class string {
        public static final int test1 = 2131492894;
        public static final int test2 = 2131492895;
        public static final int test3 = 2131492896;
        public static final int test4 = 2131492897;
        public static final int test5 = 2131492898;
        public static final int test6 = 2131492899;
    }

     //資源應(yīng)用的資源id
    /* renamed from: com.jdf.res.resapp.R$string */
    public static final class string {
        .....
        public static final int test1 = 2131492894;
        public static final int test2 = 2131492895;
        public static final int test3 = 2131492896;
        public static final int test4 = 2131492897;
        public static final int test5 = 2131492898;
        public static final int test6 = 2131492899;
    }

這種方式赦邻,在xml引用資源,同樣適用实檀,因?yàn)橘Y源文件惶洲,也是通過調(diào)用處的上下文ressource加載的。
有一個缺點(diǎn)膳犹,主應(yīng)用MainApp的桌面app名稱恬吕,還是應(yīng)用自己的,而不是ResApp的须床。

下面我們針對其他應(yīng)用獲取應(yīng)用方式铐料,針對性的解決應(yīng)用名稱不能替換的問題

應(yīng)用名稱獲取方式 1

    public static String getApplicationNameByPackageName(Context context, String packageName) {
        PackageManager pm = context.getPackageManager();
        String Name;
        try {
            Name = pm.getApplicationLabel(pm.getApplicationInfo(packageName, PackageManager.GET_META_DATA)).toString();
        } catch (PackageManager.NameNotFoundException e) {
            Name = "";
        }
        return Name;
    }

如上所示,桌面是通過獲取各個應(yīng)用的getApplicationLabel獲取應(yīng)用名稱的侨颈,所以余赢,如果達(dá)到MainApp顯示的應(yīng)用名稱也是ResApp的,我們還需要修改系統(tǒng)代碼哈垢,覆蓋MainApp的名稱

系統(tǒng)修改修改方法:

diff --git a/base/core/java/android/app/ApplicationPackageManager.java b/base/core/java/android/app/ApplicationPackageManager.java

index b1a5651..40db061
--- a/base/core/java/android/app/ApplicationPackageManager.java
+++ b/base/core/java/android/app/ApplicationPackageManager.java
@@ -1755,7 +1755,10 @@ public class ApplicationPackageManager extends PackageManager {

     @Override
     public CharSequence getApplicationLabel(ApplicationInfo info) {
-        return info.loadLabel(this);
+        
+        return LabelManager.replaceLabel(this, info);
+       
+//        return info.loadLabel(this);
     }



應(yīng)用名稱獲取方式2

    //LauncherActivityInfo.getLabel
    public CharSequence getLabel() {
        // TODO: Go through LauncherAppsService
        return mActivityInfo.loadLabel(mPm);
    }

    //PackageItemInfo
    public @NonNull CharSequence loadLabel(@NonNull PackageManager pm) {
        if (sForceSafeLabels) {
            return loadSafeLabel(pm);//里面也會調(diào)用loadUnsafeLabel方法
        } else {
            return loadUnsafeLabel(pm);
        }
    }

    /** {@hide} */
    public CharSequence loadUnsafeLabel(PackageManager pm) {
        if (nonLocalizedLabel != null) {
            CharSequence replacedLabel = LabelManager.loadUnsafeLabel(pm, packageName, labelRes, getApplicationInfo());
            return nonLocalizedLabel;
        }
        if (labelRes != 0) {
            CharSequence label = pm.getText(packageName, labelRes, getApplicationInfo());
            if (label != null) {
                return label.toString().trim();
            }
        }
        if (name != null) {
            return name;
        }
        return packageName;
    }

考慮到子類覆蓋的情況

xxx@ubuntu:~/xxx/android/xxx/frameworks/base/core/java$ grep -nr "loadUnsafeLabel(PackageManager pm)"
android/content/pm/ComponentInfo.java:100:    @Override public CharSequence loadUnsafeLabel(PackageManager pm) {
android/content/pm/PackageItemInfo.java:199:    public CharSequence loadUnsafeLabel(PackageManager pm) {

需要對ComponentInfo.java和PackageItemInfo.java兩個類的loadUnsafeLabel方法進(jìn)行處理

   //ComponentInfo.java
    /** @hide */
    @Override public CharSequence loadUnsafeLabel(PackageManager pm) {
        if (nonLocalizedLabel != null) {
            return nonLocalizedLabel;
        }
        ApplicationInfo ai = applicationInfo;
        CharSequence label;

         if (labelRes != 0) {
-            label = pm.getText(packageName, labelRes, ai);
+            /* add for res replace */
+            label = LabelManager.loadUnsafeLabel(pm, packageName, labelRes, ai);
+            //label = pm.getText(packageName, labelRes, ai);
             if (label != null) {
                 return label;
             }
@@ -126,7 +130,10 @@ public class ComponentInfo extends PackageItemInfo {
             return ai.nonLocalizedLabel;
         }
         if (ai.labelRes != 0) {
-            label = pm.getText(packageName, ai.labelRes, ai);
+            /* add for res replace */
+            label = LabelManager.loadUnsafeLabel(pm, packageName, ai.labelRes, ai);
+            //label = pm.getText(packageName, ai.labelRes, ai);
             if (label != null) {
                 return label;
             }

    //PackageItemInfo.java
    /** {@hide} */
    public CharSequence loadUnsafeLabel(PackageManager pm) {
        if (nonLocalizedLabel != null) {
            return nonLocalizedLabel;
        }
         if (labelRes != 0) {
-            CharSequence label = pm.getText(packageName, labelRes, getApplicationInfo());
+            /* add for camera res replace */
+            CharSequence label = LabelManager.loadUnsafeLabel(pm, packageName, labelRes, getApplicationInfo());
+            //CharSequence label = pm.getText(packageName, labelRes, getApplicationInfo());
             if (label != null) {
                 return label.toString().trim();
             }

LabelManager的實(shí)現(xiàn)

package android.app;

import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.util.Log;

public class LabelManager {
    public static final String TAG = "LabelManager";

    public static final boolean IS_SUPPORT_LABEL_MAR = true;

    private static final String[] REPLACE_FROM_PKGS = new String[] {
            "com.jdf.res.main",
            "com.android.camera" 
            };

    private static final String[] REPLACE_TO_PKGS = new String[] {
            "com.jdf.res.resapp",
            "test.android.camera"
            };

    private static String getReplacedPkg(String pkg) {
        if (pkg != null) {
            for (int i = 0; i < REPLACE_FROM_PKGS.length; i++) {
                if (pkg.equals(REPLACE_FROM_PKGS[i])) {
                    return REPLACE_TO_PKGS[i];
                }
            }
        }
        return null;
    }
    
    
    public static CharSequence replaceLabel(PackageManager pm, ApplicationInfo info) {
        if (IS_SUPPORT_LABEL_MAR) {
            final String replacedPkg = getReplacedPkg(info.packageName);
            if (replacedPkg != null) {//屬于label要被替代的應(yīng)用
                ApplicationInfo replacedInfo;
                try {
                    replacedInfo = pm.getApplicationInfo(replacedPkg, PackageManager.GET_META_DATA);
                    if (replacedInfo != null) {
                        final CharSequence loadLabel = replacedInfo.loadLabel(pm);
                        Log.d(TAG, "replace from [" + info.packageName + "] to [" + replacedPkg + "]");
                        Log.d(TAG, "replace from [" + info.loadLabel(pm) + "] to [" + loadLabel + "]");

                        return loadLabel;
                    }
                } catch (NameNotFoundException e) {
                    e.printStackTrace();
                }

            }
        }
        return info.loadLabel(pm);
    }

    public static CharSequence loadUnsafeLabel(PackageManager pm, String fromPkgName, int labelRes,
            ApplicationInfo appinfo) {
        String toPkgName = getReplacedPkg(fromPkgName);
        Log.d(LabelManager.TAG, "loadUnsafeLabel toPkgName:" +toPkgName);

        if (toPkgName != null) {//改應(yīng)用屬于是要被替代的應(yīng)用
            ApplicationInfo applicationInfo = null;
            try {
                //使用替代的ApplicationInfo
                applicationInfo = pm.getApplicationInfo(toPkgName, PackageManager.GET_META_DATA);
            } catch (NameNotFoundException e) {
                e.printStackTrace();
            }
            Log.d(LabelManager.TAG, "loadUnsafeLabel applicationInfo:" +applicationInfo);

            if (applicationInfo != null) {
                //獲取目標(biāo)App的應(yīng)用名稱
                CharSequence toLabel = pm.getText(toPkgName, labelRes, applicationInfo);
                Log.d(LabelManager.TAG, "loadUnsafeLabel replace[" + fromPkgName + "] to " + toPkgName + " label:" + toLabel);
                if (toLabel != null) {
                    return toLabel.toString().trim();
                }
            }
        }
        return pm.getText(fromPkgName, labelRes, appinfo);
    }

}

應(yīng)用名稱映射效果圖

應(yīng)用名稱替換效果.png

解決方案 2

通過動態(tài)資源覆蓋的方式妻柒,實(shí)現(xiàn),參考章節(jié):

優(yōu)缺點(diǎn):

方案一:

優(yōu)點(diǎn):每次修改資源耘分,只需修改和編譯ResApp举塔,方便改動和發(fā)布
缺點(diǎn):ResApp中的資源文件屬性,需要跟MainApp保持一次求泰,不能刪除和增加屬性字段央渣;
需要修改系統(tǒng),雖然改動不大

方案二:

優(yōu)點(diǎn):只需添加要覆蓋的資源文件即可
缺點(diǎn):每次資源改動渴频,需要重新編譯系統(tǒng)芽丹,改動和發(fā)布比較麻煩

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市卜朗,隨后出現(xiàn)的幾起案子拔第,更是在濱河造成了極大的恐慌咕村,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蚊俺,死亡現(xiàn)場離奇詭異懈涛,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)泳猬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進(jìn)店門批钠,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人得封,你說我怎么就攤上這事埋心。” “怎么了忙上?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵踩窖,是天一觀的道長。 經(jīng)常有香客問我晨横,道長,這世上最難降的妖魔是什么箫柳? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任手形,我火速辦了婚禮,結(jié)果婚禮上悯恍,老公的妹妹穿的比我還像新娘库糠。我一直安慰自己,他們只是感情好涮毫,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布瞬欧。 她就那樣靜靜地躺著,像睡著了一般罢防。 火紅的嫁衣襯著肌膚如雪艘虎。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天咒吐,我揣著相機(jī)與錄音野建,去河邊找鬼。 笑死恬叹,一個胖子當(dāng)著我的面吹牛候生,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播绽昼,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼唯鸭,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了硅确?” 一聲冷哼從身側(cè)響起目溉,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤明肮,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后停做,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體晤愧,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年蛉腌,在試婚紗的時候發(fā)現(xiàn)自己被綠了官份。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡烙丛,死狀恐怖舅巷,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情河咽,我是刑警寧澤钠右,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站忘蟹,受9級特大地震影響飒房,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜媚值,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一狠毯、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧褥芒,春花似錦嚼松、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至坷牛,卻和暖如春罕偎,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背京闰。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工锨亏, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人忙干。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓器予,卻偏偏與公主長得像,于是被迫代替她去往敵國和親捐迫。 傳聞我的和親對象是個殘疾皇子乾翔,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評論 2 345

推薦閱讀更多精彩內(nèi)容

  • feisky云計(jì)算反浓、虛擬化與Linux技術(shù)筆記posts - 1014, comments - 298, trac...
    不排版閱讀 3,815評論 0 5
  • 簡述: ? ?一直希望有個機(jī)會可以好好研究一下android手機(jī)的多主題功能辆雾,借此機(jī)會將自己所能分析到的內(nèi)容記錄...
    大天使之劍閱讀 1,810評論 0 6
  • 1.介紹 如果你正在查閱build.gradle文件的所有可選項(xiàng)猜揪,請點(diǎn)擊這里進(jìn)行查閱:DSL參考 1.1新構(gòu)建系統(tǒng)...
    Chuckiefan閱讀 12,118評論 8 72
  • Author:楊空明 Date:2018-8-17 一、前言 Android開發(fā)者常常面臨的一個問題就是防破解钧萍、 ...
    問心2018閱讀 28,644評論 4 66
  • 往復(fù)讀書夢划煮,挑燈夜未央 (作者:張倩倩) 歲月安然的日子里 我曾把在路上的風(fēng)景 點(diǎn)上未央的雅名 我把讀書的的夢 綴...
    紅巧兒閱讀 31評論 0 1