Android 网络访问框架retrofit2,okhttp3之简单封装,kotlin源码

本文章需要用到的引用

//okhttp
compile 'com.squareup.okhttp3:okhttp:3.8.0'
//retrofit
compile 'com.squareup.retrofit2:retrofit:2.3.0'
//rxjava
compile 'io.reactivex.rxjava2:rxjava:2.0.7'
//rxjava适配器,方便rxjava与retrofit的结合
compile 'com.jakewharton.retrofit:retrofit2-rxjava2-adapter:1.0.0'
//Log拦截器,方便DEBUG模式输出log信息
compile 'com.squareup.okhttp3:logging-interceptor:3.8.0'
//json转换器,方便将返回的数据转换为json格式
compile 'com.squareup.retrofit2:converter-gson:2.3.0'
//rxandroid
compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
compile 'com.github.bumptech.glide:glide:3.7.0'


定义:

1.基于retrofit2和okhttp3的网络访问简单封装。

基本封装类

RetrofitClient.kt
import android.content.Context
import android.util.Log
import java.io.File
import okhttp3.Cache
import com.jakewharton.retrofit2.adapter.rxjava2.RxJava2CallAdapterFactory
import retrofit2.converter.gson.GsonConverterFactory
import retrofit2.Retrofit
import okhttp3.logging.HttpLoggingInterceptor
import okhttp3.OkHttpClient
import java.util.concurrent.TimeUnit

/**
 * @author Created by qlang on 2017/7/14.
 */
class RetrofitClient private constructor(context: Context, baseUrl: String) {
    var httpCacheDirectory: File? = null
    val mContext: Context = context
    var cache: Cache? = null
    var okHttpClient: OkHttpClient? = null
    var retrofit: Retrofit? = null
    val DEFAULT_TIMEOUT: Long = 20
    val url = baseUrl

    init {
        //缓存地址
        httpCacheDirectory = httpCacheDirectory ?: File(mContext.cacheDir, "app_cache")
        
        try {
            cache = cache ?: Cache(httpCacheDirectory, 10 * 1024 * 1024)
        } catch (e: Exception) {
            Log.e("OKHttp", "Could not create http cache", e)
        }
        //okhttp创建了
        okHttpClient = OkHttpClient.Builder()
                .addNetworkInterceptor(HttpLoggingInterceptor().setLevel(HttpLoggingInterceptor.Level.BODY))
                .cache(cache)
                .addInterceptor(CacheInterceptor(context))
                .addNetworkInterceptor(CacheInterceptor(context))
                .connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS)
                .writeTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS)
                .build()
        //retrofit创建了
        retrofit = Retrofit.Builder()
                .client(okHttpClient)
                .addConverterFactory(GsonConverterFactory.create())
                .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
                .baseUrl(url)
                .build()
    }

    companion object {
        @Volatile
        var instance: RetrofitClient? = null
        var context: Context? = null

        fun init(context: Context) {
            this.context = context
        }

        fun getInstance(baseUrl: String): RetrofitClient {
            return instance ?: synchronized(RetrofitClient::class) {
                    if (context == null) throw NullPointerException("context is null.Use init(content) in your Application")
                    RetrofitClient(context!!, baseUrl).also{ instance = it }
                }
            }
        }
    }

    fun <T> create(service: Class<T>?): T? {
        if (service == null) throw RuntimeException("Api service is null!")
        return retrofit?.create(service)
    }
}


缓存辅助类:

CacheInterceptor.kt

import android.content.Context
import android.util.Log
import com.ql.ufun.utils.NetworkUtils
import okhttp3.Interceptor
import okhttp3.Response
import okhttp3.CacheControl

/**
 * @author Created by qlang on 2017/7/14.
 */
class CacheInterceptor(context: Context) : Interceptor {
    val context = context
    override fun intercept(chain: Interceptor.Chain?): Response? {
        var request = chain?.request()
        if (NetworkUtils.isNetConneted(context)) {
            val response = chain?.proceed(request)
            // read from cache for 60 s
            val maxAge = 60
            val cacheControl = request?.cacheControl().toString()
            Log.e("CacheInterceptor", "6s load cahe" + cacheControl)
            return response?.newBuilder()?.removeHeader("Pragma")?.removeHeader("Cache-Control")?.header("Cache-Control", "public, max-age=" + maxAge)?.build()
        } else {
            Log.e("CacheInterceptor", " no network load cahe")
            request = request?.newBuilder()?.cacheControl(CacheControl.FORCE_CACHE)?.build()
            val response = chain?.proceed(request)
            //set cahe times is 3 days
            val maxStale = 60 * 60 * 24 * 3
            return response?.newBuilder()?.removeHeader("Pragma")?.removeHeader("Cache-Control")?.header("Cache-Control", "public, only-if-cached, max-stale=" + maxStale)?.build()
        }
    }

}

使用:

1.定义接口

ApiServer.kt
interface ApiServer {
    companion object {
        val BASE_URL: String get() = "http://route.xxx.com/"
    }

    @GET("xxxxxx") 
    fun getHomeTxtData(@Query("xxx") apiid: String, @Query("xxx") sign: String, @Query("currPage") page: String, @Query("maxResult") num: String): Observable<Xxxxx>

2.定义Base Model

BaseModel.kt
import com.ql.xxx.net.ApiServer
import com.ql.xxx.net.RetrofitClient
import io.reactivex.Observable
import io.reactivex.android.schedulers.AndroidSchedulers
import io.reactivex.schedulers.Schedulers

/**
 * @author Created by qlang on 2017/7/18.
 */
open class BaseModel {
    fun buildServer(): RetrofitClient {
        return RetrofitClient.getInstance(ApiServer.BASE_URL)
    }

    fun <T> Observable<T>.applySchedulers(): Observable<T> {
        return subscribeOn(Schedulers.io()).
                unsubscribeOn(Schedulers.io()).
                observeOn(AndroidSchedulers.mainThread())
    }
}

3.具体的Model

MainModel.kt
/**
 * @author Created by qlang on 2017/7/18.
 */
class MainModel : BaseModel() {

    fun loadTextData(index: Int): Observable<Xxxxx>? {
        val server = buildServer().create(ApiServer::class.java)
        return server?.getHomeTxtData(ApiServer.API_ID, ApiServer.API_SIGN, "$index", "20")?.applySchedulers()
    }
}

4.发起请求

val model: MainModel by lazy {
        MainModel()
    }
val observable: Observable<Xxxxx>? = model.loadTextData(index)
        observable?.subscribe({ bean: Xxxxx -> listener?.onSuccRespone(bean) },
                { err: Throwable -> listener?.onErrRespone(err) })


附:
基于Glide的图片加载简单封装

ImageLoader.kt
import android.content.Context
import android.graphics.Bitmap
import android.widget.ImageView
import com.bumptech.glide.Glide
import com.bumptech.glide.load.DecodeFormat
import com.bumptech.glide.load.engine.DiskCacheStrategy
import com.bumptech.glide.request.animation.GlideAnimation
import com.bumptech.glide.request.target.SimpleTarget
import com.ql.ufun.R

/**
 * @author Created by qlang on 2017/7/19.
 */
class ImageLoader {

    interface BitmapListener {
        fun onCall(bitmap: Bitmap?)
    }

    companion object {
        fun <T : ImageView> loadImg(context: Context, url: String, imageView: T?) {
            loadImg(context, url, imageView, R.mipmap.ic_default_img, R.mipmap.ic_default_img)
        }

        fun <T : ImageView> loadBigImg(context: Context, url: String, imageView: T?) {
            if (imageView == null) {
                throw IllegalArgumentException("argument error")
            }
            Glide.with(context).load(url)
                    .asBitmap()
                    .format(DecodeFormat.PREFER_ARGB_8888)
                    .diskCacheStrategy(DiskCacheStrategy.ALL)
                    .placeholder(R.mipmap.ic_default_img)
                    .error(R.mipmap.ic_default_img)
                    .into(imageView)
        }

        fun <T : ImageView> loadImg(context: Context, url: String, imageView: T?, defaultImageResId: Int, errImageResId: Int) {
            if (imageView == null) {
                throw IllegalArgumentException("argument error")
            }
            Glide.with(context).load(url)
                    .diskCacheStrategy(DiskCacheStrategy.ALL)
                    .placeholder(defaultImageResId)
                    .error(errImageResId)
                    .crossFade().into(imageView)
        }

        fun loadImg(context: Context, url: String, listener: BitmapListener?) {
            Glide.with(context).load(url)
                    .asBitmap()
                    .format(DecodeFormat.PREFER_ARGB_8888)
                    .diskCacheStrategy(DiskCacheStrategy.ALL)
                    .into(object : SimpleTarget<Bitmap>() {
                        override fun onResourceReady(resource: Bitmap?, glideAnimation: GlideAnimation<in Bitmap>?) {
                            listener?.onCall(resource)
                        }
                    })
        }
    }
}





  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值