我们看到很多Android应用都具有自动更新功能,用户一键就可以完成软件的升级更新。得益于Android系统的软件包管理和安装机制,这一功能实现起来相当简单,下面我们就来实践一下。首先给出界面效果:
1. 准备知识
在AndroidManifest.xml里定义了每个Android apk的版本标识:
1.<manifest xmlns:android="http://schemas.android.com/apk/res/android"
2. package="com.myapp"
3. android:versionCode="1"
4. android:versionName="1.0.0">
5.<application></application>
6.</manifest>
1.<manifest xmlns:android="http://schemas.android.com/apk/res/android"
2. package="com.myapp" 3. android:versionCode="1" 4. android:versionName="1.0.0"> 5.<application></application> 6.</manifest> 其中,android:versionCode和android:versionName两个字段分别表示版本代码,版本名称。versionCode是整型数字,versionName是字符串。由于version是给用户看的,不太容易比较大小,升级检查时,可以以检查versionCode为主,方便比较出版本的前后大小。
那么,在应用中如何读取AndroidManifest.xml中的versionCode和versionName呢?可以使用PackageManager的API,参考以下代码:
1.public static int getVerCode(Context context) {
2. int verCode = -1;
3. try {
4. verCode = context.getPackageManager().getPackageInfo(
5. "com.myapp", 0).versionCode;
6. } catch (NameNotFoundException e) {
7. Log.e(TAG, e.getMessage());
8. }
9. return verCode;
10. }
11.
12. public static String getVerName(Context context) {
13. String verName = "";
14. try {
15. verName = context.getPackageManager().getPackageInfo(
16. "com.myapp", 0).versionName;
17. } catch (NameNotFoundException e) {
18. Log.e(TAG, e.getMessage());
19. }
20. return verName;
21.}
1.public static int getVerCode(Context context) {
2. int verCode = -1;
3. try {
4. verCode = context.getPackageManager().getPackageInfo(
5. "com.myapp", 0).versionCode;
6. } catch (NameNotFoundException e) {
7. Log.e(TAG, e.getMessage());
8. }
9. return verCode;
10. }
11.
12. public static String getVerName(Context context) {
13. String verName = "";
14. try {
15. verName = context.getPackageManager().getPackageInfo(
16. "com.myapp", 0).versionName;
17. } catch (NameNotFoundException e) {
18. Log.e(TAG, e.getMessage());
19. }
20. return verName;
21.}
或者在AndroidManifest中将android:versionName="1.2.0"写成android:versionName="@string/app_versionName",然后在values/strings.xml中添加对应字符串,这样实现之后,就可以使用如下代码获得版本名称:
1.public static String getVerName(Context context) {
2. String verName = context.getResources()
3. .getText(R.string.app_versionName).toString();
4. return verName;
5.}
1.public static String getVerName(Context context) {
2. String verName = context.getResources()
3. .getText(R.string.app_versionName).toString();
4. return verName; 5.}
同理,apk的应用名称可以这样获得:
1.public static String getAppName(Context context) {
2. String verName = context.getResources()
3. .getText(R.string.app_name).toString();
4. return verName;
5.}
1.public static String getAppName(Context context) {
2. String verName = context.getResources()
3. .getText(R.string.app_name).toString();
4. return verName; 5.}
2. 流程框架
3. 版本检查
在服务端放置最新版本的apk文件,如:http://localhost/myapp/myapp.apk
同时,在服务端放置对应此apk的版本信息调用接口或者文件,如:http://localhost/myapp/ver.json
ver.json中的内容为:
1.[{"appname":"jtapp12","apkname":"jtapp-12-updateapksamples.apk","verName":1.0.1,"verCode":2}]
1.[{"appname":"jtapp12","apkname":"jtapp-12-updateapksamples.apk","verName":1.0.1,"verCode":2}]
然后,在手机客户端上进行版本读取和检查:
1.private boolean getServerVer () {
2. try {
3. String verjson = NetworkTool.getContent(Config.UPDATE_SERVER
4. + Config.UPDATE_VERJSON);
5. JSONArray array = new JSONArray(verjson);
6. if (array.length() > 0) {
7. JSONObject obj = array.getJSONObject(0);
8. try {
9. newVerCode = Integer.parseInt(obj.getString("verCode"));
10. newVerName = obj.getString("verName");
11. } catch (Exception e) {
12. newVerCode = -1;
13. newVerName = "";
14. return false;
15. }
16. }
17. } catch (Exception e) {
18. Log.e(TAG, e.getMessage());
19. return false;
20. }
21. return true;
22. }
1.private boolean getServerVer () {
2. try {
3. String verjson = NetworkTool.getContent(Config.UPDATE_SERVER
4. + Config.UPDATE_VERJSON);
5. JSONArray array = new JSONArray(verjson);
6. if (array.length() > 0) {
7. JSONObject obj = array.getJSONObject(0);
8. try {
9. newVerCode = Integer.parseInt(obj.getString("verCode"));
10. newVerName = obj.getString("verName");
11. } catch (Exception e) {
12. newVerCode = -1;
13. newVerName = "";
14. return false;
15. }
16. }
17. } catch (Exception e) {
18. Log.e(TAG, e.getMessage());
19. return false;
20. }
21. return true;
22. }
比较服务器和客户端的版本,并进行更新操作。
1.if (getServerVerCode()) {
2. int vercode = Config.getVerCode(this); // 用到前面第一节写的方法
3. if (newVerCode > vercode) {
4. doNewVersionUpdate(); // 更新新版本
5. } else {
6. notNewVersionShow(); // 提示当前为最新版本
7. }
8. }
1.if (getServerVerCode()) {
2. int vercode = Config.getVerCode(this); // 用到前面第一节写的方法
3. if (newVerCode > vercode) {
4. doNewVersionUpdate(); // 更新新版本
5. } else {
6. notNewVersionShow(); // 提示当前为最新版本
7. }
8. }
详细方法:
1.private void notNewVersionShow() {
2. int verCode = Config.getVerCode(this);
3. String verName = Config.getVerName(this);
4. StringBuffer sb = new StringBuffer();
5. sb.append("当前版本:");
6. sb.append(verName);
7. sb.append(" Code:");
8. sb.append(verCode);
9. sb.append(",/n已是最新版,无需更新!");
10. Dialog dialog = new AlertDialog.Builder(Update.this).setTitle("软件更新")
11. .setMessage(sb.toString())// 设置内容
12. .setPositiveButton("确定",// 设置确定按钮
13. new DialogInterface.OnClickListener() {
14. @Override
15. public void onClick(DialogInterface dialog,
16. int which) {
17. finish();
18. }
19. }).create();// 创建
20. // 显示对话框
21. dialog.show();
22.}
23.private void doNewVersionUpdate() {
24. int verCode = Config.getVerCode(this);
25. String verName = Config.getVerName(this);
26. StringBuffer sb = new StringBuffer();
27. sb.append("当前版本:");
28. sb.append(verName);
29. sb.append(" Code:");
30. sb.append(verCode);
31. sb.append(", 发现新版本:");
32. sb.append(newVerName);
33. sb.append(" Code:");
34. sb.append(newVerCode);
35. sb.append(", 是否更新?");
36. Dialog dialog = new AlertDialog.Builder(Update.this)
37. .setTitle("软件更新")
38. .setMessage(sb.toString())
39. // 设置内容
40. .setPositiveButton("更新",// 设置确定按钮
41. new DialogInterface.OnClickListener() {
42. @Override
43. public void onClick(DialogInterface dialog,
44. int which) {
45. pBar = new ProgressDialog(Update.this);
46. pBar.setTitle("正在下载");
47. pBar.setMessage("请稍候...");
48. pBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);
49. downFile(Config.UPDATE_SERVER + Config.UPDATE_APKNAME);
50. }
51. })
52. .setNegativeButton("暂不更新",
53. new DialogInterface.OnClickListener() {
54. public void onClick(DialogInterface dialog,
55. int whichButton) {
56. // 点击"取消"按钮之后退出程序
57. finish();
58. }
59. }).create();// 创建
60. // 显示对话框
61. dialog.show();
62.}
1.private void notNewVersionShow() {
2. int verCode = Config.getVerCode(this);
3. String verName = Config.getVerName(this);
4. StringBuffer sb = new StringBuffer();
5. sb.append("当前版本:");
6. sb.append(verName);
7. sb.append(" Code:");
8. sb.append(verCode);
9. sb.append(",/n已是最新版,无需更新!");
10. Dialog dialog = new AlertDialog.Builder(Update.this).setTitle("软件更新")
11. .setMessage(sb.toString())// 设置内容
12. .setPositiveButton("确定",// 设置确定按钮
13. new DialogInterface.OnClickListener() {
14. @Override
15. public void onClick(DialogInterface dialog,
16. int which) {
17. finish();
18. }
19. }).create();// 创建
20. // 显示对话框
21. dialog.show();
22.}
23.private void doNewVersionUpdate() {
24. int verCode = Config.getVerCode(this);
25. String verName = Config.getVerName(this);
26. StringBuffer sb = new StringBuffer();
27. sb.append("当前版本:");
28. sb.append(verName);
29. sb.append(" Code:");
30. sb.append(verCode);
31. sb.append(", 发现新版本:");
32. sb.append(newVerName);
33. sb.append(" Code:");
34. sb.append(newVerCode);
35. sb.append(", 是否更新?");
36. Dialog dialog = new AlertDialog.Builder(Update.this)
37. .setTitle("软件更新")
38. .setMessage(sb.toString())
39. // 设置内容
40. .setPositiveButton("更新",// 设置确定按钮
41. new DialogInterface.OnClickListener() {
42. @Override
43. public void onClick(DialogInterface dialog,
44. int which) {
45. pBar = new ProgressDialog(Update.this);
46. pBar.setTitle("正在下载");
47. pBar.setMessage("请稍候...");
48. pBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);
49. downFile(Config.UPDATE_SERVER + Config.UPDATE_APKNAME);
50. }
51. })
52. .setNegativeButton("暂不更新",
53. new DialogInterface.OnClickListener() {
54. public void onClick(DialogInterface dialog,
55. int whichButton) {
56. // 点击"取消"按钮之后退出程序
57. finish();
58. }
59. }).create();// 创建
60. // 显示对话框
61. dialog.show();
62.}
注,本部分参考了前人的相关实现,见 http://www.linuxidc.com/Linux/2011-08/40309.htm
1.void downFile(final String url) {
2. pBar.show();
3. new Thread() {
4. public void run() {
5. HttpClient client = new DefaultHttpClient();
6. HttpGet get = new HttpGet(url);
7. HttpResponse response;
8. try {
9. response = client.execute(get);
10. HttpEntity entity = response.getEntity();
11. long length = entity.getContentLength();
12. InputStream is = entity.getContent();
13. FileOutputStream fileOutputStream = null;
14. if (is != null) {
15. File file = new File(
16. Environment.getExternalStorageDirectory(),
17. Config.UPDATE_SAVENAME);
18. fileOutputStream = new FileOutputStream(file);
19. byte[] buf = new byte[1024];
20. int ch = -1;
21. int count = 0;
22. while ((ch = is.read(buf)) != -1) {
23. fileOutputStream.write(buf, 0, ch);
24. count += ch;
25. if (length > 0) {
26. }
27. }
28. }
29. fileOutputStream.flush();
30. if (fileOutputStream != null) {
31. fileOutputStream.close();
32. }
33. down();
34. } catch (ClientProtocolException e) {
35. e.printStackTrace();
36. } catch (IOException e) {
37. e.printStackTrace();
38. }
39. }
40. }.start();
41.}
1.void downFile(final String url) {
2. pBar.show();
3. new Thread() {
4. public void run() {
5. HttpClient client = new DefaultHttpClient();
6. HttpGet get = new HttpGet(url);
7. HttpResponse response;
8. try {
9. response = client.execute(get);
10. HttpEntity entity = response.getEntity();
11. long length = entity.getContentLength();
12. InputStream is = entity.getContent();
13. FileOutputStream fileOutputStream = null;
14. if (is != null) {
15. File file = new File(
16. Environment.getExternalStorageDirectory(),
17. Config.UPDATE_SAVENAME);
18. fileOutputStream = new FileOutputStream(file);
19. byte[] buf = new byte[1024];
20. int ch = -1;
21. int count = 0;
22. while ((ch = is.read(buf)) != -1) {
23. fileOutputStream.write(buf, 0, ch);
24. count += ch;
25. if (length > 0) {
26. }
27. }
28. }
29. fileOutputStream.flush();
30. if (fileOutputStream != null) {
31. fileOutputStream.close();
32. }
33. down();
34. } catch (ClientProtocolException e) {
35. e.printStackTrace();
36. } catch (IOException e) {
37. e.printStackTrace();
38. }
39. }
40. }.start();
41.}
下载完成,通过handler通知主ui线程将下载对话框取消。
1.void down() {
2. handler.post(new Runnable() {
3. public void run() {
4. pBar.cancel();
5. update();
6. }
7. });
8.}
1.void down() {
2. handler.post(new Runnable() {
3. public void run() {
4. pBar.cancel();
5. update();
6. }
7. });
8.}
5. 安装应用
1.void update() {
2. Intent intent = new Intent(Intent.ACTION_VIEW);
3. intent.setDataAndType(Uri.fromFile(new File(Environment
4. .getExternalStorageDirectory(), Config.UPDATE_SAVENAME)),
5. "application/vnd.Android.package-archive");
6. startActivity(intent);
7.}
1.void update() {
2. Intent intent = new Intent(Intent.ACTION_VIEW);
3. intent.setDataAndType(Uri.fromFile(new File(Environment
4. .getExternalStorageDirectory(), Config.UPDATE_SAVENAME)),
5. "application/vnd.android.package-archive");
6. startActivity(intent);
7.}
如果你将apk应用发布到market上,那么,你会发现market内建了类似的模块,可以自动更新或者提醒你是否更新应用。那么,对于你自己的应用需要自动更新的话,自己内建一个是不是更加方便了呢?本文提到的代码大多是在UpdateActivity.java中实现,为了能够使更新过程更加友好,可以在最初launcher的Activity中建立一个线程,用来检查服务端是否有更新。有更新的时候就启动UpdateActivity,这样的使用体验更加平滑。
关于Tomcat这部分我自己整理如下:
1)在Apache 官网http://tomcat.apache.org/下载安装就可以了,我自己装的是Tomcat 7.0版本。
2)在Tomcat安装默认路径下找到webapps文件夹并在该文件夹下新建一个myapp文件夹(你可以命名成你自己喜欢的文件夹名)
3)把上面提到的ver.json 文件保存放到myapp文件夹下
4)把你要升级安装的apk文件也放到myapp文件夹下就可以了
注意事项:访问android模拟器的端口是http://10.0.2.2:8080
本篇文章来源于 Linux公社网站(www.linuxidc.com) 原文链接:http://www.linuxidc.com/Linux/2011-08/40308p3.htm
另附一篇文章:
/** * 安装新版本 * @param context 当前实例上下文 */ public void InstallApk(Context context,String fileName){ File f = new File(BasePath.DOWNLOAD_DIR,fileName); Intent intent = new Intent(); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.setAction(android.content.Intent.ACTION_VIEW); // 调用getMIMEType()来取得MimeType String type = getMIMEType(f); //设定intent的file与MimeType intent.setDataAndType(Uri.fromFile(f),type); context.startActivity(intent); } /** * 判断文件MimeType的method * @param f 要判断类型的文件 * @return 文件的MiMe类型 */ public String getMIMEType(File f) { String type=""; String fName=f.getName(); // 取得扩展名 String end=fName.substring(fName.lastIndexOf(".")+1,fName.length()).toLowerCase(); // 按扩展名的类型决定MimeType if(end.equals("m4a")||end.equals("mp3")||end.equals("mid")||end.equals("xmf")||end.equals("ogg")||end.equals("wav")){ type = "audio"; }else if(end.equals("3gp")||end.equals("mp4")){ type = "video"; }else if(end.equals("jpg")||end.equals("gif")||end.equals("png")||end.equals("jpeg")||end.equals("bmp")){ type = "image"; }else if(end.equals("apk")) { // android.permission.INSTALL_PACKAGES type = "application/vnd.android.package-archive"; } else{ type="*"; } //如果无法直接打开,就跳出软件清单给使用者选择 if(end.equals("apk")) { } else { type += "/*"; } return type; }