Android 提示版本更新
2017-02-21 14:19
405 查看
前言:在软件开发的尾声应该都会遇到这个问题,还好网上资料很多,所以基本不费什么力气就搞定了,现记录于下。这里用的PHP服务器。
效果图:(PHP服务器)
初始界面 [b]检测后,如果已是最新版[/b]
如果不是最新版,提示更新 正在下载 [b]安装新程序 [/b]
其中,android:versionCode和android:versionName两个字段分别表示版本代码,版本名称。versionCode是整型数字,versionName是字符串。由于version是给用户看的,不太容易比较大小,升级检查时,可以以检查versionCode为主,方便比较出版本的前后大小。
那么,在应用中如何读取AndroidManifest.xml中的versionCode和versionName呢?可以使用PackageManager的API,参考以下代码:
这里要注意一个地方:代码里的“com.example.try_downloadfile_progress”对应AndroidManifest.xml里的package="……"部分
这里除了上面我们提到的两个函数getVerCode和getVerName外,还有几个常量和一个函数定义,含义分别如下:
SERVER_IP:服务器IP地址(大家在拿到试验的时候,要改成自己服务器IP地址)
SERVER_ADDRESS:android程序要将请求发送到的页面地址,无须更改。
UPDATESOFTADDRESS:更新安装包存放的位置,无须更改。
函数StringBuilder post_to_server(List<NameValuePair> vps)是访问服务器并返回结果的功能封装。传进去名值对,返回StringBuilder对象
1、OnCreate函数:
先从主函数开始讲,OnCreate函数中只有两部分,一个是initVariable()初始化变量,这个不多说,难度不大;第二个是为版本检测按钮设置监听函数——btnClickListener,而在btnClickListener函数中可以明显的看到,其中也只有一个类checkNewestVersionAsyncTask,这里采用异步方式处理更新问题。下面看checkNewestVersionAsyncTask函数
2、checkNewestVersionAsyncTask函数
(1)首先看后台操作doInBackground
首先利用postCheckNewestVersionCommand2Server()函数将请求发送到服务器,该函数根据是否请求成功返回TRUE或FALSE,然后将从服务器上获取的版本代码与本地的版本代码进行比较,如果要更新返回TRUE,如果不要更新返回FASLE。
下面看看postCheckNewestVersionCommand2Server()的代码:
这里就是构建名值对,然后发向服务器,如果获取到了值就返回TURE,如果没获取到值,就返回FALSE。服务器返回的JSON值为:
对于服务器代码,由于是用PHP写的,这里就不再列出了,在源码里有。
(2)onPostExecute()
继续第一部分,在doInBackground操作完成后,如果需要更新doInBackground返回TRUE,否则返回FASLE,所以在onPostExecute中根据result不同调用不同的函数,利用doNewVersionUpdate(); 提示用户更新最新版本。利用notNewVersionDlgShow();
/提示用户当前即为最新版本,无需更新。
对于notNewVersionDlgShow()函数仅仅是创建了个对话框,没什么实体内容,就不再具体讲解。下面具体讲述doNewVersionUpdate()下载,更新与安装程序的过程。
3、doNewVersionUpdate()提示版本更新
具体代码如下:
这里创建一个具有确定按钮和取消按钮功能的对话框,如果用户点击取消按钮,会利用finish()结束掉程序运行;如果点击确定按钮,则利用 downFile(Common.UPDATESOFTADDRESS); 函数开始程序下载,其中downFile()函数传进去的参数是APP所在的服务器地址。注意这里的地址要具体到下载文件,比如这里是http://192.168.1.105/server/XXX.apk
4、downFile(final String url)下载并显示进度
具体代码如下:
通过利用httpClient的get方法,获取指定URL的内容,然后写到本地SD卡中,对于进度条,首先利用m_progressDlg.setMax((int)length);设置进度条的最大值,然后在读取返回结果并写到本地时,利用 m_progressDlg.setProgress(count);设置当前进度。在全部写完以后,调用down()函数,通知HANDER安装程序。
5、安装程序
安装程序主要利用下面两个函数:
由于在android程序中必须依循单线程操作UI,所以在非主线程中不能操作UI,否则程序会崩掉,具体参见:《AsnyncTask与handler(一)——AsyncTask异步处理》与《AsnyncTask与handler(二)——handler消息机制》。所以这里作用handler的方式更新UI。
好了,到这就全部讲完了,下面给出客户端与服务器端源码,懂PHP的童鞋赚到了有木有……
源码地址:http://download.csdn.net/detail/harvic880925/7309013 不要分,仅供分享。
请大家尊重作者原创版权,转载请标明出处:http://blog.csdn.net/harvic880925/article/details/25191159 不胜感激。
效果图:(PHP服务器)
初始界面 [b]检测后,如果已是最新版[/b]
如果不是最新版,提示更新 正在下载 [b]安装新程序 [/b]
一、准备知识
在AndroidManifest.xml里定义了每个Android apk的版本标识:<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.try_downloadfile_progress" android:versionCode="1" android:versionName="1.0" >
其中,android:versionCode和android:versionName两个字段分别表示版本代码,版本名称。versionCode是整型数字,versionName是字符串。由于version是给用户看的,不太容易比较大小,升级检查时,可以以检查versionCode为主,方便比较出版本的前后大小。
那么,在应用中如何读取AndroidManifest.xml中的versionCode和versionName呢?可以使用PackageManager的API,参考以下代码:
/** * 获取软件版本号 * @param context * @return */ public static int getVerCode(Context context) { int verCode = -1; try { //注意:"com.example.try_downloadfile_progress"对应AndroidManifest.xml里的package="……"部分 verCode = context.getPackageManager().getPackageInfo( "com.example.try_downloadfile_progress", 0).versionCode; } catch (NameNotFoundException e) { Log.e("msg",e.getMessage()); } return verCode; } /** * 获取版本名称 * @param context * @return */ public static String getVerName(Context context) { String verName = ""; try { verName = context.getPackageManager().getPackageInfo( "com.example.try_downloadfile_progress", 0).versionName; } catch (NameNotFoundException e) { Log.e("msg",e.getMessage()); } return verName; }
这里要注意一个地方:代码里的“com.example.try_downloadfile_progress”对应AndroidManifest.xml里的package="……"部分
二、XML代码
XML代码非常简单,就是如初始化界面那样,在里面加一个BUTTON而已。代码如下:<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="fill_parent" android:layout_height="fill_parent" tools:context=".MainActivity" > <Button android:id="@+id/chek_newest_version" android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_margin="5dip" android:text="检测最新版本"/> </RelativeLayout>
三、辅助类构建(Common)
这里为了开发方便,将一些公共的函数,单独放在Common类中实现,代码如下:package com.example.try_downloadfile_progress; /** * @author harvic * @date 2014-5-7 * @address http://blog.csdn.net/harvic880925 */ import java.io.BufferedReader; import java.io.InputStreamReader; import java.util.List; import org.apache.http.HttpResponse; import org.apache.http.NameValuePair; import org.apache.http.client.entity.UrlEncodedFormEntity; import org.apache.http.client.methods.HttpPost; import org.apache.http.impl.client.DefaultHttpClient; import org.apache.http.protocol.HTTP; import android.content.Context; import android.content.pm.PackageManager.NameNotFoundException; import android.util.Log; public class Common { public static final String SERVER_IP="http://192.168.1.105/"; public static final String SERVER_ADDRESS=SERVER_IP+"try_downloadFile_progress_server/index.php";//软件更新包地址 public static final String UPDATESOFTADDRESS=SERVER_IP+"try_downloadFile_progress_server/update_pakage/baidu.apk";//软件更新包地址 /** * 向服务器发送查询请求,返回查到的StringBuilder类型数据 * * @param ArrayList * <NameValuePair> vps POST进来的参值对 * @return StringBuilder builder 返回查到的结果 * @throws Exception */ public static StringBuilder post_to_server(List<NameValuePair> vps) { DefaultHttpClient httpclient = new DefaultHttpClient(); try { HttpResponse response = null; // 创建httpost.访问本地服务器网址 HttpPost httpost = new HttpPost(SERVER_ADDRESS); StringBuilder builder = new StringBuilder(); httpost.setEntity(new UrlEncodedFormEntity(vps, HTTP.UTF_8)); response = httpclient.execute(httpost); // 执行 if (response.getEntity() != null) { // 如果服务器端JSON没写对,这句是会出异常,是执行不过去的 BufferedReader reader = new BufferedReader( new InputStreamReader(response.getEntity().getContent())); String s = reader.readLine(); for (; s != null; s = reader.readLine()) { builder.append(s); } } return builder; } catch (Exception e) { // TODO: handle exception Log.e("msg",e.getMessage()); return null; } finally { try { httpclient.getConnectionManager().shutdown();// 关闭连接 // 这两种释放连接的方法都可以 } catch (Exception e) { // TODO Auto-generated catch block Log.e("msg",e.getMessage()); } } } /** * 获取软件版本号 * @param context * @return */ public static int getVerCode(Context context) { int verCode = -1; try { //注意:"com.example.try_downloadfile_progress"对应AndroidManifest.xml里的package="……"部分 verCode = context.getPackageManager().getPackageInfo( "com.example.try_downloadfile_progress", 0).versionCode; } catch (NameNotFoundException e) { Log.e("msg",e.getMessage()); } return verCode; } /** * 获取版本名称 * @param context * @return */ public static String getVerName(Context context) { String verName = ""; try { verName = context.getPackageManager().getPackageInfo( "com.example.try_downloadfile_progress", 0).versionName; } catch (NameNotFoundException e) { Log.e("msg",e.getMessage()); } return verName; } }
这里除了上面我们提到的两个函数getVerCode和getVerName外,还有几个常量和一个函数定义,含义分别如下:
SERVER_IP:服务器IP地址(大家在拿到试验的时候,要改成自己服务器IP地址)
SERVER_ADDRESS:android程序要将请求发送到的页面地址,无须更改。
UPDATESOFTADDRESS:更新安装包存放的位置,无须更改。
函数StringBuilder post_to_server(List<NameValuePair> vps)是访问服务器并返回结果的功能封装。传进去名值对,返回StringBuilder对象
四、主页面代码构建
1、首先设置AndroidManifest.xml,使其能访问网络和SD卡
在</manifest>标签上面,加入:<uses-permission android:name="android.permission.INTERNET" > </uses-permission> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" > </uses-permission> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" > </uses-permission>
2、主页代码:
先贴出全部代码,然后逐步讲解,全部代码如下:package com.example.try_downloadfile_progress; /** * @author harvic * @date 2014-5-7 * @address http://blog.csdn.net/harvic880925 */ import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.util.ArrayList; import java.util.List; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; import org.apache.http.NameValuePair; import org.apache.http.client.ClientProtocolException; import org.apache.http.client.HttpClient; import org.apache.http.client.methods.HttpGet; import org.apache.http.impl.client.DefaultHttpClient; import org.apache.http.message.BasicNameValuePair; import org.json.JSONArray; import android.net.Uri; import android.os.AsyncTask; import android.os.Bundle; import android.os.Environment; import android.os.Handler; import android.app.Activity; import android.app.AlertDialog; import android.app.Dialog; import android.app.ProgressDialog; import android.content.DialogInterface; import android.content.Intent; import android.util.Log; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; public class MainActivity extends Activity { Button m_btnCheckNewestVersion; long m_newVerCode; //最新版的版本号 String m_newVerName; //最新版的版本名 String m_appNameStr; //下载到本地要给这个APP命的名字 Handler m_mainHandler; ProgressDialog m_progressDlg; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); //初始化相关变量 initVariable(); m_btnCheckNewestVersion.setOnClickListener(btnClickListener); } private void initVariable() { m_btnCheckNewestVersion = (Button)findViewById(R.id.chek_newest_version); m_mainHandler = new Handler(); m_progressDlg = new ProgressDialog(this); m_progressDlg.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL); // 设置ProgressDialog 的进度条是否不明确 false 就是不设置为不明确 m_progressDlg.setIndeterminate(false); m_appNameStr = "haha.apk"; } OnClickListener btnClickListener = new View.OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub new checkNewestVersionAsyncTask().execute(); } }; class checkNewestVersionAsyncTask extends AsyncTask<Void, Void, Boolean> { @Override protected Boolean doInBackground(Void... params) { // TODO Auto-generated method stub if(postCheckNewestVersionCommand2Server()) { int vercode = Common.getVerCode(getApplicationContext()); // 用到前面第一节写的方法 if (m_newVerCode > vercode) { return true; } else { return false; } } return false; } @Override protected void onPostExecute(Boolean result) { // TODO Auto-generated method stub if (result) {//如果有最新版本 doNewVersionUpdate(); // 更新新版本 }else { notNewVersionDlgShow(); // 提示当前为最新版本 } super.onPostExecute(result); } @Override protected void onPreExecute() { // TODO Auto-generated method stub super.onPreExecute(); } } /** * 从服务器获取当前最新版本号,如果成功返回TURE,如果失败,返回FALSE * @return */ private Boolean postCheckNewestVersionCommand2Server() { StringBuilder builder = new StringBuilder(); JSONArray jsonArray = null; try { // 构造POST方法的{name:value} 参数对 List<NameValuePair> vps = new ArrayList<NameValuePair>(); // 将参数传入post方法中 vps.add(new BasicNameValuePair("action", "checkNewestVersion")); builder = Common.post_to_server(vps); jsonArray = new JSONArray(builder.toString()); if (jsonArray.length()>0) { if (jsonArray.getJSONObject(0).getInt("id") == 1) { m_newVerName = jsonArray.getJSONObject(0).getString("verName"); m_newVerCode = jsonArray.getJSONObject(0).getLong("verCode"); return true; } } return false; } catch (Exception e) { Log.e("msg",e.getMessage()); m_newVerName=""; m_newVerCode=-1; return false; } } /** * 提示更新新版本 */ private void doNewVersionUpdate() { int verCode = Common.getVerCode(getApplicationContext()); String verName = Common.getVerName(getApplicationContext()); String str= "当前版本:"+verName+" Code:"+verCode+" ,发现新版本:"+m_newVerName+ " Code:"+m_newVerCode+" ,是否更新?"; Dialog dialog = new AlertDialog.Builder(this).setTitle("软件更新").setMessage(str) // 设置内容 .setPositiveButton("更新",// 设置确定按钮 new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { m_progressDlg.setTitle("正在下载"); m_progressDlg.setMessage("请稍候..."); downFile(Common.UPDATESOFTADDRESS); //开始下载 } }) .setNegativeButton("暂不更新", new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int whichButton) { // 点击"取消"按钮之后退出程序 finish(); } }).create();// 创建 // 显示对话框 dialog.show(); } /** * 提示当前为最新版本 */ private void notNewVersionDlgShow() { int verCode = Common.getVerCode(this); String verName = Common.getVerName(this); String str="当前版本:"+verName+" Code:"+verCode+",/n已是最新版,无需更新!"; Dialog dialog = new AlertDialog.Builder(this).setTitle("软件更新") .setMessage(str)// 设置内容 .setPositiveButton("确定",// 设置确定按钮 new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { finish(); } }).create();// 创建 // 显示对话框 dialog.show(); } private void downFile(final String url) { m_progressDlg.show(); new Thread() { public void run() { HttpClient client = new DefaultHttpClient(); HttpGet get = new HttpGet(url); HttpResponse response; try { response = client.execute(get); HttpEntity entity = response.getEntity(); long length = entity.getContentLength(); m_progressDlg.setMax((int)length);//设置进度条的最大值 InputStream is = entity.getContent(); FileOutputStream fileOutputStream = null; if (is != null) { File file = new File( Environment.getExternalStorageDirectory(), m_appNameStr); fileOutputStream = new FileOutputStream(file); byte[] buf = new byte[1024]; int ch = -1; int count = 0; while ((ch = is.read(buf)) != -1) { fileOutputStream.write(buf, 0, ch); count += ch; if (length > 0) { m_progressDlg.setProgress(count); } } } fileOutputStream.flush(); if (fileOutputStream != null) { fileOutputStream.close(); } down(); } catch (ClientProtocolException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } }.start(); } private void down() { m_mainHandler.post(new Runnable() { public void run() { m_progressDlg.cancel(); update(); } }); } void update() { Intent intent = new Intent(Intent.ACTION_VIEW); intent.setDataAndType(Uri.fromFile(new File(Environment .getExternalStorageDirectory(), m_appNameStr)), "application/vnd.android.package-archive"); startActivity(intent); } }逐步讲解:
1、OnCreate函数:
先从主函数开始讲,OnCreate函数中只有两部分,一个是initVariable()初始化变量,这个不多说,难度不大;第二个是为版本检测按钮设置监听函数——btnClickListener,而在btnClickListener函数中可以明显的看到,其中也只有一个类checkNewestVersionAsyncTask,这里采用异步方式处理更新问题。下面看checkNewestVersionAsyncTask函数
2、checkNewestVersionAsyncTask函数
class checkNewestVersionAsyncTask extends AsyncTask<Void, Void, Boolean> { @Override protected Boolean doInBackground(Void... params) { // TODO Auto-generated method stub if(postCheckNewestVersionCommand2Server()) { int vercode = Common.getVerCode(getApplicationContext()); // 用到前面第一节写的方法 if (m_newVerCode > vercode) { return true; } else { return false; } } return false; } @Override protected void onPostExecute(Boolean result) { // TODO Auto-generated method stub if (result) {//如果有最新版本 doNewVersionUpdate(); // 更新新版本 }else { notNewVersionDlgShow(); // 提示当前为最新版本 } super.onPostExecute(result); } @Override protected void onPreExecute() { // TODO Auto-generated method stub super.onPreExecute(); } }
(1)首先看后台操作doInBackground
首先利用postCheckNewestVersionCommand2Server()函数将请求发送到服务器,该函数根据是否请求成功返回TRUE或FALSE,然后将从服务器上获取的版本代码与本地的版本代码进行比较,如果要更新返回TRUE,如果不要更新返回FASLE。
下面看看postCheckNewestVersionCommand2Server()的代码:
private Boolean postCheckNewestVersionCommand2Server() { StringBuilder builder = new StringBuilder(); JSONArray jsonArray = null; try { // 构造POST方法的{name:value} 参数对 List<NameValuePair> vps = new ArrayList<NameValuePair>(); // 将参数传入post方法中 vps.add(new BasicNameValuePair("action", "checkNewestVersion")); builder = Common.post_to_server(vps); jsonArray = new JSONArray(builder.toString()); if (jsonArray.length()>0) { if (jsonArray.getJSONObject(0).getInt("id") == 1) { m_newVerName = jsonArray.getJSONObject(0).getString("verName"); m_newVerCode = jsonArray.getJSONObject(0).getLong("verCode"); return true; } } return false; } catch (Exception e) { Log.e("msg",e.getMessage()); m_newVerName=""; m_newVerCode=-1; return false; } }
这里就是构建名值对,然后发向服务器,如果获取到了值就返回TURE,如果没获取到值,就返回FALSE。服务器返回的JSON值为:
[{"id":"1","verName":"1.0.1","verCode":"2"}]
对于服务器代码,由于是用PHP写的,这里就不再列出了,在源码里有。
(2)onPostExecute()
继续第一部分,在doInBackground操作完成后,如果需要更新doInBackground返回TRUE,否则返回FASLE,所以在onPostExecute中根据result不同调用不同的函数,利用doNewVersionUpdate(); 提示用户更新最新版本。利用notNewVersionDlgShow();
/提示用户当前即为最新版本,无需更新。
对于notNewVersionDlgShow()函数仅仅是创建了个对话框,没什么实体内容,就不再具体讲解。下面具体讲述doNewVersionUpdate()下载,更新与安装程序的过程。
3、doNewVersionUpdate()提示版本更新
具体代码如下:
private void doNewVersionUpdate() { int verCode = Common.getVerCode(getApplicationContext()); String verName = Common.getVerName(getApplicationContext()); String str= "当前版本:"+verName+" Code:"+verCode+" ,发现新版本:"+m_newVerName+ " Code:"+m_newVerCode+" ,是否更新?"; Dialog dialog = new AlertDialog.Builder(this).setTitle("软件更新").setMessage(str) // 设置内容 .setPositiveButton("更新",// 设置确定按钮 new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { m_progressDlg.setTitle("正在下载"); m_progressDlg.setMessage("请稍候..."); downFile(Common.UPDATESOFTADDRESS); //开始下载 } }) .setNegativeButton("暂不更新", new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int whichButton) { // 点击"取消"按钮之后退出程序 finish(); } }).create();// 创建 // 显示对话框 dialog.show(); }
这里创建一个具有确定按钮和取消按钮功能的对话框,如果用户点击取消按钮,会利用finish()结束掉程序运行;如果点击确定按钮,则利用 downFile(Common.UPDATESOFTADDRESS); 函数开始程序下载,其中downFile()函数传进去的参数是APP所在的服务器地址。注意这里的地址要具体到下载文件,比如这里是http://192.168.1.105/server/XXX.apk
4、downFile(final String url)下载并显示进度
具体代码如下:
private void downFile(final String url) { m_progressDlg.show(); new Thread() { public void run() { HttpClient client = new DefaultHttpClient(); HttpGet get = new HttpGet(url); HttpResponse response; try { response = client.execute(get); HttpEntity entity = response.getEntity(); long length = entity.getContentLength(); m_progressDlg.setMax((int)length);//设置进度条的最大值 InputStream is = entity.getContent(); FileOutputStream fileOutputStream = null; if (is != null) { File file = new File( Environment.getExternalStorageDirectory(), m_appNameStr); fileOutputStream = new FileOutputStream(file); byte[] buf = new byte[1024]; int ch = -1; int count = 0; while ((ch = is.read(buf)) != -1) { fileOutputStream.write(buf, 0, ch); count += ch; if (length > 0) { m_progressDlg.setProgress(count);//设置当前进度 } } } fileOutputStream.flush(); if (fileOutputStream != null) { fileOutputStream.close(); } down(); //告诉HANDER已经下载完成了,可以安装了 } catch (ClientProtocolException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } }.start(); }
通过利用httpClient的get方法,获取指定URL的内容,然后写到本地SD卡中,对于进度条,首先利用m_progressDlg.setMax((int)length);设置进度条的最大值,然后在读取返回结果并写到本地时,利用 m_progressDlg.setProgress(count);设置当前进度。在全部写完以后,调用down()函数,通知HANDER安装程序。
5、安装程序
安装程序主要利用下面两个函数:
/** * 告诉HANDER已经下载完成了,可以安装了 */ private void down() { m_mainHandler.post(new Runnable() { public void run() { m_progressDlg.cancel(); update(); } }); } /** * 安装程序 */ void update() { Intent intent = new Intent(Intent.ACTION_VIEW); intent.setDataAndType(Uri.fromFile(new File(Environment .getExternalStorageDirectory(), m_appNameStr)), "application/vnd.android.package-archive"); startActivity(intent); }
由于在android程序中必须依循单线程操作UI,所以在非主线程中不能操作UI,否则程序会崩掉,具体参见:《AsnyncTask与handler(一)——AsyncTask异步处理》与《AsnyncTask与handler(二)——handler消息机制》。所以这里作用handler的方式更新UI。
好了,到这就全部讲完了,下面给出客户端与服务器端源码,懂PHP的童鞋赚到了有木有……
源码地址:http://download.csdn.net/detail/harvic880925/7309013 不要分,仅供分享。
请大家尊重作者原创版权,转载请标明出处:http://blog.csdn.net/harvic880925/article/details/25191159 不胜感激。
相关文章推荐
- Android 提示版本更新的实现
- Android提示版本更新的实现
- Android实现版本更新提示
- Android提示版本更新
- [置顶] Android开发必有功能,更新版本提示,检测是否有新版本更新。下载完成后进行安装。
- Android 提示版本更新的实现
- Android提示版本更新
- Android提示版本更新
- Android提示用户更新版本
- Android提示版本更新
- Android提示版本更新的实现
- Android提示版本更新+notification显示下载进度
- Android提示版本更新操作流程
- Android提示版本更新
- Android 提示版本更新的实现
- Android提示版本更新的实现
- Android提示版本更新的实现
- android 更新到高德地图2.X.X版本,打包后地图提示invalid_user_scode
- Android提示版本更新
- Android版本更新,下载完成安装后提示,应用未安装