我是android的新手,这是第一次尝试在代码中使用MVP模式...据我所知,到目前为止,我的观点应与演示者和模型演示者进行交流>然后演示者再次交谈以查看..我希望我是对的!如下面的简单代码示例所示,我试图将模型中的结果值返回给演示者,然后将在演示者中使用该结果值来确定应在视图中调用哪种方法。.我有两个问题希望对您有所帮助。 1)enqueue方法正在异步工作,并且结果值将始终返回空或失败或其他任何结果。.因为它是单独工作的。当我尝试使用execute方法时,我正面临NetworkOnMainThreadException错误...所以我该怎么做才对呢? 2)这是使用MVP模式的正确方法吗?
这是SignupContract类
public class SignupContract {
public interface IView{
void signupSuccess();
void signupFailed(String message);
}
public interface IPresenter{
void signup(UserProfile userProfile);
}
public interface IModel{
String signup(UserProfile userProfile);
}
}
这是查看代码。
public class SignupActivity extends AppCompatActivity implements SignupContract.IView {
//some code
@Override
protected void onCreate(Bundle savedInstanceState) {
//some code
createAccountBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//some code
presenter.signup(userProfile);
}
});
}
@Override
public void signupSuccess() {
/*AppUtils.dismissLoadingDialog(SignupActivity.this,"","");*/
Intent intent = new Intent(SignupActivity.this, SigninActivity.class);
startActivity(intent);
finish();
}
@Override
public void signupFailed(String message) {
/*AppUtils.dismissLoadingDialog(SignupActivity.this,"","");*/
AppUtils.showErrorMessage(SignupActivity.this, message);
}
这是主持人
public class SignupPresenter implements SignupContract.IPresenter {
SignupContract.IView view;
SignupContract.IModel model;
public SignupPresenter(SignupContract.IView view){
model = new SignupModel();
this.view = view;
}
@Override
public void signup(UserProfile userProfile) {
userProfile = UserProfileCleaner.clean(userProfile, "signup");
UserProfileDTO dto = new UserProfileDTO();
String validationMessage = dto.validateUserProfile(userProfile, "signup");
if(validationMessage != null && !validationMessage.equals("")){
view.signupFailed(validationMessage);
}else{
String signupResult = model.signup(userProfile);
if(signupResult.equals("success")){
view.signupSuccess();
}else {
view.signupFailed(signupResult);
}
}
}
}
这是模型类
public class SignupModel implements SignupContract.IModel {
private String TAG = "SignupModel";
private String result = "";
@Override
public String signup(UserProfile userProfile) {
final Context context = DKApp.getContext();
ServiceWrapper serviceWrapper = new ServiceWrapper(null);
Call<SignupResponse> userSignUpCall = serviceWrapper.userSignUpCall(userProfile.getUser().getUsername(),
userProfile.getUser().getPassword(),userProfile.getPhoneNumber(), userProfile.getEmailAddress(),
userProfile.getFullName());
userSignUpCall.enqueue(new Callback<SignupResponse>() {
@Override
public void onResponse(Call<SignupResponse> call, Response<SignupResponse> response) {
if( response.body() != null && response.isSuccessful() ){
Log.e(TAG,response.body().toString());
if(response.body().getStatus() == 1){
//some code
result = "success";
}else{
result = response.body().getMessage();
}
}else{
result = context.getResources().getString(R.string.request_failed);
}
}
@Override
public void onFailure(Call<SignupResponse> call, Throwable t) {
Log.e(TAG, "Failure : " + t.toString());
result = context.getResources().getString(R.string.request_failed);
}
});
return result;
}
}
答案 0 :(得分:0)
您正在模型中进行异步调用,这可能需要100毫秒或2-4秒,因此像String signupResult = model.signup(userProfile);
这样从其获取signupResult,这是错误的。
您需要进行的更改:
1)将onComplete方法添加到IPresenter并更改IModel
public interface IPresenter{
void signup(UserProfile userProfile);
//add
void onComplete(String signUpresult);
}
public interface IModel{
//changed
void signup(UserProfile userProfile);
}
2)在您的SignupPresenter中,将演示者的实例传递给模型
public class SignupPresenter implements SignupContract.IPresenter {
..
public SignupPresenter(SignupContract.IView view){
model = new SignupModel(this);
this.view = view;
}
...
@Overrides
public void onComplete(String signupResult){
if(signupResult.equals("success")){
view.signupSuccess();
}else {
view.signupFailed(signupResult);
}
}
...
}
3)在您的SignupModel中,一旦获得结果,请从演示者调用onComplete(// result)
public class SignupModel implements SignupContract.IModel {
SignupPresenter presenter;
public SignupModel(SignupPresenter presenter){
this.presenter = presenter
}
@Override
public void signup(UserProfile userProfile) {
...
userSignUpCall.enqueue(new Callback<SignupResponse>() {
@Override
public void onResponse(Call<SignupResponse> call, Response<SignupResponse> response) {
if(response.body() != null && response.isSuccessful() ){
if(response.body().getStatus() == 1){
//some code
presenter.onComplete("success");
}else{
presenter.onComplete(response.body().getMessage());
}
}else{
presenter.onComplete(context.getResources().getString(R.string.request_failed));
}
}
@Override
public void onFailure(Call<SignupResponse> call, Throwable t) {
Log.e(TAG, "Failure : " + t.toString());
presenter.onComplete(context.getResources().getString(R.string.request_failed));
}
});
}
}
需要::在调用注册时显示进度对话框,并在SignupPresenter中的onComplete上取消该对话框。
您的理解还不错,您也知道该模型也应该与演示者对话。要了解有关MVP设计模式的更多信息,请阅读this
答案 1 :(得分:0)
在MVP模式中,应将视图设为虚拟。演示者总是告诉视图该怎么做。 这是一个示例:
In View =>
presenter.login(userName, password)
在演示者中=>
fun login(userName: String, password: String) {
//login logic
if(success) view.showLoginSuccess()
else view.showLoginError()
}
这是MVP模式的非常简短的解释。关于您的问题,您无法在主线程上进行请求。幸运的是Retrofit具有线程管理系统。但是,您不应该在这样的模型中使用它,而应该在演示者中使用它,或者以更正确的方式搜索干净的体系结构文章。如果您在演示者中使用它,您应该做类似的事情
userSignUpCall.enqueue(new Callback<SignupResponse>() {
@Override
public void onResponse(Call<SignupResponse> call, Response<SignupResponse> response) {
if( response.body() != null && response.isSuccessful() ){
Log.e(TAG,response.body().toString());
if(response.body().getStatus() == 1){
//some code
view.showMessage("success");
}else{
view.showMessage(response.body().getMessage());
}
}else{
view.showError(context.getResources().getString(R.string.request_failed));
}
}
@Override
public void onFailure(Call<SignupResponse> call, Throwable t) {
Log.e(TAG, "Failure : " + t.toString());
view.showError(context.getResources().getString(R.string.request_failed));
}
});