天天看點

Android應用開發提高系列(5)——Android動态加載(下)——加載已安裝APK中的類和資源

正文

  一、目标

    注意被調用的APK在Android系統中是已經安裝的。

    從目前APK中調用另外一個已安裝APK的字元串、顔色值、圖檔、布局檔案資源以及Activity。

     

  二、實作

    2.1   被調用工程

       基本沿用上個工程的,添加了被調用的字元串、圖檔等,是以這裡就不貼了,後面有下載下傳工程的連結。

    2.2   調用工程代碼

public class TestAActivity extends Activity {

    /** TestB包名 */

    private static final String PACKAGE_TEST_B = "com.nmbb.b";

    @Override

    public void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.main);

        try {

            final Context ctxTestB = getTestBContext();

            Resources res = ctxTestB.getResources();

            // 擷取字元串string

            String hello = res.getString(getId(res, "string", "hello"));

            ((TextView) findViewById(R.id.testb_string)).setText(hello);

            // 擷取圖檔Drawable

            Drawable drawable = res

                    .getDrawable(getId(res, "drawable", "testb"));

            ((ImageView) findViewById(R.id.testb_drawable))

                    .setImageDrawable(drawable);

            // 擷取顔色值

            int color = res.getColor(getId(res, "color", "white"));

            ((TextView) findViewById(R.id.testb_color))

                    .setBackgroundColor(color);

            // 擷取布局檔案

            View view = getView(ctxTestB, getId(res, "layout", "main"));

            LinearLayout layout = (LinearLayout) findViewById(R.id.testb_layout);

            layout.addView(view);

            // 啟動TestB Activity

            findViewById(R.id.testb_activity).setOnClickListener(

                    new OnClickListener() {

                        @Override

                        public void onClick(View v) {

                            try {

                                @SuppressWarnings("rawtypes")

                                Class cls = ctxTestB.getClassLoader()

                                        .loadClass("com.nmbb.TestBActivity");

                                startActivity(new Intent(ctxTestB, cls));

                            } catch (ClassNotFoundException e) {

                                e.printStackTrace();

                            }

                        }

                    });

        } catch (NameNotFoundException e) {

            e.printStackTrace();

        }

    }

    /**

     * 擷取資源對應的編号

     * 

     * @param testb

     * @param resName

     * @param resType

     *            layout、drawable、string

     * @return

     */

    private int getId(Resources testb, String resType, String resName) {

        return testb.getIdentifier(resName, resType, PACKAGE_TEST_B);

     * 擷取視圖

     * @param ctx

     * @param id

    public View getView(Context ctx, int id) {

        return ((LayoutInflater) ctx

                .getSystemService(Context.LAYOUT_INFLATER_SERVICE)).inflate(id,

                null);

     * 擷取TestB的Context

     * @throws NameNotFoundException

    private Context getTestBContext() throws NameNotFoundException {

        return createPackageContext(PACKAGE_TEST_B,

                Context.CONTEXT_IGNORE_SECURITY | Context.CONTEXT_INCLUDE_CODE); 

    } 

    代碼說明:

      基本原理:通過package擷取被調用應用的Context,通過Context擷取相應的資源、類。

    注意:

      a).  網上許多文章是通過目前工程的R.id來調用被調用工程的資源 ,這是錯誤的,即使不報錯那也是湊巧,因為R是自動生成的,兩個應用的id是沒有辦法對應的,是以需要通過getIdentifier來查找。

      b).   Context.CONTEXT_INCLUDE_CODE一般情況下是不需要加的,如果layout裡面包含了自定義控件,就需要加上。注意不能在目前工程強制轉換獲得這個自定義控件,因為這是在兩個ClassLoader中,無法轉換。

      c).    擷取這些資源是不需要shareUserId的。

  三、總結

    與上篇文章相比,擷取資源更加友善,但也存在一些限制:

    3.1  被調用的apk必須已經安裝,降低使用者體驗。

    3.2  style是無法動态設定的,即使能夠取到。 

    3.3  從目前研究結果來看,被調用工程如果使用自定義控件,會受到比較大的限制,不能強制轉換使用(原因前面已經講過)。

    3.4  由于一個工程裡面混入了兩個Context,比較容易造成混淆,取資源也比較麻煩。這裡分享一下批量隐射兩個apk id的辦法,可以通過反射擷取兩個apk的R類,一次擷取每一個id和值,通過名稱一一比對上,這樣就不用手工傳入字元串了。

    @SuppressWarnings("rawtypes")

    private static HashMap<String, Integer> getR(Class cls) throws ClassNotFoundException, InstantiationException, IllegalAccessException {

        HashMap<String, Integer> result = new HashMap<String, Integer>();

        for (Class r : cls.getClasses()) {

            if (!r.getName().endsWith("styleable")) {

                Object owner = r.newInstance();

                for (Field field : r.getFields()) {

                    result.put(field.getName(), field.getInt(owner));

                }

            }

        return result;

  四、下載下傳 

  五、文章

結束

  如果是做大面積的換膚,還比較複雜,這種方式也不是很友善,這也是為什麼現在市面上做換膚的很少,有也是很簡單的換膚。這幾天想到的另外一個方案,還沒有實踐,有效果了再拿出來分享,歡迎大家交流 :)

本文轉自over140 51CTO部落格,原文連結:http://blog.51cto.com/over140/844954,如需轉載請自行聯系原作者

繼續閱讀