Android多线程(一)

Android 线程简单分析(一)
Android 并发之synchronized锁住的是代码还是对象(二)
Android 并发之CountDownLatch、CyclicBarrier的简单应用(三)
Android 并发HashMap和ConcurrentHashMap的简单应用(四)(待发布)
Android 并发之Lock、ReadWriteLock和Condition的简单应用(五)
Android 并发之CAS(原子操作)简单介绍(六)
Android 并发Kotlin协程的重要性(七)(待发布)
Android 并发之AsyncTask原理分析(八)
Android 并发之Handler、Looper、MessageQueue和ThreadLocal消息机制原理分析(九)
Android 并发之HandlerThread和IntentService原理分析(十)

概述

Android沿用了Java的线程模型,一个Android应用对应着一个进程,同时会开启一个线程称为主线程或者UI线程,如果我们想要访问网络或者访问数据库等耗时的操作时,都会开启一个子线程去处理,从Android 3.0开始,系统为了避免阻塞主线程,要求网络访问必须在子线程中进行,否则将抛出异常。下面这张图是我的要讲的多线程体系导图:


多线程体系.png

1、进程和线程简介

提到线程,不得不提进程。进程是操作系统结构的基础,是程序在一个数据集合上运行的过程,是系统进行资源分配和调度的基本单位。同时进程可以被看做程序的实体,也是线程的容器。Android应用可以被看做是一个进程,但是Android系统默认分配的内存是有限的,很多开发人员为了解决这个瓶颈,引入了多进程开发,看一下我在Windows下任务管理器,列表里面的.exe程序就是一个进程,第二张图是在Android Studio下对应的一个进程,以gradle下的applicationId为进程ID,一般都是以包名命名:


image.png

image.png

1.2 线程状态

开始之前说一下时间片,时间片就时cpu分配给线程运行的时间,当时间到了线程将暂停,知道cpu重新分配时间,举个栗子:假如cpu是单核,那么同时只能运行一个线程,cpu就时这样不断切换,给人一种程序时并发运行的,其实不是的,只是这段时间比较短,我们是感觉不到的。
Java线程运行的声明周期中可能会处于5种不同的状态,Java中的线程的生命周期大体可分为5种状态。这5种状态分别如下:

  1. 新建(NEW):新创建了一个线程对象。
  2. 可运行(RUNNABLE):线程对象创建后,其他线程(比如main线程)调用了该对象的start()方法。该状态的线程位于可运行线程池中,等待被线程调度选中,获取cpu 的使用权 。
  3. 运行(RUNNING):可运行状态(runnable)的线程获得了cpu 时间片(timeslice) ,执行程序代码。
  4. 阻塞(BLOCKED):阻塞状态是指线程因为某种原因放弃了cpu 使用权,也即让出了cpu timeslice,暂时停止运行。直到线程进入可运行(runnable)状态,才有机会再次获得cpu timeslice 转到运行(running)状态。阻塞的情况分三种:
    (一). 等待阻塞:运行(running)的线程执行o.wait()方法,JVM会把该线程放入等待队列(waitting queue)中。
    (二). 同步阻塞:运行(running)的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池(lock pool)中。
    (三). 其他阻塞:运行(running)的线程执行Thread.sleep(long ms)或t.join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入可运行(runnable)状态。
  5. 死亡(DEAD):线程run()、main() 方法执行结束,或者因异常退出了run()方法,则该线程结束生命周期。死亡的线程不可再次复生。


    线程状态.jpg
  • 初始状态(new)
    实现Runnable接口和继承Thread可以得到一个线程类,new一个实例出来,线程就进入了初始状态
  • 可运行状态(Rannable)
    1、可运行状态(Rannable)只是说你资格运行,调度程序没有挑选到你,你就永远是可运行状态。
    2、调用线程的start()方法,此线程进入可运行状态。
    3、当前线程sleep()方法结束,其他线程join()结束,等待用户输入完毕,某个线程拿到对象锁,这些线程也将进入可运行状态。
    4、当前线程时间片用完了,调用当前线程的yield()方法,当前线程进入可运行状态。
    5、锁池里的线程拿到对象锁后,进入可运行状态。
  • 运行状态(Running)
    1、线程调度程序从可运行池中选择一个线程作为当前线程时线程所处的状态。这也是线程进入运行状态的唯一一种方式。
    *死亡状态
    1、当线程的run()方法完成时,或者主线程的main()方法完成时,我们就认为它死去。这个线程对象也许是活的,但是,它已经不是一个单独执行的线程。线程一旦死亡,就不能复生。
    2、在一个死去的线程上调用start()方法,会抛出ava.lang.IllegalThreadStateException异常。
  • 阻塞状态
    1、当前线程T调用Thread.sleep()方法,当前线程进入阻塞状态。
    2、运行在当前线程里的其它线程t2调用join()方法,当前线程进入阻塞状态。
    3、等待用户输入的时候,当前线程进入阻塞状态。
  • 等待队列(本是Object里的方法,但影响了线程)
    1、调用obj的wait(), notify()方法前,必须获得obj锁,也就是必须写在synchronized(obj) 代码段内。
    2、与等待队列相关的步骤和图
    2.1、线程1获取对象A的锁,正在使用对象A。
    2.2、线程1调用对象A的wait()方法。
    2.3、线程1释放对象A的锁,并马上进入等待队列。
    2.4、锁池里面的对象争抢对象A的锁。
    2.5、线程5获得对象A的锁,进入synchronized块,使用对象A。
    2.6、线程5调用对象A的notifyAll()方法,唤醒所有线程,所有线程进入锁池。||||| 线程5调用对象A的notify()方法,唤醒一个线程,不知道会唤醒谁,被唤醒的那个线程进入锁池。
    2.7、notifyAll()方法所在synchronized结束,线程5释放对象A的锁。
    锁池里面的线程争抢对象锁,但线程1什么时候能抢到就不知道了。||||| 原本锁池+第6步被唤醒的线程一起争抢对象锁。


    阻塞队列.jpg

    *锁池状态
    1、当前线程想调用对象A的同步方法时,发现对象A的锁被别的线程占有,此时当前线程进入锁池状态。简言之,锁池里面放的都是想争夺对象锁的线程。
    2、当一个线程1被另外一个线程2唤醒时,1线程进入锁池状态,去争夺对象锁。
    3、锁池是在同步的环境下才有的概念,一个对象对应一个锁池。

  • 几个方法的比较
    1、Thread.sleep(long millis),一定是当前线程调用此方法,当前线程进入阻塞,但不释放对象锁,millis后线程自动苏醒进入可运行状态。作用:给其它线程执行机会的最佳方式。
    2、Thread.yield(),一定是当前线程调用此方法,当前线程放弃获取的cpu时间片,由运行状态变会可运行状态,让OS再次选择线程。作用:让相同优先级的线程轮流执行,但并不保证一定会轮流执行。实际中无法保证yield()达到让步目的,因为让步的线程还有可能被线程调度程序再次选中。Thread.yield()不会导致阻塞。
    3、t.join()/t.join(long millis),当前线程里调用其它线程1的join方法,当前线程阻塞,但不释放对象锁,直到线程1执行完毕或者millis时间到,当前线程进入可运行状态。
    4、obj.wait(),当前线程调用对象的wait()方法,当前线程释放对象锁,进入等待队列。依靠notify()/notifyAll()唤醒或者wait(long timeout)timeout时间到自动唤醒。
    5、obj.notify()唤醒在此对象监视器上等待的单个线程,选择是任意性的。notifyAll()唤醒在此对象监视器上等待的所有线程。
  • 两个疑问
    1、当对象锁被某一线程释放的一瞬间,锁池里面的哪个线程能获得这个锁?随机?,我觉得应该是竞争锁,也是随机的。
    2、等待队列里许许多多的线程都wait()在一个对象上,此时某一线程调用了对象的notify()方法,那唤醒的到底是哪个线程?java文档就简单的写了句:选择是任意性的。

如何创建线程?

1、继承Thread类,重写run方法:

 static class MyThread extends Thread {
    @Override
    public void run() {
        System.out.println("my thread");
    }
}
public static void main(String[] args) {
    MyThread myThread = new MyThread();
    myThread.start();
}

Thread本质上也是实现了Runnable接口。需要注意的是调用thread.start方法后并不能立即执行线程,而是线程的状态变为可运行的状态,等待CPU的调度。

2、实现Callable接口,重写call方法

static class MyCallable implements Callable<String> {

    @Override
    public String call() throws Exception {
        Thread.sleep(3000);
        return "hahah";
    }
}

public static void main(String[] args) throws ExecutionException, InterruptedException {
    System.out.println("start MyCallable");
    Future<String> future = executorService.submit(new MyCallable());
    String result = future.get();
    System.out.println("result: " + result);
}

Callable实际上是属于Executor框架中的功能类,Callable接口与Runnable接口的功能是类似的,但比Runnable更强大。

  • Callable中call方法可以在任务执行完成提供一个返回值,Runanble接口是无法做到的。
  • Callable接口的call方法可以跑出异常,而Runnable接口的run方法无法跑出异常。
  • 运行Callable可以得到一个Future对象,Future对象表示异步计算的结果,它提供了检查计算是否完成,通过Future的get方法获取目标线程call方法的情况,Future的get方法会阻塞当前线程知道call方法返回值会用或者抛出异常,或者get方法设置超时。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,723评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,080评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,604评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,440评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,431评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,499评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,893评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,541评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,751评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,547评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,619评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,320评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,890评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,896评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,137评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,796评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,335评论 2 342

推荐阅读更多精彩内容