使用準(zhǔn)備
-
build.gradle
中配置引用implementation 'com.squareup.retrofit2:retrofit:2.6.2' implementation 'com.google.code.gson:gson:2.8.2' implementation 'com.squareup.retrofit2:converter-gson:2.6.2'
需要導(dǎo)入的包除了
retrofit
外還需要導(dǎo)入gson
用來(lái)轉(zhuǎn)化json
數(shù)據(jù)蕴茴,導(dǎo)入converter-gson
來(lái)使gson
可以和retrofit
配套使用恭应。這里是三個(gè)包的git地址所踊,用的時(shí)候可以從這里引用最新的包和獲取文檔支持:
-
加入網(wǎng)絡(luò)權(quán)限
<uses-permission android:name="android.permission.INTERNET"></uses-permission>
建立數(shù)據(jù)模型
我們這里使用的是Gson解析,需要事先和接口約定好對(duì)應(yīng)的json
樣式,然后在本地寫(xiě)好對(duì)應(yīng)的數(shù)據(jù)模型曹质。這里以快遞100的返回?cái)?shù)據(jù)為例,可以構(gòu)造如下模型:
public class ExpressBean {
public String message;
public String nu;
public String ischeck;
public String com;
public String status;
public String condition;
public String state;
public List<DataBean> data;
public static class DataBean {
public String time;
public String context;
public String ftime;
}
}
在構(gòu)建本地?cái)?shù)據(jù)模型時(shí),我推薦一個(gè)好用的插件妇穴,可以一鍵轉(zhuǎn)化json
數(shù)據(jù)為java
類:
網(wǎng)絡(luò)接口設(shè)置
retrofit
的請(qǐng)求是通過(guò)接口加注解的形式來(lái)實(shí)現(xiàn)的,這里我寫(xiě)了個(gè)查詢圓通快遞的簡(jiǎn)單例子:
public interface ExpressQuestionService {
@GET("query?type=yuantong")
Call<ExpressBean> queryExpress(@Query("postid") String expressCode);
}
這個(gè)方法在調(diào)用后隶债,會(huì)拼接postid
到請(qǐng)求的后面腾它,組成query?type=yuantong&postid=expressCode
;
搭建mvp接口
MVP的第一步就是寫(xiě)接口,接口書(shū)寫(xiě)的過(guò)程也就是流程梳理的過(guò)程死讹,當(dāng)接口書(shū)寫(xiě)完后携狭,只需要依次填補(bǔ)各個(gè)實(shí)現(xiàn)類就完成了,我們這個(gè)小需求很簡(jiǎn)單回俐,所以接口也不復(fù)雜逛腿。
這里我把三個(gè) 接口放到一起稀并,方便調(diào)整。
public interface ExpressQueryContract {
interface View {
/**
* 綁定返回?cái)?shù)據(jù)到頁(yè)面
*/
void bindView(ExpressBean bean);
/**
* 顯示提示信息
*/
void showToast(String message);
}
interface Presenter {
/**
* 處理查詢邏輯
*/
void queryExpress(String expressCode);
}
interface Model {
/**
* 調(diào)用接口查詢信息
*/
void queryExpress(String expressCode, Callback<ExpressBean> callback);
}
}
實(shí)現(xiàn)MVP接口
接口定義好了单默,接下來(lái)依次寫(xiě)各個(gè)接口的實(shí)現(xiàn)類就好了
-
實(shí)現(xiàn)
Model
public class ExpressQueryModel implements ExpressQueryContract.Model { @Override public void queryExpress(String expressCode, Callback<ExpressBean> callback) { Retrofit retrofit = new Retrofit.Builder() .baseUrl("http://www.kuaidi100.com/") .addConverterFactory(GsonConverterFactory.create()) .build(); ExpressQuestionService service = retrofit.create(ExpressQuestionService.class); Call<ExpressBean> repos = service.queryExpress(expressCode); repos.enqueue(callback); } }
-
實(shí)現(xiàn)
Presenter
public class ExpressQueryPresenter implements ExpressQueryContract.Presenter { private ExpressQueryContract.View view; private ExpressQueryContract.Model model; public ExpressQueryPresenter(ExpressQueryContract.View view) { this.view = view; model = new ExpressQueryModel(); } @Override public void queryExpress(String expressCode) { model.queryExpress(expressCode, new Callback<ExpressBean>() { @Override public void onResponse(Call<ExpressBean> call, Response<ExpressBean> response) { if (response != null && response.body() != null) { view.bindView(response.body()); } else { onFailure(null, null); } } @Override public void onFailure(Call<ExpressBean> call, Throwable t) { if (t != null && t.getMessage() != null) { view.showToast(t.getMessage()); } else { view.showToast("槽糕,服務(wù)開(kāi)小差了,稍后再試吧"); } } }); } }
-
實(shí)現(xiàn)
View
public class ExpressQuestionActivity extends AppCompatActivity implements ExpressQueryContract.View { private Button button; private TextView textView; private EditText editText; private ExpressQueryContract.Presenter presenter; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_express_question); findView(); presenter = new ExpressQueryPresenter(this); button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { if (TextUtils.isEmpty(getExpressCode())) { showToast("單號(hào)不能為空"); } else { presenter.queryExpress(getExpressCode()); } } }); } private String getExpressCode() { String code = editText.getText().toString().trim(); return code; } private void findView() { button = findViewById(R.id.button_get); textView = findViewById(R.id.net_result); editText = findViewById(R.id.express_code); } @Override public void bindView(ExpressBean bean) { textView.setText(bean.toString()); } @Override public void showToast(String message) { Toast.makeText(this, message, Toast.LENGTH_LONG); } }
總體包結(jié)構(gòu)
可能遇到的問(wèn)題
-
IllegalArgumentException
運(yùn)行時(shí)出現(xiàn)如下異常信息:
java.lang.IllegalArgumentException: Unable to create converter for class ****
這類異常是由于沒(méi)有添加對(duì)應(yīng)的
ConverterFactory
造成的碘举,retrofit
必須配套對(duì)應(yīng)的類型解析器才能使用,具體的代碼就是下面的第三行語(yǔ)句Retrofit retrofit = new Retrofit.Builder() .baseUrl("http://www.kuaidi100.com/") .addConverterFactory(GsonConverterFactory.create()) .build();
-
CLEARTEXT
請(qǐng)求一直失敗搁廓,直接走
onFailure
回調(diào)引颈,錯(cuò)誤信息類似下面這樣:CLEARTEXT communication to www.kuaidi100.com not permitted by network security policy
這是由于android9.0開(kāi)始,對(duì)http請(qǐng)求的限制造成的境蜕,具體的解決方法可以參考這篇文章Android9.0 http網(wǎng)絡(luò)請(qǐng)求解決方案
總結(jié)
好了蝙场,結(jié)束。