根據(jù)功能模塊劃分(Android開發(fā)推薦此方法)
趙縣網(wǎng)站制作公司哪家好,找創(chuàng)新互聯(lián)!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、成都響應(yīng)式網(wǎng)站建設(shè)等網(wǎng)站項目制作,到程序開發(fā),運(yùn)營維護(hù)。創(chuàng)新互聯(lián)自2013年創(chuàng)立以來到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗和運(yùn)維經(jīng)驗,來保證我們的工作的順利進(jìn)行。專注于網(wǎng)站建設(shè)就選創(chuàng)新互聯(lián)。
- Activity mobilesafe.activty
- 后臺服務(wù) mobilesafe.service
- 廣播接受者mobilesafe.receiver
- 數(shù)據(jù)庫mobilesafe.db.dao
- 對象(java bean)mobilesafe.domain/bean
- 自定義控件mobilesafe.view
- 工具類mobilesafe.utils
- 業(yè)務(wù)邏輯mobilesafe.engine
閃屏頁面(Splash)作用:
- 展示logo,公司品牌
- 項目初始化
- 檢測版本更新
- 校驗程序合法性(比如:判斷是否有網(wǎng)絡(luò),有的話才運(yùn)行)
AndroidMinifest.xml 四大組件都需要在這里配置
<?xml version="1.0" encoding="utf-8"?>//版本名 //項目所需的權(quán)限 //主題 //activity的注冊 //起始的activity //廣播接收者的 注冊 //服務(wù)的注冊
activity_splash.xml
SplashActivity.java
public class SplashActivity extends Activity { protected static final int CODE_UPDATE_DIALOG; protected static final int CODE_URL_ERROR; protected static final int CODE_NET_ERROR; protected static final int CODE_JSON_ERROR; protected static final int CODE_ENTER_HOME; private TextView tvVersion; private TextView tvProgress;// 下載進(jìn)度展示 // 服務(wù)器返回的信息 private String mversionName;// 版本名 private int mversionCode;// 版本號 private String mDesc;// 版本描述 private String mdowmloadurl;// 下載地址 private Handler mHandler = new Handler() { public void handleMessage(android.os.Message msg) { switch (msg.what) { case CODE_UPDATE_DIALOG: showUpdateDialog();//顯示升級對話框 break; case CODE_URL_ERROR: Toast.makeText(SplashActivity.this, "url錯誤", Toast.LENGTH_SHORT).show(); enterHome(); break; case CODE_NET_ERROR: Toast.makeText(SplashActivity.this, "網(wǎng)絡(luò)錯誤", Toast.LENGTH_SHORT).show(); enterHome(); break; case CODE_JSON_ERROR: Toast.makeText(SplashActivity.this, "json數(shù)據(jù)解析解析錯誤", Toast.LENGTH_SHORT).show(); enterHome(); break; case CODE_ENTER_HOME: enterHome(); break; default: break; } }; }; private SharedPreferences sp; private RelativeLayout rlRoot; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_splash); tvVersion = (TextView) findViewById(R.id.tv_version); tvProgress = (TextView) findViewById(R.id.tv_progress);// 默認(rèn)隱藏 tvVersion.setText("版本號:" + getVersionCode());//給版本號設(shè)置內(nèi)容,動態(tài)獲取的值 rlRoot = (RelativeLayout) findViewById(R.id.rl_root); //判斷是否需要自動更新 sp = getSharedPreferences("config", MODE_PRIVATE); boolean autoUpdate = sp.getBoolean("auto_update", true); copyDB("address.db");//拷貝歸屬地查詢數(shù)據(jù)庫 copyDB("antivirus.db");//拷貝病毒庫 //更新病毒庫 updateVirus(); if(autoUpdate){ checkVersion(); }else{ mHandler.sendEmptyMessageDelayed(CODE_ENTER_HOME, 2000); } //閃屏頁漸變動畫效果 AlphaAnimation anim = new AlphaAnimation(0.3f, 1f); anim.setDuration(2000); rlRoot.startAnimation(anim); } //更新病毒數(shù)據(jù)庫 private void updateVirus() { //聯(lián)網(wǎng)從服務(wù)器獲取到最近數(shù)據(jù)的MD5的特征碼 HttpUtils httputils = new HttpUtils(); String url = "http://172.28.3.112:8080/virus.json"; httputils.send(HttpMethod.GET, url, new RequestCallBack(){ @Override public void onFailure(HttpException arg0, String arg1) { // TODO Auto-generated method stub } @Override public void onSuccess(ResponseInfo arg0) { // TODO Auto-generated method stub //System.out.println(arg0.result); // JSONObject jsonobject = new JSONObject(arg0.result); // String md5 = jsonobject.getString("md5"); // String desc = jsonobject.getString("desc"); } }); } // 獲取本地版本號 private int getVersionCode() { PackageManager packgeManager = getPackageManager();//拿到包的管理者。。包管理器,獲取手機(jī)里面每個apk的信息(清單文件信息) try {// 獲取包的信息。。 getPackageName()當(dāng)前應(yīng)用程序的包名 等于 package="com.mxn.mobilesafe" PackageInfo packageInfo = packgeManager.getPackageInfo(getPackageName(), 0); int versionCode = packageInfo.versionCode; String versionName = packageInfo.versionName; System.out.println("versionname=" + versionName + ";" + "versioncode=" + versionCode); return versionCode; } catch (NameNotFoundException e) { // 沒有找到包名時 e.printStackTrace(); } return -1; } // 從服務(wù)器獲取版本信息進(jìn)行校驗 private void checkVersion() { final long startTime = System.currentTimeMillis(); new Thread() {// 網(wǎng)絡(luò)訪問在分線程異步加載數(shù)據(jù) public void run() { Message msg = Message.obtain(); HttpURLConnection con = null; try {// 本機(jī)地址:localhost 如果用模擬器加載本機(jī)的地址:用10.0.0.2來替換 URL url = new URL("http://10.0.2.2:8080/update.json"); // 打開連接 con = (HttpURLConnection) url.openConnection(); con.setRequestMethod("GET");//設(shè)置請求方法 con.setConnectTimeout(5000);// 設(shè)置連接超時,5S con.setReadTimeout(5000);// 設(shè)置響應(yīng)超時,鏈接上了,但服務(wù)器遲遲沒有響應(yīng) con.connect();// 鏈接服務(wù)器 int responseCode = con.getResponseCode();//獲取響應(yīng)碼 if (responseCode == 200) { // 獲取返回值 InputStream inputStream = con.getInputStream(); // 流轉(zhuǎn)化為字符串 String result = StreamUtils.readFormStream(inputStream);//自己定義的StreamUtils工具類 System.out.println("網(wǎng)絡(luò)結(jié)果返回:" + result); //result是一個json字符串,進(jìn)行解析 // 解析json JSONObject jo = new JSONObject(result); mversionName = jo.getString("versionName");//拿到服務(wù)器端的版本名 mversionCode = jo.getInt("versionCode");//拿到服務(wù)器端的版本號 mDesc = jo.getString("description");//拿到服務(wù)器端的版本描述 mdowmloadurl = jo.getString("downloadUrl");//拿到服務(wù)器端的下載鏈接 System.out.println(mDesc); System.out.println(mversionCode); // 服務(wù)器的大于 本地的,判斷是否有更新,如果大于 則有更新需要更新,彈出升級對話框 if (mversionCode > getVersionCode()) { System.out.println("進(jìn)行比較,有版本更新"); msg.what = CODE_UPDATE_DIALOG; // showUpdateDialog();//這句是在子線程更新界面,android不能在子線程更新界面,要想在子線程更新界面所以用到handler. } else {// 如果沒有版本更新 msg.what = CODE_ENTER_HOME; } } } catch (MalformedURLException e) {// url錯誤的異常 msg.what = CODE_URL_ERROR; e.printStackTrace(); } catch (IOException e) {//網(wǎng)絡(luò)錯誤異常 // 這個是可以攜帶數(shù)據(jù)的msg.obj = msg.what = CODE_NET_ERROR;// what只是一個標(biāo)識,用來區(qū)分消息! e.printStackTrace(); } catch (JSONException e) {// json解析失敗 msg.what = CODE_JSON_ERROR; e.printStackTrace(); } finally { long endTime = System.currentTimeMillis(); long timeUsed = endTime - startTime;// 訪問網(wǎng)絡(luò)花費(fèi)的時間 if (timeUsed < 2000) { try {// 強(qiáng)制休眠2s,保證閃屏頁面2S Thread.sleep(2000 - timeUsed); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } mHandler.sendMessage(msg);// 消息發(fā)送出去,在handlemessage里進(jìn)行相應(yīng)的處理 if (con != null) { con.disconnect(); } } } }.start(); } //升級對話框 private void showUpdateDialog() { System.out.println("正在升級對話框"); // 升級對話框 AlertDialog.Builder builder = new AlertDialog.Builder(this);//context對象 builder.setTitle("最新版本" + mversionName); builder.setMessage(mDesc); // builder.setCancelable(false);//不讓用戶取消對話框,用戶體驗太差 builder.setPositiveButton("立即更新", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // TODO Auto-generated method stub System.out.println("立即更新"); // download方法 download(); } }); builder.setNegativeButton("以后再說", new OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { enterHome(); } }); builder.setOnCancelListener(new OnCancelListener() { // 設(shè)置取消監(jiān)聽,用戶點擊返回鍵時觸發(fā) @Override public void onCancel(DialogInterface dialog) { enterHome(); } }); builder.show(); } protected void download() {// 下載服務(wù)器端的apk文件 if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) { // 判斷是否有sd卡,sd卡掛載的時候才可以 tvProgress.setVisibility(View.VISIBLE);// 顯示進(jìn)度 String target = Environment.getExternalStorageDirectory() + "/update.apk";//把文件下載到哪個路徑下,sd卡的根目錄 // xutils框架,使用HttpUtils工具下載文件,下載一個jar包 HttpUtils utils = new HttpUtils(); utils.download(mdowmloadurl, target, new RequestCallBack () { @Override // 文件下載進(jìn)度 public void onLoading(long total, long current, boolean isUploading) { // TODO Auto-generated method stub super.onLoading(total, current, isUploading); System.out.println("下載進(jìn)度:" + current + "/" + total); tvProgress.setText("下載進(jìn)度:" + current * 100 / total + "%"); } @Override public void onSuccess(ResponseInfo arg0) { // TODO Auto-generated method stub Toast.makeText(SplashActivity.this, "下載成功", Toast.LENGTH_SHORT).show(); // 下載完成之后,跳到系統(tǒng)的安裝界面。。Intent.ACTION_VIEW 是xml的action 標(biāo)簽 Intent intent = new Intent(Intent.ACTION_VIEW);//系統(tǒng)的安裝界面 intent.addCategory(Intent.CATEGORY_DEFAULT); intent.setDataAndType(Uri.fromFile(arg0.result), "application/vnd.android.package-archive"); // startActivity(intent); startActivityForResult(intent, 0);// 如果用戶取消安裝,會返回結(jié)果,回調(diào)方法onActivityResult,下文定義 } @Override public void onFailure(HttpException arg0, String arg1) { // TODO Auto-generated method stub Toast.makeText(SplashActivity.this, "下載失敗", Toast.LENGTH_SHORT).show(); } }); } else { Toast.makeText(SplashActivity.this, "沒有SD卡", Toast.LENGTH_SHORT).show(); } } @Override//用戶取消安裝,回調(diào)此方法 protected void onActivityResult(int requestCode, int resultCode, Intent data) { // TODO Auto-generated method stub System.out.println("出現(xiàn)安裝界面,用戶點擊取消時。"); enterHome(); super.onActivityResult(requestCode, resultCode, data); } private void enterHome() {// 進(jìn)入主界面 Intent intent = new Intent(this, HomeActivity.class); startActivity(intent); finish(); } //拷貝數(shù)據(jù)庫,從assets目錄下拷貝到data/data/com.mxn.mobilesafe/files目錄下 private void copyDB(String dbName){ //獲取文件路徑 File destFile = new File(getFilesDir(),dbName); if(destFile.exists()){ System.out.println("已存在"); } FileOutputStream out = null; InputStream in = null; try { in = getAssets().open(dbName); out = new FileOutputStream(destFile); int len = 0; byte[] buffer = new byte[1024]; while((len = in.read(buffer))!=-1){ out.write(buffer,0,len); } } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); }finally{ try { in.close(); out.close(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } } }
StreamUtils.java
/* * 讀取流的工具 * 把流對象轉(zhuǎn)換成字符串對象 */ public class StreamUtils { //將輸入流讀取成String后返回 public static String readFormStream(InputStream in) throws IOException{ // 定義字節(jié)數(shù)組輸出流對象 ByteArrayOutputStream out = new ByteArrayOutputStream(); // 定義讀取的長度 int len = 0 ; // 定義讀取的緩沖區(qū) byte[] buffer = new byte[1024]; // 按照定義的緩沖區(qū)進(jìn)行循環(huán)讀取,直到讀取完畢為止 while((len=in.read(buffer))!=-1){ // 根據(jù)讀取的長度寫入到字節(jié)數(shù)組輸出流對象中 out.write(buffer,0,len); } String result = out.toString(); // 關(guān)閉流 in.close(); out.close(); return result; // // 把讀取的字節(jié)數(shù)組輸出流對象轉(zhuǎn)換成字節(jié)數(shù)組 // byte data[] = out.toByteArray(); // // 按照指定的編碼進(jìn)行轉(zhuǎn)換成字符串(此編碼要與服務(wù)端的編碼一致就不會出現(xiàn)亂碼問題了,android默認(rèn)的編碼為UTF-8) // return new String(data, "UTF-8"); } }
系統(tǒng)安裝界面的activity的配置:
我們服務(wù)器用的是tomcat,里面放置 新版本的apk和update.json:
將代碼打包為apk文件:
涉及的知識點:
PackageManager 包管理器,獲取手機(jī)里面每個apk的信息(清單文件信息)
版本更新流程:
網(wǎng)絡(luò)請求
> * URL
> * HttpUrlConntetion
JSON解析
> * JSONObject 專門用來解析json
> * JSONArray
對話框彈出
> AlertDialog
> AlertDialog.Builder
子線程更新UI
> * Handler + message
> * runOnUiThread(runnable)
頁面之間的跳轉(zhuǎn)Intent
GitHub 一個開源的網(wǎng)站,下載xUtils框架,將下載的jar包導(dǎo)入工程。
AlertDialog.Builder(this)
子類擁有父類的所有方法, 而且可以有更多自己的方法。父類無法有子類的方法
Activity(token), Context(沒有token)
平時,要獲取context對象的話, 優(yōu)先選擇Activity, 避免bug出現(xiàn), 盡量不用getApplicationContext()
activity是context的子類
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時也希望多多支持創(chuàng)新互聯(lián)!