温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

Android应用开发的版本更新检测升级功能怎么实现

发布时间:2022-04-27 16:24:23 来源:亿速云 阅读:254 作者:zzz 栏目:开发技术

这篇文章主要介绍“Android应用开发的版本更新检测升级功能怎么实现”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Android应用开发的版本更新检测升级功能怎么实现”文章能帮助大家解决问题。

    一.版本的基础知识

    版本控制的属性包括versionCode和versionName。

    (一)versionCode

    版本号(versionCode)是相对比较重要的一个属性。versionCode是一个Integer类型的值。

    所以大家在设置的时候,不要将versionCode设置的太大,最好不要超过Integer的取值范围(当然一般也是不会超过的),一般大家在发布自己的第一个应用到市场的时候,版本取值为1(versionCode=1),这也是目前典型和普遍的做法。然后,每次发布更新版本时可以递增versionCode的值。

    (二)versionName

    版本名(versionName)一个值为String类型的属性,一般和VersionCode成对出现。

    VersionCode是方便程序开发者运行和维护Application而设置的一个有效的值。versionName是一个版本的描述,给用户看的,也是用户放在各个第3方平台上提供给使用者看的一个版本名,可以说是对VersionCode的解释和描述。一般格式可以为:1.1.2。(major.minor.point)的形式。

    (三)版本控制小结

    版本号(versionCode)是用于判断是否升级的,一般每次版本更新,版本号加一。

    如果获取服务器上的版本号比检测到本程序的版本号高,那么提示升级。

    版本名(versionName)用于显示版本改变的幅度大小,比如从2.0.1改变为2.0.2可能只是修改了一个很小的debug,如果改变为2.1.0可能是新增了一些功能,如果改变为3.0.0可能是有很大幅度的修改,比如很多UI界面或功能的添加!

    也就是版本号用于判断是否可以升级,而版本名用于显示給用户看!

    (四)版本控制的文件位置

    这个要区分你是用Eclipse开发还是Studio开发。

    在Eclipse中版本控制的属性的位置是Manifest.xml中,如:

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        android:versionCode="3"
        android:versionName="1.2.1"
        package="com.example.updateDemo">
        <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:theme="@style/AppTheme">
        </application>
    </manifest>

    上面表示android的第三个版本程序,版本名:1.2.1

    在Android Studio呢?也是可以在Manifest.xml中定义versionCode和versionName,但是这里设置是无效的!

    需要在程序的build.grade文件中设置,图解:

    Android应用开发的版本更新检测升级功能怎么实现

    上面表示android的第三个版本程序,版本名:3.0.1

    (五)版本信息的获取,代码

    这里指的是获取运行中的程序的版本号,代码如下:

    1

    /*
     * 获取当前程序的版本名
     */
    private String getVersionName() throws Exception{
        //获取packagemanager的实例
        PackageManager packageManager = getPackageManager();
        //getPackageName()是你当前类的包名,0代表是获取版本信息
        PackageInfo packInfo = packageManager.getPackageInfo(getPackageName(), 0);
        Log.e("TAG","版本号"+packInfo.versionCode);
        Log.e("TAG","版本名"+packInfo.versionName);
        return packInfo.versionName;
    }

    2

    /*
     * 获取当前程序的版本号
     */
    private int getVersionCode() throws Exception{
        //获取packagemanager的实例
        PackageManager packageManager = getPackageManager();
        //getPackageName()是你当前类的包名,0代表是获取版本信息
        PackageInfo packInfo = packageManager.getPackageInfo(getPackageName(), 0);
        Log.e("TAG","版本号"+packInfo.versionCode);
        Log.e("TAG","版本名"+packInfo.versionName);
        return packInfo.versionCode;
    }

    (六)版本更新中重要的代码块:

    1.获取本程序的版本号和版本名

    上面已经有了

    2.从服务器获取到一串json数据

    里面包含最新程序的版本号和版本名、新版本信息等数据,需要自己解析到达对应的数据,因为服务器的数据不一样,所以这里的代码也不写!

    3.检测是否更新的代码
    //对比本程序的版本号和最新程序的版本号
    public void checkVersion(View view) {//按钮!
            //如果检测本程序的版本号小于服务器的版本号,那么提示用户更新
            if (getVersionCode() < serviceVersionCOde) {
                showDialogUpdate();//弹出提示版本更新的对话框
            }else{
                //否则吐司,说现在是最新的版本
                Toast.makeText(this,"当前已经是最新的版本",Toast.LENGTH_SHORT).show();
            }
    }
    4.弹出更新提示的对话框的代码
    /**
     * 提示版本更新的对话框
     */
    private void showDialogUpdate() {
        // 这里的属性可以一直设置,因为每次设置后返回的是一个builder对象
        AlertDialog.Builder builder = new AlertDialog.Builder(this);
        // 设置提示框的标题
        builder.setTitle("版本升级").
                // 设置提示框的图标
                        setIcon(R.mipmap.ic_launcher).
                // 设置要显示的信息
                        setMessage("发现新版本!请及时更新").
                // 设置确定按钮
                        setPositiveButton("确定", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        //Toast.makeText(MainActivity.this, "选择确定哦", 0).show();
                        loadNewVersionProgress();//下载最新的版本程序
                    }
                }).
                // 设置取消按钮,null是什么都不做,并关闭对话框
                        setNegativeButton("取消", null);
        // 生产对话框
        AlertDialog alertDialog = builder.create();
        // 显示对话框
        alertDialog.show();
    }
    5.下载新版本程序的代码
    /**
     * 下载新版本程序,需要子线程
     */
    private void loadNewVersionProgress() {
      final   String uri="http://www.apk.anzhi.com/data3/apk/201703/14/4636d7fce23c9460587d602b9dc20714_88002100.apk";
        final ProgressDialog pd;    //进度条对话框
        pd = new  ProgressDialog(this);
        pd.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
        pd.setMessage("正在下载更新");
        pd.show();
        //启动子线程下载任务
        new Thread(){
            @Override
            public void run() {
                try {
                    File file = getFileFromServer(uri, pd);
                    sleep(3000);
                    installApk(file);
                    pd.dismiss(); //结束掉进度条对话框
                } catch (Exception e) {
                    //下载apk失败
                    Toast.makeText(getApplicationContext(), "下载新版本失败", Toast.LENGTH_LONG).show();
                    e.printStackTrace();
                }
            }}.start();
    }
    6.根据Uri网址获得apk文件对象的代码
    /**
     * 从服务器获取apk文件的代码
     * 传入网址uri,进度条对象即可获得一个File文件
     * (要在子线程中执行哦)
     */
    public static File getFileFromServer(String uri, ProgressDialog pd) throws Exception{
        //如果相等的话表示当前的sdcard挂载在手机上并且是可用的
        if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
            URL url = new URL(uri);
            HttpURLConnection conn =  (HttpURLConnection) url.openConnection();
            conn.setConnectTimeout(5000);
            //获取到文件的大小
            pd.setMax(conn.getContentLength());
            InputStream is = conn.getInputStream();
            long time= System.currentTimeMillis();//当前时间的毫秒数
            File file = new File(Environment.getExternalStorageDirectory(), time+"updata.apk");
            FileOutputStream fos = new FileOutputStream(file);
            BufferedInputStream bis = new BufferedInputStream(is);
            byte[] buffer = new byte[1024];
            int len ;
            int total=0;
            while((len =bis.read(buffer))!=-1){
                fos.write(buffer, 0, len);
                total+= len;
                //获取当前下载量
                pd.setProgress(total);
            }
            fos.close();
            bis.close();
            is.close();
            return file;
        }
        else{
            return null;
        }
    }
    7.安装apk文件的代码
    /**
     * 安装apk
     */
    protected void installApk(File file) {
        Intent intent = new Intent();
        //执行动作  
        intent.setAction(Intent.ACTION_VIEW);
        //执行的数据类型  
        intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");
        startActivity(intent);
    }

    上面有些代码块的格式是固定的,有些是可以根据实际情况进行修改。

    二.程序更新的简单示例一

    程序图:

    Android应用开发的版本更新检测升级功能怎么实现

    点击“版本更新”按钮,弹出版本升级对话框。

    说明:因为这里没有服务器,所以假设从服务器中获得版本号为3,而本程序的版本号为2,会弹出升级提示的对话框,点击升级后会链接到下载apk文件的地址,并实现下载,最后安装。

    (这里apk地址,很容易拿到,你上任何一个应用市场,在电脑上下载apk文件时,都会显示一个uri地址,这里的地址我用的是我之前在安智应用市场上传的一个小应用“wenzhi日历”)

    代码:

    (一)MainActivity的设计

    package com.example.updateDemo;
    import android.app.AlertDialog;
    import android.app.ProgressDialog;
    import android.content.DialogInterface;
    import android.content.Intent;
    import android.content.pm.PackageInfo;
    import android.content.pm.PackageManager;
    import android.net.Uri;
    import android.os.Environment;
    import android.os.Handler;
    import android.os.Message;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    import android.widget.TextView;
    import android.widget.Toast;
    import java.io.BufferedInputStream;
    import java.io.File;
    import java.io.FileOutputStream;
    import java.io.InputStream;
    import java.net.HttpURLConnection;
    import java.net.URL;
    import java.util.Currency;
    public class MainActivity extends AppCompatActivity {
        private static final int DOWN_ERROR = 505;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            //在页面上显示版本信息
            TextView tv_versionName = (TextView) findViewById(R.id.tv_versionName);
            try {
                tv_versionName.setText("版本名:" + getVersionName());
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        //检测本程序的版本,这里假设从服务器中获取到最新的版本号为3
        public void checkVersion(View view) {
                //如果检测本程序的版本号小于服务器的版本号,那么提示用户更新
                if (getVersionCode() < 3) {
                    showDialogUpdate();//弹出提示版本更新的对话框
                }else{
                    //否则吐司,说现在是最新的版本
                    Toast.makeText(this,"当前已经是最新的版本",Toast.LENGTH_SHORT).show();
                }
        }
        /**
         * 提示版本更新的对话框
         */
        private void showDialogUpdate() {
            // 这里的属性可以一直设置,因为每次设置后返回的是一个builder对象
            AlertDialog.Builder builder = new AlertDialog.Builder(this);
            // 设置提示框的标题
            builder.setTitle("版本升级").
                    // 设置提示框的图标
                            setIcon(R.mipmap.ic_launcher).
                    // 设置要显示的信息
                            setMessage("发现新版本!请及时更新").
                    // 设置确定按钮
                            setPositiveButton("确定", new DialogInterface.OnClickListener() {
                        @Override
                        public void onClick(DialogInterface dialog, int which) {
                            //Toast.makeText(MainActivity.this, "选择确定哦", 0).show();
                            loadNewVersionProgress();//下载最新的版本程序
                        }
                    }).
                    // 设置取消按钮,null是什么都不做,并关闭对话框
                            setNegativeButton("取消", null);
            // 生产对话框
            AlertDialog alertDialog = builder.create();
            // 显示对话框
            alertDialog.show();
        }
        /**
         * 下载新版本程序
         */
        private void loadNewVersionProgress() {
          final   String uri="http://www.apk.anzhi.com/data3/apk/201703/14/4636d7fce23c9460587d602b9dc20714_88002100.apk";
            final ProgressDialog pd;    //进度条对话框
            pd = new  ProgressDialog(this);
            pd.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
            pd.setMessage("正在下载更新");
            pd.show();
            //启动子线程下载任务
            new Thread(){
                @Override
                public void run() {
                    try {
                        File file = getFileFromServer(uri, pd);
                        sleep(3000);
                        installApk(file);
                        pd.dismiss(); //结束掉进度条对话框
                    } catch (Exception e) {
                        //下载apk失败
                        Toast.makeText(getApplicationContext(), "下载新版本失败", Toast.LENGTH_LONG).show();
                        e.printStackTrace();
                    }
                }}.start();
        }
        /**
         * 安装apk
         */
        protected void installApk(File file) {
            Intent intent = new Intent();
            //执行动作  
            intent.setAction(Intent.ACTION_VIEW);
            //执行的数据类型  
            intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");
            startActivity(intent);
        }
        /**
         * 从服务器获取apk文件的代码
         * 传入网址uri,进度条对象即可获得一个File文件
         * (要在子线程中执行哦)
         */
        public static File getFileFromServer(String uri, ProgressDialog pd) throws Exception{
            //如果相等的话表示当前的sdcard挂载在手机上并且是可用的
            if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
                URL url = new URL(uri);
                HttpURLConnection conn =  (HttpURLConnection) url.openConnection();
                conn.setConnectTimeout(5000);
                //获取到文件的大小
                pd.setMax(conn.getContentLength());
                InputStream is = conn.getInputStream();
                long time= System.currentTimeMillis();//当前时间的毫秒数
                File file = new File(Environment.getExternalStorageDirectory(), time+"updata.apk");
                FileOutputStream fos = new FileOutputStream(file);
                BufferedInputStream bis = new BufferedInputStream(is);
                byte[] buffer = new byte[1024];
                int len ;
                int total=0;
                while((len =bis.read(buffer))!=-1){
                    fos.write(buffer, 0, len);
                    total+= len;
                    //获取当前下载量
                    pd.setProgress(total);
                }
                fos.close();
                bis.close();
                is.close();
                return file;
            }
            else{
                return null;
            }
        }
        /*
         * 获取当前程序的版本名
         */
        private String getVersionName() throws Exception {
            //获取packagemanager的实例
            PackageManager packageManager = getPackageManager();
            //getPackageName()是你当前类的包名,0代表是获取版本信息
            PackageInfo packInfo = packageManager.getPackageInfo(getPackageName(), 0);
            Log.e("TAG", "版本号" + packInfo.versionCode);
            Log.e("TAG", "版本名" + packInfo.versionName);
            return packInfo.versionName;
        }
        /*
         * 获取当前程序的版本号
         */
        private int getVersionCode() {
            try {
                //获取packagemanager的实例
                PackageManager packageManager = getPackageManager();
                //getPackageName()是你当前类的包名,0代表是获取版本信息
                PackageInfo packInfo = packageManager.getPackageInfo(getPackageName(), 0);
                Log.e("TAG", "版本号" + packInfo.versionCode);
                Log.e("TAG", "版本名" + packInfo.versionName);
                return packInfo.versionCode;
            } catch (Exception e) {
                e.printStackTrace();
            }
            return  1;
        }
    }

    (二)布局文件的设计

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical"
        >
    <TextView
        android:padding="10dp"
        android:gravity="center"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:textSize="25sp"
        android:id="@+id/tv_versionName"
        android:text="版本信息"
        />
        <Button
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="检测更新"
            android:onClick="checkVersion"
            />
    </LinearLayout>

    布局文件是非常简单的,一个TextView显示版本名,一个Button检测本程序的版本号对比服务器的版本号,并提示是否升级。

    (三)最后不要忘记添加网络和读写权限,

    <uses-permission android:name="android.permission.INTERNET"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

    程序运行的效果:

    Android应用开发的版本更新检测升级功能怎么实现

    有些程序在刚启动的时候就检测版本升级,其实就是在onCreate方法里面执行。

    三.程序更新的示例二

    上面的程序升级不是覆盖原来的程序,因为我的两个程序的包名不一样的,第二个程序是直接安装在手机上面的,简单的说就是安装了两个程序。

    如果要实现覆盖效果,需要包名相同并且签名相同。下面一个程序就实现了这种效果。

    下面是两个我之前上线的程序,名字都是“wenzhi日历”,功能基本是一样的,但是一个是在“应用宝”应用市场上线的,版本是3.0;另一个是在“安智”应用市场上线的,版本是2.0;

    两个版本都有版本更新功能,在2.0版本点击更新,可以下载3.0版本的程序,在3.0版本点击更新,提示:已经是最新版本!

    注意:程序示例一的程序更新,下载下来的程序就是我在“安智”上线的应用程序,也就是本示例要用到的版本2.0的程序,点击版本更新可以直接覆盖原来的程序,得到版本3.0的程序。

    效果:

    Android应用开发的版本更新检测升级功能怎么实现

    代码:这里就不展示了,其实更新功能代码跟上面第一个程序差不多,其余的很多文件都是关于日历的设计部分。

    示例二的效果只是为了实现一种仿真升级的效果,大家没有必要模仿。

    wenzhi日历(版本2.0)

    Android应用开发的版本更新检测升级功能怎么实现

    wenzhi日历(版本3.0)

    Android应用开发的版本更新检测升级功能怎么实现

    还有一个需要注意的是如果手机安装了版本高的程序,想在安装同一个包名并且版本低的程序,是无法安装的(提示安装失败),只有先卸载高版本程序后才能安装低版本程序。

    关于“Android应用开发的版本更新检测升级功能怎么实现”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注亿速云行业资讯频道,小编每天都会为大家更新不同的知识点。

    向AI问一下细节

    免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

    AI