Android应用升级构想和要点总结

2023-11-02 20:30

本文主要是介绍Android应用升级构想和要点总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文链接:http://blog.csdn.net/richway2010/article/details/6408258

下面就如何对Android 手机上的应用进行升级相关知识的讲解:

一、原理及要点概述:

  1. 手机软件一般在运行时会把服务端的版本信息和当前手机中的版本进行比较。从而得知需不需要更新。Android上推介版本比较更新这种方式。
  2. 如果服务器端有新版本,我们需要先下载这个APK到我们的sdcard中,然后对其进行安装。
  3. 我们一定要保证每次安装的keystore密钥文件是相同的。这样Android手机才会提醒你替换新版本。
  4. Android区分软件的不同是通过包名,身份的认证是通过签名。只有相同签名的APK才可以安装。不然安装就会失败。

注:如果你每次发布APK文件时都新建一个key文件会导致无法正确安装,我们必须先卸载老的版本才能执行新版本安装。这样你就必须要通知每个客户去把软件卸载后重新下载安装。

二、接下来具体介绍一下升级步骤:

  1. 签名步骤流程讲一下,很多初学者不是很清楚或容易犯错:

签名的作用: 是为了保证每个应用程序开发商合法ID,防止部分开放商可能通过使用相同的Package Name来混淆替换已经安装的程序,我们需要对我们发布的APK文件进行唯一签名,保证我们每次发布的版本的一致性(如自动更新不会因为版本不一致而无法安装)。

首先eclipse如图所示打开:

clip_image002

然后设置APK名称,一般默认项目名称:

clip_image004

点击下一步,新建一个keys,输入密钥,:

clip_image006

注:这个密钥很重要,每次升级都需要使用到。忘记密钥也只能是重新生成。从而会引发后面升级的一系列问题。

下一步如图,设置别名,组织机构等等基本信息,根据具体情况:

clip_image008

下一步如图选择保存的APK目录:

clip_image010

至此apk和密钥文件生成完毕:

clip_image012

接下来,当程序需要改变,我们如何打包升级APK文件:

第一步和上面一样,从第二步开始讲解:

注:一定要使用现有的密钥,输入之前设置的密码,不然安装APK将不被覆盖。

clip_image014

下一步如图,输入密码:

clip_image016

然后下一步,Finish掉。

至此APK升级文件创建好。

三、服务器端和客户端设计

  1. 服务器端设计:

设计方法应该有很多,下面介绍我的一种方法:

  • a.首先在服务器项目下建立一个文件夹来存放APK安装文件:
  • b.其次在src下建立一个资源文件,apkVersion.properties,属性定义如下:
[plain] view plain copy print ?
  1. apkVersion=1 存版本号apkSize=550kb 大小apkPath=http://xx8080/srv/apk/Demo.apk 升级文件 
apkVersion=1 存版本号apkSize=550kb 大小apkPath=http://xx8080/srv/apk/Demo.apk 升级文件
  • c.定义一个servlet来获取资源中的信息:

定义类:UpdateApkServlet.java

[java] view plain copy print ?
  1. //获取资源文件信息 
  2. static
  3. Properties ppt = new Properties(); 
  4. try {      
  5.     ppt.load(UpdateApkServlet.class .getResourceAsStream("/apkVersion.properties"));      
  6.     apkVersion = ppt.getProperty("apkVersion");      
  7.     apkSize = ppt.getProperty("apkSize");      
  8.     apkPath = ppt.getProperty("apkPath");  
  9. }catch (Exception e) {      
  10.     e.printStackTrace(); 
  11. }  
//获取资源文件信息
static {
Properties ppt = new Properties();
try {     ppt.load(UpdateApkServlet.class .getResourceAsStream("/apkVersion.properties"));     apkVersion = ppt.getProperty("apkVersion");     apkSize = ppt.getProperty("apkSize");     apkPath = ppt.getProperty("apkPath"); 
}catch (Exception e) {     e.printStackTrace();
}
} 

获取资源,然后生成JSON字串返回客户端处理。 注:当客户端版本有更新,服务器端只要把APK文件拷贝到APK目录,然后更新apkVersion.properties文件中的信息就可以了,切记。

  • 客户端设计:
  • 1、 客户端首先获取服务器的版本信息(http方式获取)。
  • 2、 如何获取本地客户端的版本信息 如下参考代码:
[java] view plain copy print ?
  1. /**
  2. * 得到本地应用的版本信息 
  3. * @return
  4. */ 
  5. private int getAPKVersion(){ 
  6.     //APK版本判断 
  7.     int sdcardVersion = 0
  8.     String apkFilePath="sdcard/demo.apk"//安装包路径 
  9.     PackageManager pm = getPackageManager();         
  10.     PackageInfo info = pm.getPackageArchiveInfo(apkFilePath, PackageManager.GET_ACTIVITIES);          
  11. if(info != null){              
  12.     sdcardVersion=info.versionCode;       //得到版本信息              
  13.     Log.v(TAG, "Version="+sdcardVersion);         
  14. }          
  15. return sdcardVersion; 
/** * 得到本地应用的版本信息  * @return 
*/
private int getAPKVersion(){//APK版本判断int sdcardVersion = 0;String apkFilePath="sdcard/demo.apk";  //安装包路径PackageManager pm = getPackageManager();        PackageInfo info = pm.getPackageArchiveInfo(apkFilePath, PackageManager.GET_ACTIVITIES);         
if(info != null){             sdcardVersion=info.versionCode;       //得到版本信息             Log.v(TAG, "Version="+sdcardVersion);        
}         
return sdcardVersion;
}

  • 3、 版本比较,如果版本相同,则不执行更新,不同才进行更新操作。 这里插入客户端版本设置介绍: 客户端版本设置在AndroidManifest.xml文件中,里面有两个属性可进行版本信息设置, android:versionCode="1" 版本号 android:versionName="1.1" 版本名称 这个版本号需要和服务器端对应。
  • 4、 需要的权限设置
[plain] view plain copy print ?
  1. Sdcard访问权限: uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" 
  2. 访问网络权限: uses-permission android:name="android.permission.INTERNET"  
Sdcard访问权限: uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
访问网络权限: uses-permission android:name="android.permission.INTERNET" 
  • 5、 更新安装 当用户点击应用时执行检查更新。相关代码参考:

//弹出框提示

[java] view plain copy print ?
  1. public Handler handler = new Handler() { 
  2. public void handleMessage(Message msg) { 
  3. super.handleMessage(msg);        
  4. Dialog dialog = new AlertDialog.Builder(MainActivity.this).setTitle("系统更新").setMessage("发现新版本,请更新!")  
  5. // 设置内容.setPositiveButton("确定",// 设置确定按钮new DialogInterface.OnClickListener() { 
  6. @Override 
  7. public void onClick(DialogInterface dialog, int which) { 
  8. pBar = new ProgressDialog(MainActivity.this); 
  9. pBar.setTitle("正在下载");pBar.setMessage("请稍候..."); 
  10. pBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);downFile(apkPath);}}).setNegativeButton("取消", new DialogInterface.OnClickListener() { 
  11. public void onClick(DialogInterface dialog, int whichButton) { 
  12. // 点击"取消"按钮操作}}).create();// 创建 
  13. // 显示对话框 
  14. dialog.show(); 
  15.         } 
  16. }; 
public Handler handler = new Handler() {
public void handleMessage(Message msg) {
super.handleMessage(msg);       
Dialog dialog = new AlertDialog.Builder(MainActivity.this).setTitle("系统更新").setMessage("发现新版本,请更新!") 
// 设置内容.setPositiveButton("确定",// 设置确定按钮new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
pBar = new ProgressDialog(MainActivity.this);
pBar.setTitle("正在下载");pBar.setMessage("请稍候...");
pBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);downFile(apkPath);}}).setNegativeButton("取消", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) {
// 点击"取消"按钮操作}}).create();// 创建
// 显示对话框
dialog.show();}
};

//下载

[java] view plain copy print ?
  1. /**
  2. * DOWNLOAD APK FILE BY URL
  3. * @param url
  4. */ 
  5. public void downFile(final String url) { 
  6. pBar.show(); 
  7. new Thread() { 
  8. public void run() { 
  9. HttpClient client = new DefaultHttpClient(); 
  10. // params[0]代表连接的 
  11. urlHttpGet get = new HttpGet(url); 
  12. HttpResponse response; 
  13. try
  14. response = client.execute(get); 
  15. HttpEntity entity = response.getEntity(); 
  16. long length = entity.getContentLength(); 
  17. InputStream is = entity.getContent(); 
  18. FileOutputStream fileOutputStream = null
  19. if (is != null) { 
  20. File file = new File(Environment.getExternalStorageDirectory(),"demo.apk"); 
  21. fileOutputStream = new FileOutputStream(file); 
  22. byte[] buf = new byte[1024]; 
  23. int ch = -1
  24. int count = 0
  25. while ((ch = is.read(buf)) != -1) { 
  26. // baos.write(buf, 0, ch); 
  27. fileOutputStream.write(buf, 0, ch); 
  28. count += ch;if (length > 0) {} 
  29. fileOutputStream.flush(); 
  30. if (fileOutputStream != null) { 
  31. fileOutputStream.close(); 
  32. down(); 
  33. } catch (ClientProtocolException e) { 
  34. e.printStackTrace(); 
  35. } catch (IOException e) { 
  36. e.printStackTrace(); 
  37. }.start(); 
  38. public void down() { 
  39. handler.post(new Runnable()  
  40. public void run() { 
  41. pBar.cancel(); 
  42. update(); 
  43. }}); 
/** * DOWNLOAD APK FILE BY URL * @param url 
*/
public void downFile(final String url) {
pBar.show();
new Thread() {
public void run() {
HttpClient client = new DefaultHttpClient();
// params[0]代表连接的
urlHttpGet get = new HttpGet(url);
HttpResponse response;
try {
response = client.execute(get);
HttpEntity entity = response.getEntity();
long length = entity.getContentLength();
InputStream is = entity.getContent();
FileOutputStream fileOutputStream = null;
if (is != null) {
File file = new File(Environment.getExternalStorageDirectory(),"demo.apk");
fileOutputStream = new FileOutputStream(file);
byte[] buf = new byte[1024];
int ch = -1;
int count = 0;
while ((ch = is.read(buf)) != -1) {
// baos.write(buf, 0, ch);
fileOutputStream.write(buf, 0, ch);
count += ch;if (length > 0) {}
}
}
fileOutputStream.flush();
if (fileOutputStream != null) {
fileOutputStream.close();
}
down();
} catch (ClientProtocolException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}.start();
}
public void down() {
handler.post(new Runnable() 
{
public void run() {
pBar.cancel();
update();
}});
}

//更新升级

[java] view plain copy print ?
  1. public void update() { 
  2. Intent intent = new Intent(Intent.ACTION_VIEW); 
  3. intent.setDataAndType(Uri.fromFile(new File("/sdcard/demo.apk")),"application/vnd.android.package-archive"); 
  4. startActivity(intent); 
public void update() {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.fromFile(new File("/sdcard/demo.apk")),"application/vnd.android.package-archive");
startActivity(intent);
}


结束,供参考。

2011-5

 

这篇关于Android应用升级构想和要点总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/333118

相关文章

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动

Python中logging模块用法示例总结

《Python中logging模块用法示例总结》在Python中logging模块是一个强大的日志记录工具,它允许用户将程序运行期间产生的日志信息输出到控制台或者写入到文件中,:本文主要介绍Pyt... 目录前言一. 基本使用1. 五种日志等级2.  设置报告等级3. 自定义格式4. C语言风格的格式化方法

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

Debian 13升级后网络转发等功能异常怎么办? 并非错误而是管理机制变更

《Debian13升级后网络转发等功能异常怎么办?并非错误而是管理机制变更》很多朋友反馈,更新到Debian13后网络转发等功能异常,这并非BUG而是Debian13Trixie调整... 日前 Debian 13 Trixie 发布后已经有众多网友升级到新版本,只不过升级后发现某些功能存在异常,例如网络转

Spring 依赖注入与循环依赖总结

《Spring依赖注入与循环依赖总结》这篇文章给大家介绍Spring依赖注入与循环依赖总结篇,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Spring 三级缓存解决循环依赖1. 创建UserService原始对象2. 将原始对象包装成工

Ubuntu如何升级Python版本

《Ubuntu如何升级Python版本》Ubuntu22.04Docker中,安装Python3.11后,使用update-alternatives设置为默认版本,最后用python3-V验证... 目China编程录问题描述前提环境解决方法总结问题描述Ubuntu22.04系统自带python3.10,想升级

解决升级JDK报错:module java.base does not“opens java.lang.reflect“to unnamed module问题

《解决升级JDK报错:modulejava.basedoesnot“opensjava.lang.reflect“tounnamedmodule问题》SpringBoot启动错误源于Jav... 目录问题描述原因分析解决方案总结问题描述启动sprintboot时报以下错误原因分析编程异js常是由Ja

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

PostgreSQL简介及实战应用

《PostgreSQL简介及实战应用》PostgreSQL是一种功能强大的开源关系型数据库管理系统,以其稳定性、高性能、扩展性和复杂查询能力在众多项目中得到广泛应用,本文将从基础概念讲起,逐步深入到高... 目录前言1. PostgreSQL基础1.1 PostgreSQL简介1.2 基础语法1.3 数据库

MySQL中查询和展示LONGBLOB类型数据的技巧总结

《MySQL中查询和展示LONGBLOB类型数据的技巧总结》在MySQL中LONGBLOB是一种二进制大对象(BLOB)数据类型,用于存储大量的二进制数据,:本文主要介绍MySQL中查询和展示LO... 目录前言1. 查询 LONGBLOB 数据的大小2. 查询并展示 LONGBLOB 数据2.1 转换为十