实现多线程的几种方式

方式一:实现继承Thread类

步骤:
1:定义一个类去继承Thread类,比如ThreadDemo
2:重写run()方法
3:在测试类中创建ThreadDemo类的对象
4:启动线程

ThreadDemo类

public class ThreadDemo extends Thread {
    @Override
    public void run() {
        for (int i = 0; i <100 ; i++) {
            System.out.println("多线程运行..."+i);
        }
    }
}

测试类

public class TestDemo {
    public static void main(String[] args){
       	ThreadDemo threadDemo1 = new ThreadDemo();
        ThreadDemo threadDemo2 = new ThreadDemo();
        //线程执行的方法,开启了俩个线程
        threadDemo1.start();
        threadDemo2.start();
    }
}

运行结果
继承Thread类方式创建线程的运行结果
看了一下代码,其实这个方式也是通过实现Runnable接口完成的

方式二:实现Runnable接口

步骤:
1:定义一个类,实现Runnable接口,比如RunnableDemo类
2:重写run()方法
3:创建测试类,并创建RunnableDemo类对象
4:创建Thread对象,将RunnableDeme类对象作为构造方法的参数传进去
5:启动线程

RunableDemo类

public class RunnableDemo implements Runnable {
    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            System.out.println(Thread.currentThread().getName()+"-----"+i);
        }
    }
}

测试类

public class TestDemo {
    public static void main(String[] args){
        RunnableDemo runnableDemo = new RunnableDemo();
        Thread thread1 = new Thread(runnableDemo, "线程1");
        Thread thread2 = new Thread(runnableDemo, "线程2");
        thread1.setPriority(10);//这是设置线程优先级的方法,10优先级最高,1最低
        thread2.setPriority(1);
        thread1.start();
        thread2.start();
    }
}

运行结果
实现Runnable接口的运行结果
与第一种方式看起来其实区别不大哈,就是实现和继承的区别,还有创建线程时需要将那个实现了Runnable接口的对象传入Thread中当构造方法的参数

方式三:实现Callable接口

步骤:
1:定义一个类实现Callable接口,并指定返回值类型,比如CallableDemo
2:重写call()方法
3:创建测试类,创建CallableDemo对象
4:创建Future的实现类FutureTask对象,并将CallableDemo对象作为构造方法的参数传进去
5:创建Thread类的对象,将FutureTask对象作为构造方法的参数传进去
6:启动线程
7:可以通过get方法获取线程结束后的结果

CallableDemo 类

//返回值是string类型
public class CallableDemo implements Callable<String> {
    @Override
    public String call() throws Exception {
        for (int i = 0; i < 100; i++) {
            System.out.println("正在执行" + i);
        }	
        //返回值就表示线程运行完毕之后的结果
        return "哦豁";
    }
}

测试类

public class Demo {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        //线程开启之后需要执行里面的call方法
        CallableDemo mc = new CallableDemo ();
        FutureTask<String> ft = new FutureTask<>(mc);
        //创建线程对象
        Thread t1 = new Thread(ft);
        String s = ft.get();
        //开启线程
        t1.start();
        System.out.println(s);
    }
}

这个跟之前的不同之处就是有个返回值,然后创建时有所不同

总结:
1:在阿里巴巴开发手册中明确提到,不允许显示创建线程,应该用线程池的方式提供
2:上面三种实现多线程的方式第一种因为是继承,扩展性弱,不推荐使用,第二种和第三种的区别就是有无返回值
3:反正都不咋用,看看了解一下得了,重点是下面的线程池创建线程

方式三:用线程池的方式实现多线程

1:什么是线程池
简单理解就是线程池是一个存储线程的池子,来了个任务我不需要专门去新建,之后又销毁,线程池中有大量的空闲线程,启动这个线程做任务,做完之后也不用销毁,线程会继续变成空闲状态
就好像是一个水池,我每次想喝水不用去挖个井,喝完之后再把井填上,直接从水池里面喝就行了

2:线程池的存在的意义
因为创建销毁线程比较消耗系统资源,为了避免频繁的去创建并销毁那些短暂的线程;
减少在创建和销毁线程上所花的时间以及系统资源的开销,
如果不使用线程池,有可能造成系统创建大量同类线程而导致消耗完内存或者“过度切换”的问题。

3:如何使用线程池
线程池有几种创建方式,比如
FixedThreadPool 定长线程池
ScheduledThreadPool,定时线程池
CachedThreadPool 可缓存线程池
SingleThreadExecutor 单线程化线程池
但是,阿里手册明确要求,不能使用他们
阿里规范
所以,线程池的使用重点是ThreadPoolExecutor

关于ThreadPoolExecutor的创建

//创建线程池
ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(核心线程数量,最大线程数量,空闲线程最大存活时间,任务队列,创建线程工厂,任务的拒绝策略);

在idea中ctrl+P查看创建线程池的参数如下
参数

下面来详细解释这些参数
参数1(必需):int corePoolSize :核心线程数;核心线程会一直存活,如果将 allowCoreThreadTimeout 设置为 true 时,核心线程也会被超时回收;
设置代码如下

threadPoolExecutor.allowCoreThreadTimeOut(true);

参数2(必需):int maximumPoolSize:线程池所能容纳的最大线程数。当活跃线程数达到该数值后,后续的新任务将会阻塞。如果线程数超过了核心线程数,那么会创建新的线程,最多就只能创建这么多;

参数3(必需):long keepAliveTime :线程闲置超时时长。如果超过该时长,非核心线程就会被回收。如果将 allowCoreThreadTimeout 设置为 true 时,核心线程也会超时回收。

参数4(必需):TimeUnit unit:指定 keepAliveTime 参数的时间单位。常用的有:TimeUnit.MILLISECONDS(毫秒)、TimeUnit.SECONDS(秒)、TimeUnit.MINUTES(分)。

参数5(必需):BlockingQueue workQueue : 任务队列。通过线程池的 execute() 方法提交的 Runnable 对象将存储在该参数中。其采用阻塞队列实现。
有以下几种

            new ArrayBlockingQueue<Runnable>(20);
            new LinkedBlockingDeque<Runnable>();
            new PriorityBlockingQueue<>();
            new SynchronousQueue<>();
            new LinkedBlockingDeque<>();
            new LinkedBlockingQueue<>();
            new DelayQueue<Delayed对象>();

参数6(可选):ThreadFactory threadFactory:线程工厂。用于指定为线程池创建新线程的方式。

参数7(可选):RejectedExecutionHandler handler:拒绝策略。当达到最大线程数时需要执行的饱和策略。默认丢弃任务并抛出 RejectedExecutionException 异常。

可选的这俩个参数有默认提供的

关于线程池的部分,参考大佬文章
彻底搞懂线程池

  • 6
    点赞
  • 68
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android实现多线程几种方式有以下几种方法。首先,可以使用Java的Thread类来创建和管理线程。这种方式需要手动创建线程对象,并在run()方法中编写线程的逻辑。其次,可以使用Java的Runnable接口来实现多线程。这种方式需要创建一个实现了Runnable接口的类,并将其作为参数传递给Thread类的构造函数。然后,可以使用Handler类来实现多线程。通过Handler的post()或postDelayed()方法,可以将任务添加到主线程的消息队列中,从而在主线程中执行异步操作。此外,还可以使用AsyncTask类来实现多线程。AsyncTask是Android提供的一个封装好的线程池,可以方便地在后台执行耗时操作,并在主线程中更新UI。通过重写AsyncTask的doInBackground()方法来执行耗时操作,并通过onPostExecute()方法来更新UI。需要注意的是,AsyncTask必须在主线程中创建实例,并且execute()方法也必须在主线程中调用。另外,Android 3.0之后,可以使用executeOnExecutor()方法来实现并发执行多个AsyncTask任务。总结起来,Android实现多线程几种方式包括使用Thread类、Runnable接口、Handler类和AsyncTask类。\[1\]\[2\]\[3\] #### 引用[.reference_title] - *1* [Android——多线程的多种实现方式](https://blog.csdn.net/abliudede/article/details/104891324)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [Android开发中四种常用的多线程实现方式](https://blog.csdn.net/star_nwe/article/details/130140238)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值