轉載:android MVC設計模式

来源:http://www.cnblogs.com/xiaoxiaing/archive/2017/02/07/6372876.html
-Advertisement-
Play Games

Controller控制器 import android.app.Dialog; import android.app.ProgressDialog; import android.os.Bundle; import android.support.v7.app.ActionBarActivity; ...


Controller控制器

import android.app.Dialog;  

import android.app.ProgressDialog;  

  1. import android.os.Bundle;  
  2. import android.support.v7.app.ActionBarActivity;  
  3. import android.view.View;  
  4. import android.widget.EditText;  
  5. import android.widget.TextView;  
  6. import android.widget.Toast;  
  7. import com.xjp.androidmvcdemo.R;  
  8. import com.xjp.androidmvcdemo.entity.Weather;  
  9. import com.xjp.androidmvcdemo.entity.WeatherInfo;  
  10. import com.xjp.androidmvcdemo.model.OnWeatherListener;  
  11. import com.xjp.androidmvcdemo.model.WeatherModel;  
  12. import com.xjp.androidmvcdemo.model.WeatherModelImpl;  
  13. public class MainActivity extends ActionBarActivity implements OnWeatherListener, View.OnClickListener {  
  14.     private WeatherModel weatherModel;  
  15.     private Dialog loadingDialog;  
  16.     private EditText cityNOInput;  
  17.     private TextView city;  
  18.     private TextView cityNO;  
  19.     private TextView temp;  
  20.     private TextView wd;  
  21.     private TextView ws;  
  22.     private TextView sd;  
  23.     private TextView wse;  
  24.     private TextView time;  
  25.     private TextView njd;  
  26.     @Override  
  27.     protected void onCreate(Bundle savedInstanceState) {  
  28.         super.onCreate(savedInstanceState);  
  29.         setContentView(R.layout.activity_main);  
  30.         weatherModel = new WeatherModelImpl();  
  31.         initView();  
  32.     }  
  33.     /** 
  34.      * 初始化View 
  35.      */  
  36.     private void initView() {  
  37.         cityNOInput = findView(R.id.et_city_no);  
  38.         city = findView(R.id.tv_city);  
  39.         cityNO = findView(R.id.tv_city_no);  
  40.         temp = findView(R.id.tv_temp);  
  41.         wd = findView(R.id.tv_WD);  
  42.         ws = findView(R.id.tv_WS);  
  43.         sd = findView(R.id.tv_SD);  
  44.         wse = findView(R.id.tv_WSE);  
  45.         time = findView(R.id.tv_time);  
  46.         njd = findView(R.id.tv_njd);  
  47.         findView(R.id.btn_go).setOnClickListener(this);  
  48.         loadingDialog = new ProgressDialog(this);  
  49.         loadingDialog.setTitle(載入天氣中...);  
  50.     }  
  51.     /** 
  52.      * 顯示結果 
  53.      * 
  54.      * @param weather 
  55.      */  
  56.     public void displayResult(Weather weather) {  
  57.         WeatherInfo weatherInfo = weather.getWeatherinfo();  
  58.         city.setText(weatherInfo.getCity());  
  59.         cityNO.setText(weatherInfo.getCityid());  
  60.         temp.setText(weatherInfo.getTemp());  
  61.         wd.setText(weatherInfo.getWD());  
  62.         ws.setText(weatherInfo.getWS());  
  63.         sd.setText(weatherInfo.getSD());  
  64.         wse.setText(weatherInfo.getWSE());  
  65.         time.setText(weatherInfo.getTime());  
  66.         njd.setText(weatherInfo.getNjd());  
  67.     }  
  68.     /** 
  69.      * 隱藏進度對話框 
  70.      */  
  71.     public void hideLoadingDialog() {  
  72.         loadingDialog.dismiss();  
  73.     }  
  74.     @Override  
  75.     public void onClick(View v) {  
  76.         switch (v.getId()) {  
  77.             case R.id.btn_go:  
  78.                 loadingDialog.show();  
  79.                 weatherModel.getWeather(cityNOInput.getText().toString().trim(), this);  
  80.                 break;  
  81.         }  
  82.     }  
  83.     @Override  
  84.     public void onSuccess(Weather weather) {  
  85.         hideLoadingDialog();  
  86.         displayResult(weather);  
  87.     }  
  88.     @Override  
  89.     public void onError() {  
  90.         hideLoadingDialog();  
  91.         Toast.makeText(this, 獲取天氣信息失敗, Toast.LENGTH_SHORT).show();  
  92.     }  
  93.     private <t extends="" view=""> T findView(int id) {  
  94.         return (T) findViewById(id);  
  95.     }  
  96. }  
從上面代碼可以看到,Activity持有了WeatherModel模型的對象,當用戶有點擊Button交互的時候,Activity作為Controller控制層讀取View視圖層EditTextView的數據,然後向Model模型發起數據請求,也就是調用WeatherModel對象的方法 getWeathre()方法。當Model模型處理數據結束後,通過介面OnWeatherListener通知View視圖層數據處理完畢,View視圖層該更新界面UI了。然後View視圖層調用displayResult()方法更新UI。至此,整個MVC框架流程就在Activity中體現出來了。

Model模型

來看看WeatherModelImpl代碼實現

  1. package com.xjp.androidmvcdemo.model;  
  2.    
  3. /** 
  4.  * Description:請求網路數據介面 
  5.  * User: xjp 
  6.  * Date: 2015/6/3 
  7.  * Time: 15:40 
  8.  */  
  9.    
  10. public interface WeatherModel {  
  11.     void getWeather(String cityNumber, OnWeatherListener listener);  
  12. }  
  13.    
  14. ................  
  15.    
  16.    
  17. package com.xjp.androidmvcdemo.model;  
  18.    
  19. import com.android.volley.Response;  
  20. import com.android.volley.VolleyError;  
  21. import com.xjp.androidmvcdemo.entity.Weather;  
  22. import com.xjp.androidmvcdemo.volley.VolleyRequest;  
  23.    
  24. /** 
  25.  * Description:從網路獲取天氣信息介面實現 
  26.  * User: xjp 
  27.  * Date: 2015/6/3 
  28.  * Time: 15:40 
  29.  */  
  30.    
  31. public class WeatherModelImpl implements WeatherModel {  
  32.    
  33.     @Override  
  34.     public void getWeather(String cityNumber, final OnWeatherListener listener) {  
  35.    
  36.         /*數據層操作*/  
  37.         VolleyRequest.newInstance().newGsonRequest(http://www.weather.com.cn/data/sk/ + cityNumber + .html,  
  38.                 Weather.class, new Response.Listener<weather>() {  
  39.                     @Override  
  40.                     public void onResponse(Weather weather) {  
  41.                         if (weather != null) {  
  42.                             listener.onSuccess(weather);  
  43.                         } else {  
  44.                             listener.onError();  
  45.                         }  
  46.                     }  
  47.                 }, new Response.ErrorListener() {  
  48.                     @Override  
  49.                     public void onErrorResponse(VolleyError error) {  
  50.                         listener.onError();  
  51.                     }  
  52.                 });  
  53.     }  
  54. }  

以上代碼看出,這裡設計了一個WeatherModel模型介面,然後實現了介面WeatherModelImpl類。controller控制器activity調用WeatherModelImpl類中的方法發起網路請求,然後通過實現OnWeatherListener介面來獲得網路請求的結果通知View視圖層更新UI 。至此,Activity就將View視圖顯示和Model模型數據處理隔離開了。activity擔當contronller完成了model和view之間的協調作用。

至於這裡為什麼不直接設計成類裡面的一個getWeather()方法直接請求網路數據?你考慮下這種情況:現在代碼中的網路請求是使用Volley框架來實現的,如果哪天老闆非要你使用Afinal框架實現網路請求,你怎麼解決問題?難道是修改 getWeather()方法的實現? no no no,這樣修改不僅破壞了以前的代碼,而且還不利於維護, 考慮到以後代碼的擴展和維護性,我們選擇設計介面的方式來解決著一個問題,我們實現另外一個WeatherModelWithAfinalImpl類,繼承自WeatherModel,重寫裡面的方法,這樣不僅保留了以前的WeatherModelImpl類請求網路方式,還增加了WeatherModelWithAfinalImpl類的請求方式。Activity調用代碼無需要任何修改。

MVC使用總結

利用MVC設計模式,使得這個天氣預報小項目有了很好的可擴展和維護性,當需要改變UI顯示的時候,無需修改Contronller(控制器)Activity的代碼和Model(模型)WeatherModel模型中的業務邏輯代碼,很好的將業務邏輯和界面顯示分離。

Android項目中,業務邏輯,數據處理等擔任了Model(模型)角色,XML界面顯示等擔任了View(視圖)角色,Activity擔任了Contronller(控制器)角色。contronller(控制器)是一個中間橋梁的作用,通過介面通信來協同 View(視圖)和Model(模型)工作,起到了兩者之間的通信作用。

什麼時候適合使用MVC設計模式?當然一個小的項目且無需頻繁修改需求就不用MVC框架來設計了,那樣反而覺得代碼過度設計,代碼臃腫。一般在大的項目中,且業務邏輯處理複雜,頁面顯示比較多,需要模塊化設計的項目使用MVC就有足夠的優勢了。

4.在MVC模式中我們發現,其實控制器Activity主要是起到解耦作用,將View視圖和Model模型分離,雖然Activity起到交互作用,但是找Activity中有很多關於視圖UI的顯示代碼,因此View視圖和Activity控制器並不是完全分離的,也就是說一部分View視圖和Contronller控制器Activity是綁定在一個類中的。

MVC的優點:

(1)耦合性低。所謂耦合性就是模塊代碼之間的關聯程度。利用MVC框架使得View(視圖)層和Model(模型)層可以很好的分離,這樣就達到瞭解耦的目的,所以耦合性低,減少模塊代碼之間的相互影響。

(2)可擴展性好。由於耦合性低,添加需求,擴展代碼就可以減少修改之前的代碼,降低bug的出現率。

(3)模塊職責劃分明確。主要劃分層M,V,C三個模塊,利於代碼的維護。

 
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這個問題很傻很天真,如何正確的添加圖像資源 我有一個名為 back.png 的圖片,直接拷貝到 Finder 的 project 目錄下了。 用如下的語句: roleImageView.image = UIImage(named: "back") 顯示不了圖片。 其實不是語句有問題,是添加圖像的方法 ...
  • 當你在一個Xcode版本上編輯Storyboard並儲存後(比如 8.1)在另一個版本上(比如8.2.1)打開想繼續編輯的時候,有時候會無法打開Storyboard。 所以兩人合作編寫一個 Xcode project 的時候,一定要使用相同的版本。不然以後麻煩多多,會出現很多莫名其妙的問題。 有兩個 ...
  • AToolsActivity.java shake.xml AddressDao.java ...
  • 科技的仿生學無處不在,給予我們啟發。為了延長電池是使用壽命,google從蛇的冬眠中得到體會,那就是在某種情況下也讓手機進入類冬眠的情況,從而引入了今天的主題,Doze模式,Doze中文是打盹兒,打盹當然比活動節約能量了。 手機打盹兒的時候會怎樣呢? 按照google的官方說法,Walklocks, ...
  • 1:倉庫地址:https://github.com/jsonmodel/jsonmodel 主要作用是把JSON字元串轉成Model實體,也可以把實體轉化成JSON字元串;還包含一些轉字典的內容;JOSN字元串通過AFNetworking轉化成字典,然後再能過字典跟實體進行轉換; 2:原理實現: 主 ...
  • 效果: main_activity.xml ...
  • 如果我們的程式是在單線程下運行,或者是不必考慮到線程同步問題,我們應該優先使用StringBuilder類;如果要保證線程安全,自然是StringBuffer。 除了對多線程的支持不一樣外,這兩個類的使用方式和結果幾乎沒有任何差別, 區別在於StringBufferd支持併發操作,線性安全的,適 合 ...
  • 修改定價將你的app定價修改成0.99刀 修改你的發行範圍,全取消後只選中國。 save這時候你的app status將會變成pending contract。 將之前的修改都改回來,修改定價free,全選區域,然後save 這時候app status又會變成Ready for sale了,過個半小 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...