• Android进阶:6、使用okhttp下载图片


    因为在网上找到的资源不只是一些字符串或是一些二进制数字,还是有的是可以肉眼看见的图片,那么我们在发送网络请求时拿到的是一张图片该如何显示在组件上面呢?

    这就要使用到okhttp的请求返回结果的一个属性(byteStream):

    1. var input = result.byteStream()
    2. bitmap=BitmapFactory.decodeStream(input)
    3. myhanlder.sendEmptyMessage(3)

    结合之前学习的通过okhttp发送get异步请求方式,可以写出这样的代码:

    1. OkhttpUtils.get().doGet(url,object :NetWorkCallBack{
    2. override fun onSuccess(result: ResponseBody) {
    3. //TODO("Not yet implemented")
    4. var input = result.byteStream()
    5. bitmap=BitmapFactory.decodeStream(input)
    6. myhanlder.sendEmptyMessage(3)
    7. }
    8. override fun onFail(msg: String) {
    9. //TODO("Not yet implemented")
    10. Toast.makeText(this@Okhttp,"请求失败",Toast.LENGTH_SHORT).show()
    11. }
    12. })

    这里的OkhttpUtils是经过了封装的类,所以才有两个结果状态函数(onSuccess和onFail):

    1、请求的封装

    之前使用这个框架的时候我使用了很多的方法,有get、json、表单,现在又有一个图片的下载,所以我就直接讲这些请求方法封装在了一个工具类里面,这里面是这样写的:

    1. package com.example.hanlderpractice.service
    2. import com.example.hanlderpractice.entity.NetWorkCallBack
    3. import okhttp3.*
    4. import okhttp3.MediaType.Companion.toMediaType
    5. import okhttp3.RequestBody.Companion.toRequestBody
    6. import org.json.JSONObject
    7. import java.io.IOException
    8. import java.util.concurrent.TimeUnit
    9. class OkhttpUtils public constructor() {
    10. companion object {
    11. private var instance: OkhttpUtils? = null
    12. get() {
    13. if (field == null) {
    14. field = OkhttpUtils()
    15. }
    16. return field
    17. }
    18. fun get(): OkhttpUtils {
    19. return instance!!
    20. }
    21. }
    22. private var client:OkHttpClient?=null
    23. init {
    24. client=OkHttpClient.Builder()
    25. .readTimeout(5,TimeUnit.SECONDS)
    26. .connectTimeout(5,TimeUnit.SECONDS)
    27. .build()
    28. }
    29. fun doGet(url:String,netWorkCallBack: NetWorkCallBack):Unit{
    30. var request=Request.Builder()
    31. .url(url)
    32. .build()
    33. client?.newCall(request)?.enqueue(object :Callback{
    34. override fun onFailure(call: Call, e: IOException) {
    35. //TODO("Not yet implemented")
    36. netWorkCallBack.onFail("请求失败")
    37. }
    38. override fun onResponse(call: Call, response: Response) {
    39. //TODO("Not yet implemented")
    40. netWorkCallBack.onSuccess(response.body!!)
    41. }
    42. })
    43. }
    44. fun doJson(url:String,jsonobject:JSONObject,netWorkCallBack: NetWorkCallBack):Unit{
    45. var jsonstr=jsonobject.toString().toRequestBody("application/json;charset=utf-8".toMediaType())
    46. var request: Request=Request.Builder()
    47. .url(url)
    48. .post(jsonstr)
    49. .build()
    50. client?.newCall(request)?.enqueue(object :Callback{
    51. override fun onFailure(call: Call, e: IOException) {
    52. //TODO("Not yet implemented")
    53. netWorkCallBack.onFail("请求失败")
    54. }
    55. override fun onResponse(call: Call, response: Response) {
    56. //TODO("Not yet implemented")
    57. netWorkCallBack.onSuccess(response.body!!)
    58. }
    59. })
    60. }
    61. fun doForm(url:String,formBody:FormBody,netWorkCallBack: NetWorkCallBack):Unit{
    62. var request: Request=Request.Builder()
    63. .url(url)
    64. .post(formBody)
    65. .build()
    66. client?.newCall(request)?.enqueue(object :Callback{
    67. override fun onFailure(call: Call, e: IOException) {
    68. //TODO("Not yet implemented")
    69. netWorkCallBack.onFail("请求失败")
    70. }
    71. override fun onResponse(call: Call, response: Response) {
    72. //TODO("Not yet implemented")
    73. netWorkCallBack.onSuccess(response.body!!)
    74. }
    75. })
    76. }
    77. }

    通过派生对象来重写这个工具类的get方法,使他只能够实例化一次,也就是常见的单例模式;

    还有一个对象就是用来存储请求到的数据的接口,所以这两个返回结果状态函数是这个接口里面的:

    1. package com.example.hanlderpractice.entity
    2. import okhttp3.ResponseBody
    3. interface NetWorkCallBack {
    4. public fun onSuccess(result:ResponseBody)
    5. public fun onFail(msg:String)
    6. }

    这是一个接口,我在调用了发送请求的方法之后就会拿到一个请求结果对象,于是我就将这个结果(工具类请求到的数据)丢在这个接口里面,我就可以再别的地方(调用工具类请求方法的类)拿到了:

    1. var url="http://119.img.pp.sohu.com/images/blog/2007/9/28/18/15/115e6595d75.jpg"
    2. OkhttpUtils.get().doGet(url,object :NetWorkCallBack{
    3. override fun onSuccess(result: ResponseBody) {
    4. //TODO("Not yet implemented")
    5. var input = result.byteStream()
    6. bitmap=BitmapFactory.decodeStream(input)
    7. myhanlder.sendEmptyMessage(3)
    8. }

    就比如这里,这里使用了工具类里面的doGEt方法,而且第二个参数就是这个接口,他的成功函数里面的参数就是请求到的数据体,这时候直接创建他的字节流然后转化成bitmap对象设置到imageview组件上就可以实现图片的展示了,当然如果不是发送很多请求的话也可以直接发送get请求不用封装,这里是一样的效果的,到时候拿到数据体的字节流就可以转化为bitmap对象了;

    其他的post表单请求可以同样的方法,通过这个接口来返回这个数据体;

    2、设置图片

    1. //创建弱引用handler
    2. private class MyHanlder(var wk: WeakReference): Handler(Looper.getMainLooper()){
    3. override fun handleMessage(msg: Message) {
    4. super.handleMessage(msg)
    5. wk.get()?.run {
    6. if(msg.what==1){
    7. Toast.makeText(this,"登录成功", Toast.LENGTH_SHORT).show()
    8. var intent: Intent = Intent(this,MainActivity::class.java)
    9. startActivity(intent)
    10. }else if(msg.what==2){
    11. }
    12. else if(msg.what==3){
    13. iv_header.setImageBitmap(bitmap);
    14. }else{
    15. Toast.makeText(this,"登录失败", Toast.LENGTH_SHORT).show()
    16. }
    17. }
    18. }
    19. }

    同样的,发送请求只能在子线程,而更新界面需要发送消息,这里就是what=3的时候的消息,于是将这个通过网络数据构建成的bitmap设置到组件上面;

    无论是哪一张图片,这里是样例,也可以是其他的照片,可以使用同样的方式来显示到组件上;

     

  • 相关阅读:
    【微信小程序】WXML模板语法
    做对这 6 件事,提高软件测试的能力?
    如何将本地项目上传到GitHub
    设计一个缓存策略,动态缓存热点数据
    [附源码]计算机毕业设计springboot基于JAVA技术的旅游信息交互系统
    关于修改了mysql的my_conf文件之后,不能生效问题
    手动导出URL 证书
    GaussDB SQL基础语法-变量&常量
    ​LeetCode解法汇总2525. 根据规则将箱子分类
    rosbag遍历数据出错:(unicode error) 'utf-8' codec can't decode byte 0xcd in position 31: invalid continuation byte
  • 原文地址:https://blog.csdn.net/aiwanchengxu/article/details/128028132