目录
MVP模式是Android常见的的一种架构模式,全称是Model、View、Presenter。其中,Model负责数据处理部分,View负责UI界面展示及用户操作交互,而Presenter则是负责大部分运行逻辑的编写。在Android中,常见的MVC架构模式一般是直接在Activity中实现运行逻辑,这样如果功能比较复杂,Activity的代码可读性将会很低,同时耦合性将会很高。与MVC模式相比,MVP模式断开了Model与View的直接交互,降低了代码的耦合性,也提高了代码的可复用性。
本文以登录注册功能的编写演示MVP架构的实现方式。登录注册UI界面的编写可以参考这篇文章:Android开发:登录/注册界面的编写,在这里不再赘述。
Android中Activity一般充当View的角色,在Activity中实现对用户输入的检测以及交互;Presenter则负责将View传来的用户输入信息(Username、Password等)传给Model,由Model进行具体的数据判断,并将判断结果(用户是否存在、密码是否正确等)回传给Presenter,Presenter在收到Model的回传后,再调用View中的UI更新方法对界面进行更新,具体实现如下:
我们编写一个LoginModelInterface接口用于表征Model,并新建一个LoginModel类实现LoginModelInterface接口。在LoginModelInterface中增加一个isUserValid(String username, String password)方法,并在LoginModel中对该方法进行实现:
public class LoginModel implements LoginModelInterface {
@Override
public boolean isUserValid(String username, String password) {
// 在本方法中对用户输入的用户名及密码进行判断,并返回登录结果
return false;
}
}
我们使用MainActivity充当View的角色。同样的,我们新建一个LoginViewInterface接口,在LoginViewInterface接口中编写两个方法:loginButtonClicked()以及showLoginResult():
public interface LoginViewInterface {
void loginButtonClicked(String username, String password);
void showLoginResult(boolean result);
}
让MainActivity实现该接口,并实现上述两个方法:
@Override
public void loginButtonClicked(String username, String password) {
mLoginPresenter.loginButtonClicked(username, password);
}
@Override
public void showLoginResult(boolean isValid) {
if(isValid) {
Toast.makeText(getApplicationContext(), "Login Success !", Toast.LENGTH_LONG).show();
} else {
Toast.makeText(getApplicationContext(), "Login Failed !", Toast.LENGTH_LONG).show();
}
}
在loginButtonClicked()方法中调用LoginPresenter的对应方法进行具体处理;而在showLoginResult()中根据LoginPresenter回传的结果设置具体的展示。为了方便演示,我们通过Toast的方式提示登录结果。
同样的,新建一个LoginPresenterInterface接口用于表征Presenter,并新建一个LoginPresenter类实现上述接口,重写该接口中的loginButtonClicked()方法:
@Override
public void loginButtonClicked(String username, String password) {
boolean isValid = mLoginModel.isUserValid(username, password);
mLoginView.showLoginResult(isValid);
}
在该方法中,通过调用LoginModel的isUserValid()方法进行用户输入的合法性判断,并调用LoginView的showLoginResult()方法对UI界面进行更新,以此实现“中间人”的角色。
LoginView为例,在LoginPresenter中,为了实现对LoginView中界面更新方法的调用,在LoginPresenter的构造方法中传入了LoginView的引用。如果不定义一个LoginViewInterface而是直接传入MainActivity,那么这个LoginPresenter只能与MainActivity一一对应,而如果传入的参数设置为LoginViewInterfce,那么只要让其他的View实现LoginViewInterface,这些View就能使用LoginPresenter,提高了代码的可复用性。LoginModelInterfacepublic interface LoginModelInterface {
boolean isUserValid(String username, String password);
}
LoginModelpublic class LoginModel implements LoginModelInterface {
@Override
public boolean isUserValid(String username, String password) {
// 在本方法中对用户输入的用户名及密码进行判断,并返回登录结果
return false;
}
}
LoginViewInterfacepublic interface LoginViewInterface {
void loginButtonClicked(String username, String password);
void showLoginResult(boolean isValid);
}
MainActivitypublic class MainActivity extends AppCompatActivity implements LoginViewInterface {
private Button mBtLogin;
private EditText mEtUsername, mEtPassword;
private LoginPresenterInterface mLoginPresenter;
String TAG = "LoginPage";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
init();
// set click listener
mBtLogin.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//gain the content of usename and password,for further use
String username = mEtUsername.getText().toString();
String password = mEtPassword.getText().toString();
//Tell Presenter that login button clicked
loginButtonClicked(username, password);
}
});
}
@Override
public void loginButtonClicked(String username, String password) {
mLoginPresenter.loginButtonClicked(username, password);
}
@Override
public void showLoginResult(boolean isValid) {
if(isValid) {
Toast.makeText(getApplicationContext(), "Login Success !", Toast.LENGTH_LONG).show();
} else {
Toast.makeText(getApplicationContext(), "Login Failed !", Toast.LENGTH_LONG).show();
}
}
void init() {
mBtLogin = findViewById(R.id.btn_login);
mEtUsername = findViewById(R.id.et_username);
mEtPassword = findViewById(R.id.et_password);
mLoginPresenter = new LoginPresenter(this);
}
}
LoginPresenterInterfacepublic interface LoginPresenterInterface {
void loginButtonClicked(String username, String password);
}
LoginPresenterpublic class LoginPresenter implements LoginPresenterInterface {
private LoginViewInterface mLoginView;
private LoginModelInterface mLoginModel;
public LoginPresenter(LoginViewInterface loginView) {
mLoginView = loginView;
mLoginModel = new LoginModel();
}
@Override
public void loginButtonClicked(String username, String password) {
boolean isValid = mLoginModel.isUserValid(username, password);
mLoginView.showLoginResult(isValid);
}
}
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最
了解Rails缓存如何工作的人可以真正帮助我。这是嵌套在Rails::Initializer.runblock中的代码:config.after_initializedoSomeClass.const_set'SOME_CONST','SOME_VAL'end现在,如果我运行script/server并发出请求,一切都很好。然而,在我的Rails应用程序的第二个请求中,一切都因单元化常量错误而变得糟糕。在生产模式下,我可以成功发出第二个请求,这意味着常量仍然存在。我已通过将以上内容更改为以下内容来解决问题:config.after_initializedorequire'some_cl
我经常迷上ruby的一件事是递归模式。例如,假设我有一个数组,它可能包含无限深度的数组作为元素。所以,例如:my_array=[1,[2,3,[4,5,[6,7]]]]我想创建一个方法,可以将数组展平为[1,2,3,4,5,6,7]。我知道.flatten可以完成这项工作,但这个问题是作为我经常遇到的递归问题的一个例子-因此我试图找到一个更可重用的解决方案。简而言之-我猜这种事情有一个标准模式,但我想不出任何特别优雅的东西。任何想法表示赞赏 最佳答案 递归是一种方法,它不依赖于语言。您在编写算法时要考虑两种情况:再次调用函数的情
我是一名决定学习Ruby和RubyonRails的ASP.NETMVC开发人员。我已经有所了解并在RoR上创建了一个网站。在ASP.NETMVC上开发,我一直使用三层架构:数据层、业务层和UI(或表示)层。尝试在RubyonRails应用程序中使用这种方法,我发现没有关于它的信息(或者也许我只是找不到它?)。也许有人可以建议我如何在RubyonRails上创建或使用三层架构?附言我使用ruby1.9.3和RubyonRails3.2.3。 最佳答案 我建议在制作RoR应用程序时遵循RubyonRails(RoR)风格。Rails
这应该是一个简单的问题,但我找不到任何相关信息。给定一个Ruby中的正则表达式,对于每个匹配项,我需要检索匹配的模式$1、$2,但我还需要匹配位置。我知道=~运算符为我提供了第一个匹配项的位置,而string.scan(/regex/)为我提供了所有匹配模式。如果可能,我需要在同一步骤中获得两个结果。 最佳答案 MatchDatastring.scan(regex)do$1#Patternatfirstposition$2#Patternatsecondposition$~.offset(1)#Startingandendingpo
我想开始使用“Sinatra”框架进行编码,但我找不到该框架的“MVC”模式。是“MVC-Sinatra”模式或框架吗? 最佳答案 您可能想查看Padrino这是一个围绕Sinatra构建的框架,可为您的项目提供更“类似Rails”的感觉,但没有那么多隐藏的魔法。这是使用Sinatra可以做什么的一个很好的例子。虽然如果您需要开始使用这很好,但我个人建议您将它用作学习工具,以对您来说最有意义的方式使用Sinatra构建您自己的应用程序。写一些测试/期望,写一些代码,通过测试-重复:)至于ORM,你还应该结帐Sequel其中(imho
有没有一种方法可以自动生成种子数据文件并创建种子数据,就像您在下面链接中的Laravel中看到的那样?LaravelDatabaseMigrations&Seed我在另一个应用程序上看到在Rails的db文件夹下创建了一些带有时间戳的文件,其中包含种子数据。创建它的好方法是什么? 最佳答案 我建议你使用Fabrication的组合gem和Faker.Fabrication允许您编写一个模式来构建您的对象,而Faker为您提供虚假数据,如姓名、电子邮件、电话号码等。这是制造商的样子:Fabricator(:user)dousernam