淺談Retrofit2+Rxjava2

来源:http://www.cnblogs.com/shenchanghui/archive/2017/07/17/7197677.html
-Advertisement-
Play Games

近幾年,Retrofit猶如燎原之火搬席卷了整個Android界。要是不懂Retrofit,簡直不好意思出門。。。由於近幾個項目都沒用到Retrofit,無奈只能業餘時間自己擼一下,寫的不好的地方,還請不吝賜教。要集成retrofit,在app的bild.gradle中添加庫以來就可以: 如果需要集 ...


近幾年,Retrofit猶如燎原之火搬席卷了整個Android界。要是不懂Retrofit,簡直不好意思出門。。。
由於近幾個項目都沒用到Retrofit,無奈只能業餘時間自己擼一下,寫的不好的地方,還請不吝賜教。
要集成retrofit,在app的bild.gradle中添加庫以來就可以:

compile 'com.squareup.retrofit2:retrofit:2.3.0'

如果需要集成json解析,還需要添加庫:

compile 'com.squareup.retrofit2:converter-gson:2.3.0'

如果還需要集成rxjava,還需要添加庫:

compile 'com.squareup.retrofit2:adapter-rxjava2:2.3.0'
compile 'io.reactivex.rxjava2:rxandroid:2.0.1'

添加庫完畢,接下來需要定義請求介面:

public interface TestService {
    /**
     * 獲取新聞  使用rxjava
     * @return
     */
    @POST(AppConstance.NEWS_URL)
    Observable<NewsBean> getNewsWithRxJava(@Query("key") String key, @Query("type") String type);

    /**
     * 獲取新聞  不使用rxjava
     * @return
     */
    @POST(AppConstance.NEWS_URL)
    Call<ResponseBody> getNewsWithoutRxJava(@Query("key") String key, @Query("type") String type);
}

這個介面是我在聚合數據申請的測試介面,我將其分為兩種情況:使用rajava、不使用rxjava。
正常使用中,都會將Retrofit進行封裝,我在這裡將其簡單的封裝:

public class RetrofitUtil {
    private volatile static RetrofitUtil sInstance;
    private Retrofit mRetrofit;
    private TestService mTestService;
    private RetrofitUtil(){
        mRetrofit = new Retrofit.Builder()
                .baseUrl(AppConstance.APP_URL)
                .addConverterFactory(GsonConverterFactory.create())
                .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
                .build();
        mTestService = mRetrofit.create(TestService.class);
    }
    public static RetrofitUtil getInstance(){
        if (sInstance == null){
            synchronized(RetrofitUtil.class){
                if (sInstance == null){
                    sInstance = new RetrofitUtil();
                }
            }
        }
        return sInstance;
    }
    public TestService getTestService(){
        return mTestService;
    }
}

萬事具備,現在開始在Activity中測試Retrofit。
xml文件:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="org.shenchanghui.retrofit2withrxjava2_demo.MainActivity">


    <Button
        android:id="@+id/btn_get_news_with_rx_java"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentEnd="true"
        android:layout_alignParentLeft="true"
        android:layout_alignParentRight="true"
        android:layout_alignParentStart="true"
        android:layout_alignParentTop="true"
        android:text="獲取新聞(使用rxjava)" />

    <Button
        android:id="@+id/btn_get_news_without_rx_java"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentEnd="true"
        android:layout_alignParentLeft="true"
        android:layout_alignParentRight="true"
        android:layout_alignParentStart="true"
        android:layout_below="@+id/btn_get_news_with_rx_java"
        android:text="獲取新聞(不使用Rxjava)" />
</RelativeLayout>

在Activity中請求數據,首先,不使用rajava:

 findViewById(R.id.btn_get_news_without_rx_java).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                long time1 = System.currentTimeMillis();
                Call<ResponseBody> call = RetrofitUtil.getInstance().getTestService()
                        .getNewsWithoutRxJava("8bf17cf1c321723f060d5dc5c4da871a", "top");
                long time2 = System.currentTimeMillis();
                Log.e("MainActivity", "請求耗時:" + (time2 - time1) + "ms");
                call.enqueue(new Callback<ResponseBody>() {
                    @Override
                    public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response) {
                        try {
                            String result = response.body().string();
                            Toast.makeText(MainActivity.this, result, Toast.LENGTH_SHORT).show();
                            Log.e("MainActivity", "Thread.currentThread():" + Thread.currentThread());
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                    }

                    @Override
                    public void onFailure(Call<ResponseBody> call, Throwable t) {
                        Toast.makeText(MainActivity.this, t.toString(), Toast.LENGTH_SHORT).show();
                    }
                });
            }
        });

運行項目,成功獲取返回的json字元串。
創建實體類,用GsonFormat插件將獲取的json字元串生成實體類,用以解析json。
實體類:

public class NewsBean {

    private String reason;
    private ResultBean result;
    private int error_code;

    @Override
    public String toString() {
        return "NewsBean{" +
                "reason='" + reason + '\'' +
                ", result=" + result +
                ", error_code=" + error_code +
                '}';
    }

    public String getReason() {
        return reason;
    }

    public void setReason(String reason) {
        this.reason = reason;
    }

    public ResultBean getResult() {
        return result;
    }

    public void setResult(ResultBean result) {
        this.result = result;
    }

    public int getError_code() {
        return error_code;
    }

    public void setError_code(int error_code) {
        this.error_code = error_code;
    }

    public static class ResultBean {

        private String stat;
        private List<DataBean> data;

        @Override
        public String toString() {
            return "ResultBean{" +
                    "stat='" + stat + '\'' +
                    ", data=" + data +
                    '}';
        }

        public String getStat() {
            return stat;
        }

        public void setStat(String stat) {
            this.stat = stat;
        }

        public List<DataBean> getData() {
            return data;
        }

        public void setData(List<DataBean> data) {
            this.data = data;
        }

        public static class DataBean {

            private String uniquekey;
            private String title;
            private String date;
            private String category;
            private String author_name;
            private String url;
            private String thumbnail_pic_s;
            private String thumbnail_pic_s02;
            private String thumbnail_pic_s03;

            @Override
            public String toString() {
                return "DataBean{" +
                        "uniquekey='" + uniquekey + '\'' +
                        ", title='" + title + '\'' +
                        ", date='" + date + '\'' +
                        ", category='" + category + '\'' +
                        ", author_name='" + author_name + '\'' +
                        ", url='" + url + '\'' +
                        ", thumbnail_pic_s='" + thumbnail_pic_s + '\'' +
                        ", thumbnail_pic_s02='" + thumbnail_pic_s02 + '\'' +
                        ", thumbnail_pic_s03='" + thumbnail_pic_s03 + '\'' +
                        '}';
            }

            public String getUniquekey() {
                return uniquekey;
            }

            public void setUniquekey(String uniquekey) {
                this.uniquekey = uniquekey;
            }

            public String getTitle() {
                return title;
            }

            public void setTitle(String title) {
                this.title = title;
            }

            public String getDate() {
                return date;
            }

            public void setDate(String date) {
                this.date = date;
            }

            public String getCategory() {
                return category;
            }

            public void setCategory(String category) {
                this.category = category;
            }

            public String getAuthor_name() {
                return author_name;
            }

            public void setAuthor_name(String author_name) {
                this.author_name = author_name;
            }

            public String getUrl() {
                return url;
            }

            public void setUrl(String url) {
                this.url = url;
            }

            public String getThumbnail_pic_s() {
                return thumbnail_pic_s;
            }

            public void setThumbnail_pic_s(String thumbnail_pic_s) {
                this.thumbnail_pic_s = thumbnail_pic_s;
            }

            public String getThumbnail_pic_s02() {
                return thumbnail_pic_s02;
            }

            public void setThumbnail_pic_s02(String thumbnail_pic_s02) {
                this.thumbnail_pic_s02 = thumbnail_pic_s02;
            }

            public String getThumbnail_pic_s03() {
                return thumbnail_pic_s03;
            }

            public void setThumbnail_pic_s03(String thumbnail_pic_s03) {
                this.thumbnail_pic_s03 = thumbnail_pic_s03;
            }
        }
    }
}

接下里,使用rxjava,請求數據(自動返回解析好的數據):

findViewById(R.id.btn_get_news_with_rx_java).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                RetrofitUtil.getInstance().getTestService()
                        .getNewsWithRxJava("8bf17cf1c321723f060d5dc5c4da871a", "top")
                        .subscribeOn(Schedulers.io())
                        .observeOn(AndroidSchedulers.mainThread())
                        .subscribe(new Observer<NewsBean>() {
                            private Disposable mDisposable;

                            @Override
                            public void onSubscribe(Disposable d) {
                                mDisposable = d;
                            }

                            @Override
                            public void onNext(NewsBean value) {
                                Toast.makeText(MainActivity.this, value.toString(), Toast.LENGTH_SHORT).show();
                                mDisposable.dispose();//註銷
                            }

                            @Override
                            public void onError(Throwable e) {
                                Toast.makeText(MainActivity.this, e.toString(), Toast.LENGTH_SHORT).show();
                                mDisposable.dispose();//註銷
                            }

                            @Override
                            public void onComplete() {

                            }
                        });

            }
        });

運行程式,成功獲取請求結果。

擼碼暫告一段落,休息下。。。

源碼下載地址:https://github.com/shench5612390/Retrofit2WithRxjava2_Demo/tree/master


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

-Advertisement-
Play Games
更多相關文章
  • Spring Boot 熱部署 實際開發中,修改某個頁面數據或邏輯功能都需要重啟應用。這無形中降低了開發效率,所以使用熱部署是十分必要的。 什麼是熱部署? 應用啟動後會把編譯好的Class文件載入的虛擬機中,正常情況下在項目修改了源文件是需要全部重新編譯並重新載入(需要重啟應用)。而熱部署就是監聽C ...
  • 1 概述 1 概述 軟體架構是一門學問,並且是一門很深邃的學問,從本篇文章開始,我們就來聊聊架構,所用到的主流語言為.NET、Java和php。本篇文章作為架構的開篇文章,主要從廣度上敘述軟體架構的發展與演變,從軟體架構系列第二篇文章開始,將結合具體的產品或項目實例,來與大家分享架構。 本篇文章先簡 ...
  • 使用docker也有段時間了,寫了不少文章與總結,下麵把它整理個目錄出來,方便大家去學習與檢索! docker~學習筆記索引 docker~linux下的部署和基本命令(2017-04-07 22:47) docker~docker-machine的介紹(2017-04-12 12:02) dock ...
  • HP-Socket 是一套通用的高性能 TCP/UDP/HTTP 通信框架,包含服務端組件、客戶端組件和 Agent 組件,廣泛適用於各種不同應用場景的 TCP/UDP/HTTP 通信系統,提供 C/C++、C#、Delphi、E(易語言)、Java、Python 等編程語言介面。HP-Socket... ...
  • 代碼: #import "ViewController.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; // Do any a ...
  • 目前將項目中的leancloud的即時通訊改為環信的即時通訊。當引入easeui的時候 出現方法數超過上限的問題。 搜索一下問題,解決方法很簡單。 這裡簡單記錄一下,順序記錄一下此解決方案導致的另一個問題。 一、解決方法數超過64k的問題 問題描述: 解決方案: 1、app目錄下 build.gra ...
  • 思路 思路很簡單,對模型數據操作或則控制界面顯示 先看下json部分數據 這種數據對應的一般都是個tableView, 然後根據章節分開,最終界面如下: 分析 這裡採用UITableViewStylePlain樣式,chapterDtoList對應章,subChapterList對應節。章的話我們使 ...
  • `MBProgressHUD`是一個顯示提示視窗的三方庫,常用於用戶交互、後臺耗時操作等的提示。通過顯示一個提示框,通知用戶操作或任務的執行狀態;同時,利用動畫效果,降低用戶等待的焦慮心理,增強用戶體驗。 本篇文章從源碼角度來看一下 是如何實現的,所用的知識都是比較基礎的,不過還是值得我們學習一下。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...