动态加载案例

定义

所谓的动态加载就是动态的执行不通的代码,只需要将最新的代码放在服务器上即可,避免了频繁的更新apk,提高了用户的体验

分类

Android项目中,动态加载按技术实现上的区别大致可以分为两种:

一 . 动态加载.dex/jar/apk(现在动态加载普遍说的是这种)
- 步骤(以动态引用APK文件为例子):
(1)编写Remote工程——新建一个Remote项目,并在其中添加一个如下类:

public class RemoteService {
    public String getInfo(String info){
        return "动态加载成功了" + info;
    }
}

运行这个Remote工程,在bin目录先会生成一个remote.apk文件,使用studio开发的同学会在build/output/apk/目录下,将这个apk文件放在你的服务器上,也可以使用.jar和.dex的形式
(2)编写MyProject工程
首先创建一个App类,继承Applicaion,在oncreate方法中下载最新的apk文件

package com.stv.demo;

import android.app.Application;
import android.widget.Toast;

import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;

/**
 * Created by lenovo on 2016/12/20.
 */
public class App extends Application {
    FileOutputStream fos = null;
    HttpURLConnection conn = null;

    @Override
    public void onCreate() {
        super.onCreate();
        new Thread(new Runnable() {
            @Override
            public void run() {
                loadApk();
            }
        }).start();
    }

    private void loadApk() {
        try {
            File file = new File("data/data/com.stv.demo/remote.apk");
            if (file.exists()) {
                file.delete();
            }
            URL url = new URL("http://10.58.180.214:8080/remote.apk");
            conn = (HttpURLConnection) url.openConnection();//一定是这种类型
            conn.setReadTimeout(5000);//设置客户端连接超时间隔,如果超过指定时间  服务端还没有响应 就不要等了
            //判断服务端正常的反馈是否已经到达了 客户端
            if(conn.getResponseCode()==HttpURLConnection.HTTP_OK){

                //获得网络字节输入流对象
                InputStream is=conn.getInputStream();

                //建立内存到硬盘的连接
                fos=new FileOutputStream(file);

                //老三样 写文件
                byte[] b=new byte[1024];
                int len=0;
                while((len=is.read(b))!=-1){  //先读到内存
                    fos.write(b, 0, len);
                }
                fos.flush();
                Toast.makeText(App.this,"下载成功",Toast.LENGTH_LONG).show();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

在主界面添加一个按钮,在按钮的点击事件中使用反射的方式去加载apk中的代码

package com.stv.demo;

import android.os.Bundle;
import android.os.Environment;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Toast;

import java.lang.reflect.Method;

import dalvik.system.DexClassLoader;

public class MainActivity extends AppCompatActivity {
    private DexClassLoader dexClassLoader;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        //第 1 步:装载APK文件
        //定义优化目录:/data/data/com.songshi.myproject
        String optimizedDirectory = Environment.getDataDirectory().toString() + "/data/" + getPackageName();
        dexClassLoader = new DexClassLoader("/data/data/com.stv.demo/remote.apk", optimizedDirectory, null, ClassLoader.getSystemClassLoader());

        findViewById(R.id.load).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                try {
                    //第 2 步:装载要访问的类
                    Class c = dexClassLoader.loadClass("com.stv.demo.remote.RemoteService");   //Call requires API level 14 (current min is 8)

                    //第 3 步:创建类的对象
                    Object obj = c.newInstance();
                    //第 4 步:用Java反射技术调用ServiceClass类中的addService方法
                    Method method = obj.getClass().getMethod("getInfo", String.class);
                    String info = String.valueOf(method.invoke(obj, "Hello World"));

                    Toast.makeText(MainActivity.this, info, Toast.LENGTH_LONG).show();
                } catch (Exception e) {
                    Toast.makeText(MainActivity.this, "error:" + e.getMessage(), Toast.LENGTH_LONG).show();
                }
            }
        });
    }
}

效果图如下:
这里写图片描述

修改Remote工程的代码如下:并将新生成的apk文件更新到服务器上

public class RemoteService {
    public String getInfo(String info){
        return "修改了代码" + info;
    }
}

不需要修改Demo工程的代码,效果图如下:
这里写图片描述

二 . 动态加载.so库
动态加载so库没有使用过,请自行研究

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值