Android 多线程编程实验_android UI线程

Android 多线程编程实验_android UI线程线程的基本用法Android的多线程编程与Java多线程编程基本是使用相同的语法,比如定义一个线程只需要新建一个类继承自Thread,重写父类的run()方法classMyThread:Thread(){overridefunrun(){//编写具体的逻辑}}启动这个线程也很简单,创建MyThread的实例,调用start()方法,这样run()方法中的代码就会在子线程中运行了MyThread().start().

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全家桶1年46,售后保障稳定

线程的基本用法

Android 的多线程编程与 Java 多线程编程基本是使用相同的语法,比如定义一个线程只需要新建一个类继承自 Thread,重写父类的 run() 方法

class MyThread : Thread() { 
   
    override fun run() { 
   
        // 编写具体的逻辑
    }
}

Jetbrains全家桶1年46,售后保障稳定

启动这个线程也很简单,创建 MyThread 的实例,调用 start() 方法,这样 run() 方法中的代码就会在子线程中运行了

MyThread().start()

当然,使用继承的方式耦合性有点高,可以选择实现 Runnable 接口的方式定义一个线程

class MyThread : Runnable { 
   
    override fun run() { 
   
        // 编写具体的逻辑
    }
}

如果使用这种写法,启动线程的方式也需要进行相应的改变

val myThread = MyThread()
Thread(myThread).start()

也可以使用 Lambda 的方式

Thread { 
   
	// 编写具体的逻辑
}.start()

以上几种方式相比都不陌生,Kotlin 还给我们提供了更简单的开启线程的方式,我们只需要在 Lambda 表达式中编写具体的逻辑即可,连 start() 方法都不用调用

thread { 
   
    // 编写具体的逻辑
}

异步消息处理机制

Android 的 UI 是线程不安全的,如果想更新应用程序里的 UI 元素,必须在主线程中进行,否则会出现异常。但有些时候,我们必须在子线程执行一些耗时任务,然后根据任务的执行结果更新 UI,这时可以使用 Android 提供的一套异步消息处理机制解决

class MainActivity : AppCompatActivity() { 
   

    // 定义一个整型变量 updateText,用于表示更新 UI 动作
    val updateText = 1

    val handler = object : Handler(Looper.getMainLooper()) { 
   
        override fun handleMessage(msg: Message) { 
   
            // 在这里进行 UI 操作
            when (msg.what) { 
   
                // 更新 UI
            }
        }
    }

    override fun onCreate(savedInstanceState: Bundle?) { 
   
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        button.setOnClickListener { 
   
            thread { 
   
                // 创建一个 Message 对象
                val msg = Message()
                // 指定 what 值为 updateText
                msg.what = updateText
                // 将这条 msg 发送给 handler
                handler.sendMessage(msg)
            }
        }
    }
}       

下面对 Android 的异步机制做简要介绍,主要由四个部分组成:

  1. Message

    Message 是在线程之间传递的消息,可以在内部携带少量的信息,用于不同线程之间传递数据。比如上述代码使用了 Message 的 what 字段,除此之外还可以使用 arg1 和 arg2 字段携带一些整型数据,使用 obj 字段携带一个 Object 对象

  2. Handler

    处理者,主要用于发送和处理消息。发送消息一般使用 sendMessage() 方法、post() 方法等,而发出的消息经过一系列地辗转处理后,最终会传递到 Handler 的 handlerMessage() 方法中

  3. MessageQueue

    消息队列,主要用于存放所有通过 Handler 发送的消息,这部分消息会一直存在于消息队列中,等待被处理,每个线程中只会有一个 MessageQueue 对象

  4. Looper

    Looper 是每个线程中 MessageQueue 的管家,调用 Looper 的 loop() 方法后,就会进入一个无限循环中,然后每当发现 MessageQueue 中存在一条消息时,就会将它取出,并传递到 Handler 的 handlerMessage() 方法中,每个线程只会有一个 Looper 对象

现在再来把异步消息处理的流程梳理一下:首先需要在主线程中创建一个 Handler 对象,并重写 handlerMessage() 方法。然后子线程创建一个 Message 对象,并通过 Handler 将这条消息发送出去。之后这条消息会被添加到 MessageQueue 的队列等待被处理。Looper 会一直尝试从 MessageQueue 中取出待处理消息,由 handleMessage() 方法处理

AsyncTask

AsyncTask 的实现原理也是基于异步消息处理机制,只是 Android 帮我们做了更好的封装

AsyncTask 是一个抽象类,使用时必须创建一个子类去继承它,在继承时我们可以为 AsyncTask 类指定三个泛型参数:

  • Params:在执行 AsyncTask 时需要传入的参数,可用于在后台任务中使用
  • Progress:在后台任务执行时,如果需要在界面上显示当前进度,则使用这里指定的泛型作为进度单位
  • Result:当任务执行完毕后,,如果需要对结果进行返回,则使用这里指定的泛型作为返回值类型

一个最简单的自定义 AsyncTask 可以写成如下形式:

class DownloadTask : AsyncTask<Unit, Int, Boolean>() { 
   
    ...
}

这里我们把 AsyncTask 的第一个泛型参数指定为 Unit,表示在执行 AsyncTask 时不需要传入参数给后台任务。第二个泛型参数指定为 Int,表示使用整型数据作为进度显示单位。第三个泛型参数指定为 Boolean,表示使用布尔型数据反馈执行结果

AsyncTask 经常需要重写的方法有以下四个:

  1. onPreExecute()

    这个方法会在后台任务开始执行之前调用,用于进行一些界面上的初始化操作,比如显示一个进度条对话框

  2. doInBackgroound(Params…)

    这个方法中的所有代码都会在子线程中运行,应该在这里处理所有耗时任务,任务一旦完成,就可以通过 return 语句返回结果。如果需要更新 UI,比如反馈当前任务的执行进度,可以调用 publishProgress(Progress…) 方法来完成

  3. onProgressUpdate(Progress…)

    当调用了 publishProgress(Progress…) 方法后,该方法就会被调用,携带的参数就是后台任务中传递过来的,在这里可以对 UI 进行操作

  4. onPostExecute(Result)

    当后台任务执行完毕并通过 return 语句返回时,这个方法很快就会被调用,返回的数据会作为参数传递到此方法中,可以利用返回的数据进行一些 UI 操作

因此,一个完整的自定义 AsyncTask 就可以写成如下形式:

class DownloadTask : AsyncTask<Unit, Int, Boolean>() { 
   

    override fun onPreExecute() { 
   
        // 显示进度对话框
        progressDialog.show()
    }

    override fun doInBackground(vararg params: Unit?) = try { 
   
        while (true) { 
   
            // 这是一个虚构的方法
            val downloadPercent = doDownload()
            publishProgress(downloadPercent)
            if (downloadPercent >= 100) { 
   
                break
            }
        }
        true
    } catch (e: Exception) { 
   
        false
    }

    override fun onProgressUpdate(vararg values: Int?) { 
   
        // 在这里更新下载进度
        progressDialog.setMessage("Download ${ 
     values[0]}%")
    }

    override fun onPostExecute(result: Boolean?) { 
   
        // 关闭进度对话框
        progressDialog.dismiss()
        // 在这里提示下载结果
        if (result) { 
   
            Toast.makeText(context, "Download succeeded", Toast.LENGTH_SHORT).show()
        } else { 
   
            Toast.makeText(context, "Download failed", Toast.LENGTH_SHORT).show()
        }
    }
}

想要启动这个任务,只需要以下代码即可

DownloadTask().execute()

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/234259.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 线程池介绍及创建线程池的4种方式是什么_程序可以创建几个线程池

    线程池介绍及创建线程池的4种方式是什么_程序可以创建几个线程池1.什么是线程池Java中的线程池是运用场景最多的并发框架,几乎所有需要异步或并发执行任务的程序都可以使用线程池。在开发过程中,合理地使用线程池能够带来3个好处。第一:降低资源消耗。通过重复利用已创建的线程降低线程创建和销毁造成的消耗。第二:提高响应速度。当任务到达时,任务可以不需要等到线程创建就能立即执行。第三:提高线程的可管理性。线程是稀缺资源,如果无限制地创建,不仅会消耗系统资源…

  • navicat15for激活码-激活码分享

    (navicat15for激活码)最近有小伙伴私信我,问我这边有没有免费的intellijIdea的激活码,然后我将全栈君台教程分享给他了。激活成功之后他一直表示感谢,哈哈~IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.cn/100143.html…

  • 查看Python安装路径_Python安装路径

    查看Python安装路径_Python安装路径查看Python安装路径方法在使用python的时候,有时候会需要找到python包的安装位置,怎么办?法一、对于Windows平台,打开cmd输入命令wherePython,回车(即按下Enter键)可输出Python的安装路径。参见下图:【如何清除命令行窗口内容输入cls回车】法二、在IDLE(Python自带的简洁的集成开发环境)中先输入impor…

  • 分布式服务框架 Zookeeper安装和配置详解

    分布式服务框架 Zookeeper安装和配置详解

  • 很震撼的HTML5视频播放器,电影院的感觉建议收藏

    效果很震撼!有电影院的感觉了。呵呵。看了下代码,依然是在一个canvas里嵌入然后getImageData点击这里查看效果代码:varcanvas=document.createElement

    2021年12月20日
  • springboot启动成功访问404_springboot启动执行

    springboot启动成功访问404_springboot启动执行今天在做一个springboot项目的时候,是接着别人的项目写的,写完之后想做一下测试,于是就启动了springboot,然后在放问的时候,一直包404的错误,然后百度了一下网上给的方法,包括注解使用@RestController,然后去除掉方法@RequestMapping(value="/add")中的“value=”,这个方法对我无用,因为我的项目之前就是用的@RestC…

    2022年10月13日

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号