溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android 輕松實現(xiàn)后臺搭建+APP版本更新

發(fā)布時間:2020-07-30 01:22:52 來源:網(wǎng)絡(luò) 閱讀:1404 作者:Hyman1994 欄目:移動開發(fā)

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

看完本文,您可以學(xué)到:


1.版本更新的方法

2.與后臺的交互

3.Android中Handler的使用

4.Android中ProgressDialog的使用

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

Android 輕松實現(xiàn)后臺搭建+APP版本更新

Android 輕松實現(xiàn)后臺搭建+APP版本更新


Android 輕松實現(xiàn)后臺搭建+APP版本更新


一、大致思路闡述


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

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

OK,有了新版本的信息,我們要怎么做?
我們要獲取到最新的版本號,然后與當(dāng)前APP的版本號進行對比。如果低于最新版本,就到下載地址中去下載。

二、詳細(xì)代碼解釋


首先,新建一個UpdateInfo類,用來與update.txt的內(nèi)容對應(yīng),這個很簡單:

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 { 			// 創(chuàng)建一個url對象 			URL url = new URL(path); 			// 通過url對象,創(chuàng)建一個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; 	}  } 

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

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

OK,到了這一步,準(zhǔn)備工作都做完了,接下來只剩一個類了!即我們的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的方式去進行異步加載版本信息,主要是因為在安卓中要把耗時任務(wù)放在非主線程中執(zhí)行,否則會造成阻塞,拋出無響應(yīng)異常。還有另外的實現(xiàn)方式是安卓封裝的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; 		} 	}  	// 獲取當(dāng)前版本的版本號 	private String getVersion() { 		try { 			PackageManager packageManager = getPackageManager(); 			PackageInfo packageInfo = packageManager.getPackageInfo( 					getPackageName(), 0); 			return packageInfo.versionName; 		} catch (NameNotFoundException e) { 			e.printStackTrace(); 			return "版本號未知"; 		} 	}
這段里面要注意的是怎么獲取當(dāng)前版本,方法是使用PackageManager提供的getPackageInfo方法,返回的是manifest文件中的版本號。其他的代碼挺簡單,注釋也挺全的。如果有問題,歡迎留言。

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

	void downFile(final String url) {  		pBar = new ProgressDialog(MainActivity.this);    //進度條,在下載的時候?qū)崟r更新進度,提高用戶友好度 		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);                            //設(shè)置進度條的總長度 					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];   //這個是緩沖區(qū),即一次讀取10個比特,我弄的小了點,因為在本地,所以數(shù)值太大一 下就下載完了,看不出progressbar的效果。 						int ch = -1; 						int process = 0; 						while ((ch = is.read(buf)) != -1) {        							fileOutputStream.write(buf, 0, ch); 							process += ch; 							pBar.setProgress(process);       //這里就是關(guān)鍵的實時更新進度了! 						}  					} 					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在下載的時候?qū)崟r更新進度,主要利用的是一個字節(jié)數(shù)組的緩沖區(qū)。即每次獲取到的內(nèi)容填滿緩沖區(qū)后就寫入到本地本件中。這里我把緩沖區(qū)的大小設(shè)置為10個字節(jié)(1024會比較好),理由是因為在同一個局域網(wǎng)中速度特別快,刷一下就下載完了,看不出進度條效果,緩沖區(qū)調(diào)小點就OK了。

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

寫在后面:

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

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



向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI