前言:在软件开发的尾声应该都会遇到这个问题,还好网上资料很多,所以基本不费什么力气就搞定了,现记录于下。这里用的PHP服务器。
效果图:(PHP服务器)
初始界面 检测后,如果已是最新版
如果不是最新版,提示更新 正在下载 安装新程序
一、准备知识
在AndroidManifest.xml里定义了每个Android apk的版本标识:
[html]
view plain
copy
- <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,参考以下代码:
[java]
view plain
copy
- /**
- * 获取软件版本号
- * @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而已。代码如下:
[html]
view plain
copy
- <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类中实现,代码如下:
[java]
view plain
copy
- 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>标签上面,加入:
[html]
view plain
copy
- <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、主页代码:
先贴出全部代码,然后逐步讲解,全部代码如下:
[java]
view plain
copy
- 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函数
[java]
view plain
copy
- 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()的代码:
[java]
view plain
copy
- 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值为:
[html]
view plain
copy
- [{"id":"1","verName":"1.0.1","verCode":"2"}]
对于服务器代码,由于是用PHP写的,这里就不再列出了,在源码里有。
(2)onPostExecute()继续第一部分,在doInBackground操作完成后,如果需要更新doInBackground返回TRUE,否则返回FASLE,所以在onPostExecute中根据result不同调用不同的函数,利用doNewVersionUpdate(); 提示用户更新最新版本。利用notNewVersionDlgShow(); /提示用户当前即为最新版本,无需更新。
对于notNewVersionDlgShow()函数仅仅是创建了个对话框,没什么实体内容,就不再具体讲解。下面具体讲述doNewVersionUpdate()下载,更新与安装程序的过程。
3、doNewVersionUpdate()提示版本更新具体代码如下:
[java]
view plain
copy
- 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)下载并显示进度
具体代码如下:
[java]
view plain
copy
- 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、安装程序
安装程序主要利用下面两个函数:
[java]
view plain
copy
- /**
- * 告诉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的童鞋赚到了有木有……