溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android項(xiàng)目多服務(wù)端接口適配的方法有哪些

發(fā)布時(shí)間:2020-08-12 15:22:25 來源:億速云 閱讀:209 作者:小新 欄目:開發(fā)技術(shù)

小編給大家分享一下Android項(xiàng)目多服務(wù)端接口適配的方法有哪些,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

現(xiàn)狀

Android項(xiàng)目如果是多服務(wù)端接口時(shí),一般怎么弄呢?

方法1:服務(wù)器地址放在Header中

把服務(wù)器地址放在接口Header中,然后通過攔截器來動態(tài)修改請求地址而實(shí)現(xiàn)的。除了默認(rèn)服務(wù)器的接口,其它都要加一個(gè)Header,有點(diǎn)麻煩??雌饋硪膊凰?,不簡潔。

interface ApiHeaderCase {
  /************************** server A ****************************/
  @Headers("host:$SERVER_HOST_A")
  @GET("user/loginWithScanCode")
  fun aMethod1(@Query("id") id: Int): Observable<ResponseBody>

  /************************** server B ****************************/
  @Headers("host:$SERVER_HOST_B")
  @GET("user/loginWithScanCode")
  fun bMethod1(@Query("id") id: Int): Observable<ResponseBody>
}

方法2:多套服務(wù)類,實(shí)例化為多個(gè)對象,準(zhǔn)確查找接口歸屬服務(wù)

定義多個(gè)類,每個(gè)類定義一套服務(wù)接口。然后分別實(shí)例化為多個(gè)對象,再使用準(zhǔn)確的對象來調(diào)用接口。這種方法運(yùn)行效率是最高的,但是在開發(fā)時(shí),可能無法快速知道接口歸屬與哪個(gè)服務(wù),需要查看代碼才能準(zhǔn)確知曉,可以說是少了代碼提示能力。

interface ApiA {
  @GET("user/loginWithScanCode")
  fun methodA(@Query("id") id: Int): Observable<ResponseBody>
}

interface ApiB {
  @GET("user/loginWithScanCode")
  fun methodB(@Query("id") id: Int): Observable<ResponseBody>
}

方法3:全寫在一起,實(shí)例化為多個(gè)對象,準(zhǔn)確調(diào)用方法

把所有接口都寫在一個(gè)類中,然后根據(jù)服務(wù)地址分別實(shí)例化為多個(gè)對象。再準(zhǔn)確調(diào)用方法,為了保證準(zhǔn)確調(diào)用方法,可以給每個(gè)接口加個(gè)服務(wù)名的前綴,以減少方法調(diào)錯(cuò)的問題。

interface ApiAllInOne {
  /************************** server A ****************************/
  @GET("user/loginWithScanCode")
  fun aMethod1(@Query("id") id: Int): Observable<ResponseBody>

  /************************** server B ****************************/
  @GET("user/loginWithScanCode")
  fun bMethod1(@Query("id") id: Int): Observable<ResponseBody>
}

const val SERVER_HOST_A = "https://www.a.com/"
const val SERVER_HOST_B = "https://www.b.com/"
fun getApi(retrofit: Retrofit, host: String): ApiAllInOne {
  return retrofit.newBuilder()
      .baseUrl(host).build()
      .create(ApiAllInOne::class.java)
}

fun showNomalUseCase(retrofit: Retrofit) {
  val apiA = getApi(retrofit, SERVER_HOST_A)//save as single instance for repeated usage
  apiA.aMethod1(1).subscribe()
  apiA.bMethod1(1).subscribe()//invalid usage, but no compile error

  val apiB = getApi(retrofit, SERVER_HOST_B)
  apiB.bMethod1(1).subscribe()
  apiB.aMethod1(1).subscribe()//invalid usage, but no compile error
}

有更簡單的方法嗎?

當(dāng)然有了,而且超方便!

定義接口

(建議)在一個(gè)KT文件中定義所有接口,方便查找和維護(hù)。

interface ApiHolder : ApiA, ApiB

  @BaseUrl("https://www.a.com/")
  interface ApiA {
    @GET("user/loginWithScanCode")
    fun methodA(@Query("id") id: Int): Observable<ResponseBody>
  }

  @BaseUrl("https://www.b.com/")
  interface ApiB {
    @GET("user/loginWithScanCode")
    fun methodB(@Query("id") id: Int): Observable<ResponseBody>
  }

建工具類

一般都需要個(gè)工具類的,方便配置攔截器等。如果沒有自定義的需求,也可以直接實(shí)例化來用。

可以重寫invokeApi方法,全局給每個(gè)Observable設(shè)定線程。

class ApiUtil : ApiHolderUtil<ApiHolder>(ApiHolder::class) {
  companion object {
    val apiUtil = ApiUtil()
    val api = apiUtil.api
  }

  override fun invokeApi(api: Any, method: Method, args: Array<*>&#63;): Any {
    val observable = super.invokeApi(api, method, args) as Observable<*>
    return observable.subscribeOn(Schedulers.io())
        .observeOn(AndroidSchedulers.mainThread())
  }
}

動態(tài)更新服務(wù)地址

還可以動態(tài)更新服務(wù)地址,比如實(shí)現(xiàn)測試服務(wù)和正式服務(wù)間切換。

//update api baseUrl when needed
  apiUtil.updateApi(ApiA::class, https://www.a2.com/)

調(diào)用接口

api.methodA(1).subscribe()
  api.methodB(1).subscribe()

引入依賴

dependencies {
  implementation 'com.github.DonaldDu:ApiHolder:x.x.x'//JitPack version
}

該項(xiàng)目使用的三方庫

  • OkHttp3
  • Retrofit2
  • rxjava3(可以修改為rxjava2)
api 'com.squareup.okhttp3:okhttp:4.7.2'
  api "com.squareup.retrofit2:retrofit:2.9.0"
  api "com.squareup.retrofit2:converter-gson:2.9.0"
  api "com.squareup.retrofit2:adapter-rxjava3:2.9.0"
  api 'io.reactivex.rxjava3:rxandroid:3.0.0'

其它說明

rxjava3 ->rxjava2

可以根據(jù)需要調(diào)整為rxjava2,建議用最新的。

//重寫ApiHolderUtil如下方法,RxJava3CallAdapterFactory ->RxJava2CallAdapterFactory即可。
  protected open fun getRetrofit(client: OkHttpClient): Retrofit {
    return Retrofit.Builder()
        .validateEagerly(validateEagerly)
        .addConverterFactory(getGsonConverterFactory())
        .addCallAdapterFactory(RxJava3CallAdapterFactory.create())
        .baseUrl("http://www.demo.com/")
        .client(client)
        .build()
  }

Timeout

可以給每套服務(wù)設(shè)置不同的超時(shí)

@BaseUrl("https://www.b.com/")
@Timeout(read = 100, timeUnit = TimeUnit.SECONDS)
interface ApiB {
  @GET("user/loginWithScanCode")
  fun methodB(@Query("id") id: Int): Observable<ResponseBody>
}

以上是Android項(xiàng)目多服務(wù)端接口適配的方法有哪些的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI