Android 輕鬆實現後臺搭建+APP版本更新

(本文講解了在Android中實現APP版本更新,文末附有源碼。)

看完本文,您可以學到:


1.版本更新的方法

2.與後臺的交互

3.Android中Handler的使用

4.Android中ProgressDialog的使用

話不多說,先來看看效果圖:





一、大致思路闡述


首先,我們要有一個可以被手機訪問的後臺。
這裏有兩種方法,在調試的時候我們可以利用手機和筆記本連到同一個局域網的方式,在電腦上開啓個類似PHP或者JAVAEE一樣樣的後臺服務。
但是對於沒有相關後臺開發經驗的朋友,這裏有一種更好的方式:利用Github等免費空間來實現。詳細請戳我的另一篇博文利用Github建立你的個人網站 。
OK,有了存放資源的後臺,我們要放點什麼東西呢?很簡單,一個包含最新版本信息的update.txt文件和一個.apk文件足矣!

txt文件裏寫啥?看下我的例子:
XXX&1.3&這裏寫點描述&http://192.168.1.100:8080/PersonalHomePage/new.apk
解釋一下: &是分隔符,用於手機端獲取到信息後的分割。1.3代表着最新版本號,之後的是新版本的描述,最後的是新版本APK的下載地址(這裏我用了局域網)。一開始的是啥呢?我當時在試驗的時候,在開頭並沒有加額外信息,即以1.3開頭,實驗之後,發現手機端獲取到TXT文本信息後不能正確解析,原因我覺得是因爲TXT文件的開頭包含有一些自帶的字符,手機解析時會有問題。(感興趣的朋友可以去深究,還望不吝賜教!)

OK,有了新版本的信息,我們要怎麼做?
我們要獲取到最新的版本號,然後與當前APP的版本號進行對比。如果低於最新版本,就到下載地址中去下載。

二、詳細代碼解釋


首先,新建一個UpdateInfo類,用來與update.txt的內容對應,這個很簡單:

package com.example.appupdatedemo;  public class UpdateInfo {         private String version;         private String description;         private String url;                  public String getVersion()         {                 return version;         }         public void setVersion(String version)         {                 this.version = version;         }         public String getDescription()         {                 return description;         }         public void setDescription(String description)         {                 this.description = description;         }         public String getUrl()         {                 return url;         }         public void setUrl(String url)         {                 this.url = url;         }          } 


然後,寫一個類去獲取更新的信息,即我們的update.txt文件:

UpdateInfoService:

package com.example.appupdatedemo;  import java.io.BufferedReader; import java.io.InputStreamReader; import java.net.HttpURLConnection; import java.net.URL;  import android.content.Context;  public class UpdateInfoService { 	public UpdateInfoService(Context context) { 	}  	public UpdateInfo getUpDateInfo() throws Exception { 		String path = GetServerUrl.getUrl() + "/update.txt"; 		StringBuffer sb = new StringBuffer(); 		String line = null; 		BufferedReader reader = null; 		try { 			// 創建一個url對象 			URL url = new URL(path); 			// 通過url對象,創建一個HttpURLConnection對象(連接) 			HttpURLConnection urlConnection = (HttpURLConnection) url 					.openConnection(); 			// 通過HttpURLConnection對象,得到InputStream 			reader = new BufferedReader(new InputStreamReader( 					urlConnection.getInputStream())); 			// 使用io流讀取文件 			while ((line = reader.readLine()) != null) { 				sb.append(line); 			} 		} catch (Exception e) { 			e.printStackTrace(); 		} finally { 			try { 				if (reader != null) { 					reader.close(); 				} 			} catch (Exception e) { 				e.printStackTrace(); 			} 		} 		String info = sb.toString(); 		UpdateInfo updateInfo = new UpdateInfo(); 		updateInfo.setVersion(info.split("&")[1]); 		updateInfo.setDescription(info.split("&")[2]); 		updateInfo.setUrl(info.split("&")[3]); 		return updateInfo; 	}  } 

這裏獲取文件的方法是先創建一個HttpURLConnection,再獲取輸入流。細心的朋友可能注意到其中有個類,叫GetServerUrl,這個類是用來存放後臺地址信息的:

package com.example.appupdatedemo;  /**  * 獲取服務器IP地址  */  public class GetServerUrl{ 	static String url="http://192.168.1.100:8080/PersonalHomePage";   //沒錯,我這裏用的是本地的JAVAEE工程,各位根據實際情況修改。 			 	public static String getUrl() { 		return url; 	} } 

OK,到了這一步,準備工作都做完了,接下來只剩一個類了!即我們的MainActicity,一共一百多行,我們分幾部分來講。

第一部分代碼,做的工作是獲取版本更新信息。

public class MainActivity extends Activity {  	// 更新版本要用到的一些信息 	private UpdateInfo info; 	private ProgressDialog pBar;  	@Override 	protected void onCreate(Bundle savedInstanceState) { 		super.onCreate(savedInstanceState); 		setContentView(R.layout.activity_main);  		Toast.makeText(MainActivity.this, "正在檢查版本更新..", Toast.LENGTH_SHORT).show(); 		// 自動檢查有沒有新版本 如果有新版本就提示更新 		new Thread() { 			public void run() { 				try { 					UpdateInfoService updateInfoService = new UpdateInfoService( 							MainActivity.this); 					info = updateInfoService.getUpDateInfo(); 					handler1.sendEmptyMessage(0); 				} catch (Exception e) { 					e.printStackTrace(); 				} 			}; 		}.start(); 	}  	@SuppressLint("HandlerLeak") 	private Handler handler1 = new Handler() { 		public void handleMessage(Message msg) { 			// 如果有更新就提示 			if (isNeedUpdate()) {   //在下面的代碼段 				showUpdateDialog();  //下面的代碼段 			} 		}; 	};

這裏我們用到了new Thread+ Handler的方式去進行異步加載版本信息,主要是因爲在安卓中要把耗時任務放在非主線程中執行,否則會造成阻塞,拋出無響應異常。還有另外的實現方式是安卓封裝的AsyncTask,具體可以參考這篇博文:Android AsyncTask詳解


第二部分,判斷是否是最新版本,如果不是,跳出對話框選擇是否更新:

	private void showUpdateDialog() { 		AlertDialog.Builder builder = new AlertDialog.Builder(this); 		builder.setIcon(android.R.drawable.ic_dialog_info); 		builder.setTitle("請升級APP至版本" + info.getVersion()); 		builder.setMessage(info.getDescription()); 		builder.setCancelable(false);  		builder.setPositiveButton("確定", new DialogInterface.OnClickListener() {  			@Override 			public void onClick(DialogInterface dialog, int which) { 				if (Environment.getExternalStorageState().equals( 						Environment.MEDIA_MOUNTED)) { 					downFile(info.getUrl());     //在下面的代碼段 				} else { 					Toast.makeText(MainActivity.this, "SD卡不可用,請插入SD卡", 							Toast.LENGTH_SHORT).show(); 				} 			} 		}); 		builder.setNegativeButton("取消", new DialogInterface.OnClickListener() {  			@Override 			public void onClick(DialogInterface dialog, int which) { 			}  		}); 		builder.create().show(); 	}  	private boolean isNeedUpdate() { 		 		String v = info.getVersion(); // 最新版本的版本號 		Log.i("update",v); 		Toast.makeText(MainActivity.this, v, Toast.LENGTH_SHORT).show(); 		if (v.equals(getVersion())) { 			return false; 		} else { 			return true; 		} 	}  	// 獲取當前版本的版本號 	private String getVersion() { 		try { 			PackageManager packageManager = getPackageManager(); 			PackageInfo packageInfo = packageManager.getPackageInfo( 					getPackageName(), 0); 			return packageInfo.versionName; 		} catch (NameNotFoundException e) { 			e.printStackTrace(); 			return "版本號未知"; 		} 	}
這段裏面要注意的是怎麼獲取當前版本,方法是使用PackageManager提供的getPackageInfo方法,返回的是manifest文件中的版本號。其他的代碼挺簡單,註釋也挺全的。如果有問題,歡迎留言。

接下來是最後一部分,下載文件。

	void downFile(final String url) {  		pBar = new ProgressDialog(MainActivity.this);    //進度條,在下載的時候實時更新進度,提高用戶友好度 		pBar.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL); 		pBar.setTitle("正在下載"); 		pBar.setMessage("請稍候..."); 		pBar.setProgress(0); 		pBar.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(); 					int length = (int) entity.getContentLength();   //獲取文件大小                                         pBar.setMax(length);                            //設置進度條的總長度 					InputStream is = entity.getContent(); 					FileOutputStream fileOutputStream = null; 					if (is != null) { 						File file = new File( 								Environment.getExternalStorageDirectory(), 								"Test.apk"); 						fileOutputStream = new FileOutputStream(file); 						byte[] buf = new byte[10];   //這個是緩衝區,即一次讀取10個比特,我弄的小了點,因爲在本地,所以數值太大一 下就下載完了,看不出progressbar的效果。 						int ch = -1; 						int process = 0; 						while ((ch = is.read(buf)) != -1) {        							fileOutputStream.write(buf, 0, ch); 							process += ch; 							pBar.setProgress(process);       //這裏就是關鍵的實時更新進度了! 						}  					} 					fileOutputStream.flush(); 					if (fileOutputStream != null) { 						fileOutputStream.close(); 					} 					down(); 				} catch (ClientProtocolException e) { 					e.printStackTrace(); 				} catch (IOException e) { 					e.printStackTrace(); 				} 			}  		}.start(); 	}  	void down() { 		handler1.post(new Runnable() { 			public void run() { 				pBar.cancel(); 				update(); 			} 		}); 	} //安裝文件,一般固定寫法 	void update() {                     		Intent intent = new Intent(Intent.ACTION_VIEW); 		intent.setDataAndType(Uri.fromFile(new File(Environment 				.getExternalStorageDirectory(), "Test.apk")), 				"application/vnd.android.package-archive"); 		startActivity(intent); 	}
這一段主要是利用progressdialog在下載的時候實時更新進度,主要利用的是一個字節數組的緩衝區。即每次獲取到的內容填滿緩衝區後就寫入到本地本件中。這裏我把緩衝區的大小設置爲10個字節(1024會比較好),理由是因爲在同一個局域網中速度特別快,刷一下就下載完了,看不出進度條效果,緩衝區調小點就OK了。

========================================

寫在後面:

源代碼已上傳到我的Github,或者到CSDN下載區下載。

任何問題,歡迎留言交流!



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章