标签:
MVC(Model View Controller 模型-视图-控制器)
那么,MVC各层与Android又有什么对应关系呢?
View:自定义View或ViewGroup,负责将用户的请求通知Controller,并根据model更新界面;
Controller:接收用户请求并更新model;
下面以一个简单的登录demo示例。
使用安卓自带的SQLite建立一个用户,在登录时输入用户名和密码后匹配数据库记录,并显示登录成功或者失败的原因等。
其中,bean中User为实体类,包括用户名和密码;callback作为回调接口在控制层和模型层传递数据处理结果;db为帮助我们在数据库建立一个用户,并提供查询;
public class User { private String name; private String pwd; public String getPwd() { return pwd; } public void setPwd(String pwd) { this.pwd = pwd; } public String getName() { return name; } public void setName(String name) { this.name = name; }
public interface LoginCallBack { void success(User user); void fail(int status); }
1 public class MyDataBaseHelper extends SQLiteOpenHelper { 2 3 4 private static final String CREATE_LOGIN = "create table LoginInfo(" + 5 "id integer primary key autoincrement," + 6 "name text," + 7 "password text)"; 8 9 10 public MyDataBaseHelper(Context context, String name, SQLiteDatabase.CursorFactory factory, int version) { 11 super(context, name, factory, version); 12 } 13 14 @Override 15 public void onCreate(SQLiteDatabase db) { 16 db.execSQL(CREATE_LOGIN); 17 } 18 19 @Override 20 public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { 21 22 } 23 24 public void initData(SQLiteDatabase db) { 25 26 String sql = "insert into LoginInfo(name,password) values(‘test‘,‘test‘)"; 27 db.execSQL(sql); 28 // ContentValues values= new ContentValues(); 29 // values.put("name","test"); 30 // values.put("password","test"); 31 // db.insert("LoginInfo",null,values); 32 33 } 34 35 public int queryData( User user) { 36 37 String sql = "select * from LoginInfo where name=?"; 38 Cursor cursor = this.getReadableDatabase().rawQuery(sql, new String[]{user.getName()}); 39 40 try{ 41 if (cursor.moveToFirst()) { 42 if (cursor.getString(cursor.getColumnIndex("password")).equals(user.getPwd())) { 43 44 return 1;//正确 45 } else { 46 return 0;//密码错误 47 } 48 } 49 return -1;//用户名错误(没有当前用户) 50 }catch (Exception e){ 51 e.printStackTrace(); 52 }finally { 53 cursor.close(); 54 } 55 return -2;//发生错误 56 57 }
先从View层讲起:自定义一个LoginView继承自LinearLayout
<?xml version="1.0" encoding="utf-8"?> <prodigalwang.androidframe.mvc.view.LoginView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/activity_login_mvc" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <EditText android:id="@+id/et_name_mvc" android:layout_width="200dp" android:layout_height="40dp" android:layout_gravity="center_horizontal" android:hint="请输入用户名" /> <EditText android:id="@+id/et_pwd_mvc" android:layout_width="200dp" android:layout_height="40dp" android:layout_gravity="center_horizontal" android:hint="请输入密码" android:inputType="textPassword" /> <Button android:id="@+id/bt_login_mvc" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center_horizontal" android:layout_marginTop="20dp" android:text="登录" /> </prodigalwang.androidframe.mvc.view.LoginView>
通常我们直接在activity中直接初始化各种控件,而我们这样做之后只不过是将布局的初始化化代码移入到自定义的view中。
1 /** 2 * Author:ProdigalWang 3 * Time: 2016/9/26 4 * 视图层,直接展示给用户。通过视图层将数据请求传送到控制层(Controller) 5 * 作为视图层来说,它只是作为接受用户数据和展示数据的方式 6 */ 7 public class LoginView extends LinearLayout { 8 9 private Context mContext; 10 private EditText mUsername; 11 private EditText mPassword; 12 private Button mLoginBtn; 13 14 public LoginView(Context context, AttributeSet attrs) { 15 super(context, attrs); 16 mContext = context; 17 } 18 19 public void initView() { 20 mUsername = (EditText) findViewById(R.id.et_name_mvc); 21 mPassword = (EditText) findViewById(R.id.et_pwd_mvc); 22 mLoginBtn = (Button) findViewById(R.id.bt_login_mvc); 23 } 24 25 public String getName() { 26 return mUsername.getText().toString(); 27 } 28 29 public String getPwd() { 30 return mPassword.getText().toString(); 31 } 32 33 public void setOnclikLister(OnClickListener onclikLister) { 34 mLoginBtn.setOnClickListener(onclikLister); 35 } 36 37 public void userNameEpty() { 38 Toast.makeText(mContext, "用户名不能为空", Toast.LENGTH_SHORT).show(); 39 } 40 41 public void passWordEpty() { 42 Toast.makeText(mContext, "密码不能为空", Toast.LENGTH_SHORT).show(); 43 } 44 45 public void userNameError() { 46 Toast.makeText(mContext, "用户名错误", Toast.LENGTH_SHORT).show(); 47 } 48 49 public void passWordError() { 50 Toast.makeText(mContext, "密码错误", Toast.LENGTH_SHORT).show(); 51 } 52 53 public void loginSuccess() { 54 Toast.makeText(mContext, "登录成功", Toast.LENGTH_SHORT).show(); 55 } 56 57 public void loginFailure() { 58 Toast.makeText(mContext, "登录失败", Toast.LENGTH_SHORT).show(); 59 } 60 }
这样我们就完成了View层的抽取。
下面实现Model层:首先我们需要先抽取出一个接口,由于我们的需求很简单,只是一个登陆操作,只需要验证用户输入的用户名和密码是否正确,所以我们的抽取的接口中也只有一个login()方法。
/** * 模型层———登录接口 */ public interface ILoginModel { void login(String name, String pwd, LoginCallBack loginCallBack); }
/** * Author:ProdigalWang * Time: 2016/9/26 * 模型层实现,完成具体的数据操作。 */ public class LoginModelImpl implements ILoginModel { private MyDataBaseHelper myDataBaseHelper; @Override public void login(String name, String pwd, LoginCallBack loginCallBack) { User user=new User(); user.setName(name); user.setPwd(pwd); myDataBaseHelper = new MyDataBaseHelper(MyAppliction.getContext(), "Login.db", null, 1); int result = myDataBaseHelper.queryData(user); //发出处理结果,用户得到反馈 if (result == 1) { loginCallBack.success(user); } else { loginCallBack.fail(result); } } }
最后,完成Controller:
/** * Author:ProdigalWang * Time: 2016/9/26 */ public class LoginController implements View.OnClickListener { private LoginView loginView; private ILoginModel iLoginModel; public LoginController( LoginView loginView){ this.loginView=loginView; iLoginModel=new LoginModelImpl(); } @Override public void onClick(View v) { switch (v.getId()){ case R.id.bt_login_mvc: String username=loginView.getName(); String pwd=loginView.getPwd(); if (TextUtils.isEmpty(username)){ loginView.userNameEpty(); break; } if (TextUtils.isEmpty(pwd)){ loginView.passWordEpty(); break; } //调用模型层去处理具体的请求 iLoginModel.login(username, pwd, new LoginCallBack() { @Override public void success(User user) { loginView.loginSuccess(); } @Override public void fail(int status) { if (status==0){ //模型层完成数据处理后,通知视图层做出相应的改变。用户得到反馈。 loginView.passWordError(); }else if (status==-1){ loginView.userNameError(); }else { loginView.loginFailure(); } } }); break; } } }
那么,我们实现了MVC框架模式后,我们的Activity里的代码又变成怎么样了呢?
** * 此时Activity就变为了承载视图层的容器。 */ public class MvcLoginActivity extends AppCompatActivity { private MyDataBaseHelper myDataBaseHelper; private SQLiteDatabase db; private LoginView loginView; private LoginController loginController; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_mvc_login); //初始化一个本地用户用来模拟登陆。 myDataBaseHelper = new MyDataBaseHelper(this, "Login.db", null, 1); db = myDataBaseHelper.getWritableDatabase(); //myDataBaseHelper.initData(db); MVC(); } private void MVC(){ //控件初始化和绑定 loginView= (LoginView) findViewById(R.id.activity_login_mvc); loginView.initView(); loginController=new LoginController(loginView);//视图层结合控制层 loginView.setOnclikLister(loginController); } }
没错,这个时候activity里面的代码就是如此的少。关键之处的代码注释写了,这里就不做详细解释了。
那么,来总结一下整个流程:首先用户打开应用,loginView.initView()调用,显示整个布局。这时候用户输入用户名名和密码后点击登录按钮,Controller层通过View层的getName()和getPwd()获取到用户输入的数据,紧接着Controller层会调用Model的接口中的login()方法,将数据传递给Model层进行具体的处理。当Model完成处理后,会通过LoginCallBack回调处理的结果,并通知View层进行视图的改变,这是用户得到反馈结果。
最后,总结一下MVC带来的好处与不足 :
好处:1.总所周知,采用各种各种框架模式都是为了实现高内聚低耦合,实现分层能够实现良好的分工合作,各层独立,修改哪一层对其他层的影响都能降低很多。
2.重用性高,由于模型返回的数据没有进行格式化,所以同样的构件能被不同的界面使用。例如,很多数据可能用HTML来表示,但是也有可能用WAP来表示,而这些表示所需要的命令是改变视图层的实现方式,而控制层和模型层无需做任何改变。
3.维护性高,部署快。不同的开发人员只需要专注于一层的实现。
不足之处:1.不适于小型程序,如果我们在小程序上为了实现MVC框架模式而实现的话,会浪费大量不必要的时间和精力。
2.增加系统结构和实现的复杂性等。
参考资料:
1.http://baike.baidu.com/link?url=3hzUtpjpvrAJCJwjQ1OlGZZVU7Ri_7cXK0gZSqZf-IuR7sRpNRruaG6TtpV3tgJNWQ6l4YR4N6pyk99j3TX9Y2aj1yBAb837M2cLMK5J5MUEBCyjzmVOhT_3Q2_HDPMv
2.https://github.com/jpush/jchat-android
标签:
原文地址:http://www.cnblogs.com/FuckHard/p/5917801.html