Android多线程学习:线程

news2024/11/22 5:50:52

一、概念

进程:系统资源分配的基本单位,进程之间相互独立,不能直接访问其他进程的地址空间。

线程:CPU调度的基本单位,线程之间共享所在进程的资源,包括共享内存,公有数据,全局变量等。

后台线程:后台线程又称为守护线程(Daemon Thread),JVM的垃圾回收线程就是典型的后台线程。

举例记忆:以下纯属本人瞎编,方便记忆

  • 进程就是一个鞋子工厂,鞋子由鞋带、鞋底、鞋帮三部分组成。线程就是工厂下的流水线,一条流水线做鞋带,一条流水线做鞋底,一条流水线做鞋帮,最后再把做好的组件组装起来变成一个鞋子。
  • 我们会把原材料直接提供给工厂,工厂统一接收而不是里面具体的某个流水线。所以工厂(进程)就是我们系统分配资源的最小单位。
  • 如果想做出鞋子,至少要开启一个流水线工作,这个流水线可以先做鞋带,在做鞋帮,在做鞋底,最后再组装成鞋子,如果没有流水线工作,一双鞋子也做不出来。所以流水线是系统可调度执行的基本单位。
  • 工厂包含多条流水线,即进程包含多个线程,而多条流水线上的工人又共享工厂里的食堂、厕所、宿舍,线程之间共享所在进程的资源。

二、线程三种实现

1、继承Thread类

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //3、创实例,并调用start()方法开启线程。
        new MyThread().start();
        new MyThread().start();
    }

    //1、定义一个类MyThread继承Thread,并重写run方法
    class MyThread extends Thread {

        @Override
        public void run() {
            //2、将执行的代码写在run方法中。
            Log.d(TAG, "线程名字:" + Thread.currentThread().getName());
        }
    }
}

2、实现Runnable接口

public class MainActivity extends AppCompatActivity {
    public static final String TAG = MainActivity.class.getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //3、创建Thread对象, 传入MyRunnable的实例,并调用start()方法开启线程。
        Thread thread = new Thread(new MyRunnable());
        thread.start();
        Thread thread1 = new Thread(new MyRunnable());
        thread1.start();
    }

    // 1、定义一个类MyRunnable实现Runnable接口,并重写run方法。
    class MyRunnable implements Runnable {
        public void run() {
            //2、将执行的代码写在run方法中。
            Log.d(TAG, "线程名字:" + Thread.currentThread().getName());
        }
    }
}

3 、通过Callable和Future创建有返回值的多线程

public class MainActivity extends AppCompatActivity {
    private final String TAG = this.getClass().getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //3、创建线程池对象,调用submit()方法执行MyCallable任务,并返回Future对象
        ExecutorService pool = Executors.newSingleThreadExecutor();
        Future<Integer> f1 = pool.submit(new MyCallable());
        //4、调用Future对象的get()方法获取call()方法执行完后的值
        try {
            Log.d(TAG, "sum = " + f1.get());
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        //5、关闭线程池
        pool.shutdown();
    }

    //1、自定义一个类MyCallable实现Callable接口,并重写call()方法
    public class MyCallable implements Callable<Integer> {
        @Override
        public Integer call() throws Exception {
            //2、将要执行的代码写在call()方法中
            int sum = 0;
            for (int i = 0; i <= 100; i++) {
                sum += i;
            }
            return sum;
        }
    }
}

三、线程的生命周期

1、线程的生命周期包括:新建New,就绪Runnable,运行Running,阻塞Blocked,和死亡Dead,5种状态。

新建:程序使用new关键字之后,该线程就处于新建状态,jvm为其分配内存,并初始化成员变量。

就绪:程序调用start() 方法之后,该线程就处于就绪状态,jvm为其创建方法调用栈和PC计数器。

运行:如果就绪状态的线程获得了CPU,那么程序就处于运行状态。

阻塞:指一个线程在执行过程中暂停,以等待某个条件的触发。

死亡:线程执行体执行结束,以及抛出一个未捕获的ExceptionError,或者直接调用stop()方法结束该线程。可以通过线程对象的isAlive()方法,来判断线程对象的状态。

2、生命周期转换

(1)运行到阻塞:

  • 线程调用sleep()方法,主动放弃所占有的CPU资源;
  • 线程调用了一个阻塞式IO方法,在该方法返回之前,该线程被阻塞;
  • 线程试图获得一个同步监视器,但是该同步监视器被其他线程所持有;
  • 线程等待某个通知notify()notify()通常与wait()配合使用;
  • 线程调用suspend(),挂起,该方法容易造成死锁,不建议使用。

(2)阻塞到就绪:

  • sleep()方法的线程经过了指定的sleep的时间;
  • 阻塞式IO方法返回值;
  • 成功获得了同步监视器;
  • 线程获得了其他线程发出的通知,被唤醒;
  • 挂起的线程调用了resume()方法恢复。

(3)状态转换图
线程生命周期.png

四、线程常用方法

1、setPriority(int newPriority) - 设置线程优先级

  • 设置线程的优先级来改变线程争抢到时间片的概率,优先级高的争抢到时间片的概率越大;
  • 优先级的取值范围:1~10,默认为5,数字越大,优先级越高;
  • 这个方法的调用必须在start之前,否则没有任何意义;
  • 使用方法getPriority(),获取当前线程优先级。

2、setDeamon(boolean b) - 设置后台线程

  • 如果所有的前台线程死亡,那么后台线程会自动死亡;
  • 默认情况下所有的线程都是前台线程;
  • 这个方法的调用必须在start之前。

3、sleep(long millis) - 设置线程休眠

  • 让当前线程暂停millis毫秒,并进入阻塞状态,睡眠状态的线程不会释放同步监视器,在此期间该线程不会获得执行的机会;
  • 使用sleep方法时需要捕捉InterruptedException或者抛出该异常。

4、interrupt() - 线程中断

  • 表示的并不是将线程结束,而是表示清除阻塞状态;
  • 中断线程操作实质上是修改了一下中断标示位为true
  • 如果线程处于阻塞状态,抛出异常InterruptedException
public class MainActivity extends AppCompatActivity {
    public static final String TAG = MainActivity.class.getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Thread thread = new Thread(new MyRunnable());
        thread.start();

        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //打断休眠线程
        thread.interrupt();
    }

    class MyRunnable implements Runnable {
        public void run() {
            try {
                Log.i(TAG, "----开始休眠-----");
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
                Log.i(TAG, "----线程中断-----");
            }
            Log.i(TAG, "----休眠后执行-----");
        }
    }
}

执行结果:报异常,提示中断,取消线程阻塞,执行休眠后操作。interrupt.PNG

public class MainActivity extends AppCompatActivity {
    public static final String TAG = MainActivity.class.getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Thread thread = new Thread(new MyRunnable());
        thread.start();
        thread.interrupt();
    }

    class MyRunnable implements Runnable {
        public void run() {
            for (int i = 0; i < 5; i ++){
                Log.i(TAG,"打印: " + i);
                if (Thread.interrupted()){
                    Log.i(TAG, "----线程被中断了-----");
                }
            }
            Log.i(TAG, "----执行完了- interrupt state: " + Thread.interrupted());
        }
    }
}

执行结果:interrupted()只有主线程调用的时候才会是truefor循环结束后再次执行是falseinterrupt2.PNG

5、join() - 线程合并

  • 在执行原来的线程的过程中,如果遇到了合并线程,则优先执行合并进来的线程,当合并线程执行完毕之后,再接着执行原来的线程;
  • 调用join方法之前,一定要将线程start
  • join(0)的意思不是A线程等待B线程0秒,而是A线程等待B线程无限时间,直到B线程执行完毕,即join(0)等价于join()
public class MainActivity extends AppCompatActivity {
   public static final String TAG = MainActivity.class.getSimpleName();

   @Override
   protected void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_main);
       Thread thread1 = new Thread(new JoinRunnable());
       thread1.start();

       for (int i = 0; i < 5; i++) {
       //主线程执行1的时候,把执行权让给了子线程
           if (i == 1) {
               try {
                   thread1.join();
               } catch (InterruptedException e) {
                   e.printStackTrace();
               }
           }
           Log.i(TAG, Thread.currentThread().getName() + "正在运行....");
       }
   }


   class JoinRunnable implements Runnable {

       @Override
       public void run() {
           for (int i = 0; i < 3; i++) {
               try {
                   Thread.sleep(1000);
                   Log.i(TAG, Thread.currentThread().getName() + "正在运行....");
               } catch (InterruptedException e) {
                   e.printStackTrace();
               }
           }
       }
   }
}

执行结果:
join3.PNG
6、yield() - 线程让步

  • 使得正在执行的线程暂停,但不会阻塞线程,释放自己拥有的CPU,线程进入就绪状态。
  • 只有优先级与当前线程相同,或者优先级比当前线程更高的线程才有可能获得执行机会,但是可能是当前线程又进入到“运行状态”继续运行。
public class MainActivity extends AppCompatActivity {
    public static final String TAG = "test";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Thread thread1 = new Thread(new MyRunnable());
        thread1.start();
    }


    class MyRunnable implements Runnable {

        @Override
        public void run() {
            long begainTime = System.currentTimeMillis();
            int count = 0;
            for (int i = 0; i < 5000000; i++) {
                //结果2需要将下面注释放开
                //Thread.yield();
                count = count + (i + 1);
            }
            long endTime = System.currentTimeMillis();
            Log.i(TAG, "用时:" + (endTime - begainTime) + "ms");
        }
    }
}

运行结果1:
yield1.PNG

运行结果2,执行时放开Thread.yield()
yield3.PNG

  • yield()也不会释放锁标志,示例如下:
public class MainActivity extends AppCompatActivity {
    public static final String TAG = "test";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Thread thread1 = new Thread(new MyRunnable());
        Thread thread2 = new Thread(new MyRunnable());
        thread1.start();
        thread2.start();
    }

    static class MyRunnable implements Runnable {
        private static Object obj = new Object();

        @Override
        public void run() {
            synchronized (obj) {
                for(int i = 0;i < 5;i++) {
                    Log.i(TAG, Thread.currentThread().getName() + "正在执行i: " + i);
                    if(i == 2) {
                        Thread.currentThread().yield();
                    }
                }
            }
        }
    }
}

运行结果,Thread-2获取锁以后,就算yield,也没释放锁:
yield4.PNG
7、stop() - 线程停止

  • 官方不建议使用该方法,因为stop不安全,stop会解除由线程获取的所有锁定,当在一个线程对象上调用stop()方法时,这个线程对象所运行的线程就会立即停止,假如一个线程正在执行:synchronized void { x = 3; y = 4;} 由于方法是同步的,多个线程访问时总能保证x,y被同时赋值,而如果一个线程正在执行到x = 3;时,被调用了stop()方法,即使在同步块中,它也会马上stop了,这样就产生了不完整的脏数据。

参考文章:
对进程、线程、多线程、线程池的理解
带你通俗易懂的理解——线程、多线程与线程池
线程类的常见方法介绍
java多线程以及Android多线程

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1069687.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Pod】

Pod 一、Pod基本概念二、Pod的使用方式pause容器&#xff08;pod的基础容器&#xff09;核心功能pause容器使得Pod中所有容器可以共享两种资源&#xff1a;网络和存储网络存储 三、Pod分类自主式Pod/静态pod控制器管理的Pod 四、三种容器五、镜像拉取策略&#xff08;image Pul…

云计算安全和云原生安全的关系

云计算安全(Cloud Computing Security)指的是在云环境中保护数据、应用程序和基础设施的安全性。它包括保护云服务提供商的基础设施和平台&#xff0c;以及云服务用户的数据和应用程序。 云原生安全(Cloud-Native Security)则是指在云原生环境中保护应用程序和服务的安全性。云…

谁说手机没有高质量抓拍?华为Mate 60系列与Mate X5让你体验“时间凝固”!

我们日常拍照时&#xff0c;经常会出现“照片糊了”的现象&#xff0c;这是由于被拍摄的人或者物快速移动导致。 来源网图&#xff0c;侵删 抓拍&#xff0c;Snap photography&#xff0c;“抓住时机&#xff0c;把瞬间出现的情景拍摄下来拍照”&#xff0c;又名写实抓拍&…

手写Demo体验volatile可见性的作用

volatile是java的关键字&#xff0c;作用&#xff1a;①保证线程间的可见性&#xff1b;②防止指令重排。下面看一个demo&#xff0c;启动2个线程&#xff0c;一个线程读取flag变量的值&#xff0c;另外一个线程修改flag变量的值。 public class VolatileDemo {private static…

前端自动化测试入门教程

&#x1fab4; 背景 前端的自动化测试主要可以分为以下四种&#xff1a; 单元测试&#xff08;Unit Test&#xff09;&#xff1a;对一个函数/组件进行测试&#xff0c;一般用于公共函数/公共组件的测试维护。常用框架有 Jest、Jasmine、Mocha等&#xff1b; 集成测试&#x…

VR全景拍摄酒店,为用户消除“不透明度”

近日在各大社交平台上&#xff0c;出现了不少吐槽国庆期间酒店价格太贵的帖子&#xff0c;而一些热门旅游地的度假酒店、网红民宿的热门房型已经“一房难求”&#xff0c;这就出现酒店房型与预定房型不同的现象出现&#xff0c;VR全景拍摄技术同酒店行业的结合&#xff0c;就可…

韩语学习|韩语零基础|柯桥韩语学校,每日一词

今日一词:개방도 평지 韩语每日一词打卡:개방도[개방도]【名词】开放度,开放程度 原文&#xff1a;한 지역의 개방도는 경제 발전 수준에 달려 있습니다. 意思&#xff1a;一个地区的开放程度取决于经济发展水平。 【原文分解】 1、경제[경제]经济 2、지역[지역]地域 3、발전[발…

代码随想录算法训练营第23期day12| 239. 滑动窗口最大值 、347. 前K个高频元素

目录 一、&#xff08;leetcode 239&#xff09;滑动窗口最大值​​​jiao 二、&#xff08;leetcode 347&#xff09;前 K 个高频元素 优先级队列与大小顶堆 一、&#xff08;leetcode 239&#xff09;滑动窗口最大值​​​jiao 力扣题目链接 状态&#xff1a;待回顾&…

kafka、rabbitmq 、rocketmq的区别

一、语言不同 RabbitMQ是由内在高并发的erlanng语言开发&#xff0c;用在实时的对可靠性要求比较高的消息传递上。 kafka是采用Scala语言开发&#xff0c;它主要用于处理活跃的流式数据,大数据量的数据处理上 二、结构不同 RabbitMQ采用AMQP&#xff08;Advanced Message Q…

Excel·VBA使用ADO读取工作簿工作表数据

目录 查询遍历写入数组查询整体写入数组查询工作簿所有工作表名称查询工作簿所有工作表数据 不打开工作簿读取数据&#xff0c;以下举例都为《ExcelVBA合并工作簿》中 7&#xff0c;合并子文件夹同名工作簿中同名工作表&#xff0c;纵向汇总数据所举例的工作簿&#xff0c;使用…

国庆作业day10.4

QT实现TCP服务器客户端搭建的代码&#xff0c;现象 服务器 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);//实例化一个服务器servernew QTcpServer(this);c…

探秘小米增程汽车与仿生机器人的未来:AI大模型的潜在影响及苹果iPhone15Pro发热问题解决之道

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

智能井盖传感器:城市安全卫士

随着城市人口的不断增加和城市基础设施的不断发展&#xff0c;井盖作为城市道路和排水系统的重要组成部分&#xff0c;承担着确保城市安全和便利性的关键角色。然而&#xff0c;井盖在日常使用中常常面临倾斜、水浸和翻转等问题&#xff0c;这些问题可能导致交通阻塞、行人坠井…

漏洞挖掘--edusrc两连杀

免责声明&#xff1a;文章中涉及的漏洞均已修复&#xff0c;敏感信息均已做打码处理&#xff0c;文章仅做经验分享用途&#xff0c;切勿当真&#xff0c;未授权的攻击属于非法行为&#xff01; 最近闲来无事&#xff0c;研究了下通达的day&#xff0c;由于本人太菜了&#xff…

软考网络工程师考前如何复习?

先看一下这知识点总结图&#xff0c;在备考复习前大致简单了解一遍&#xff01; 第一步&#xff1a; 通读教程&#xff08;《网络工程师》&#xff09;&#xff0c;首先对教程中的各章节及知识点有一个基本的认识&#xff0c;第一阶段基本以泛读为主&#xff0c;不求立刻记得所…

Maven 配置阿里云镜像

1. 查找maven setting.xml配置文件 find / -name "setting.xml" 2. 添加阿里云镜像 修改maven根目录下的conf文件夹中的setting.xml文件中的mirrors下添加mirror标签 <settings> <localRepository>E:\Maven\repository</localRepository> <…

LLMs Python解释器程序辅助语言模型(PAL)Program-aided language models (PAL)

正如您在本课程早期看到的&#xff0c;LLM执行算术和其他数学运算的能力是有限的。虽然您可以尝试使用链式思维提示来克服这一问题&#xff0c;但它只能帮助您走得更远。即使模型正确地通过了问题的推理&#xff0c;对于较大的数字或复杂的运算&#xff0c;它仍可能在个别数学操…

【Kotlin精简】第2章 集合

1 简介 在 Kotlin 中集合主要分为可变集合与只读集合&#xff0c;其中可变集合使用 “Mutable” 前缀 集合类名表示&#xff0c;比如 MutableList、MutableSet、MutableMap 等。而对于只读集合就是和 Java 中集合类名是一致。 Java 中的 List 非 Kotlin 中的 List , 因为 Kot…

深度学习——权重衰减(weight_decay)

深度学习——权重衰减&#xff08;weight_decay) 文章目录 前言一、权重衰减1.1. 范数与权重衰减1.2. 高维线性回归1.3. 从零开始实现1.3.1.初始化模型参数1.3.2. 定义L₂范数惩罚1.3.3. 定义训练代码实现1.3.4. 不管正则化直接训练1.3.5. 使用权重衰减 1.4. 简洁实现 总结 前言…

寒露到了,冬天还会远吗?

寒露惊秋晚&#xff0c;朝看菊渐黄。 日复一日间&#xff0c;光影如梭&#xff0c;我们便很快将告别了秋高气爽&#xff0c;白日将变得幽晦&#xff0c; 天寒夜长&#xff0c;风气萧索&#xff0c;雾结烟愁。 还没好好体会秋高气爽,寒露就到了。 今天晚上9点多&#xff0c;我们…