多线程的四种实现方式

1.继承Thread类,重写run方法

2.实现Runnable接口,重写run方法

3.通过Callable和ExecutorService创建线程

4.通过线程池创建线程

前面两种可以归结为一类:无返回值(通过重写run方法,run方式的返回值是void,所以没有办法返回结果)
后面两种可以归结成一类:有返回值,通过Callable接口,就要实现call方法,这个方法的返回值是Object,所以返回的结果可以放在Object对象中

第一种:继承Thread类,重写run方法

public class Test1 extends Thread{
    public Test1(){
        //编写子类的构造方法
    }
    public void run(){
     //   System.out.println(111);
        //编写自己的线程代码
        //Thread.currentThread表示当前代码段正在被哪个线程调用的相关信息
        System.out.println(Thread.currentThread().getName());

    }

    public static void main(String[] args) {
        Test1 test1 = new Test1();
        test1.setName("我是线程1");
        test1.start();
        System.out.println(Thread.currentThread().toString());
    }

}

运行结果:

Thread[main,5,main]
我是线程1

第二种:实现Runnable接口,重写run方法

public class Test2 {
    public static void main(String[] args) {
        System.out.println(Thread.currentThread().getName());
        Thread t1 = new Thread(new MyThread());
        t1.start();
    }
}
class MyThread implements Runnable{
    public void run() {
        System.out.println(Thread.currentThread().getName()+"->我是线程2");
    }
}

运行结果:

main
Thread-0->我是线程2

第三种:通过Callable和ExecutorService创建线程

public class Test3 {
    public static void main(String[] args) {

        Callable<Object> oneCallable = new Tickets<Object>();
        FutureTask<Object> oneTask=new FutureTask<Object>(oneCallable);

        Thread t = new Thread(oneTask);
        System.out.println(Thread.currentThread().getName());

        t.start();
    }
}
class Tickets<Object> implements Callable<Object>{
    public Object call() throws Exception {
        System.out.println(Thread.currentThread().getName()+"我是线程3");
        return null;
    }
}

运行结果:

main
Thread-0我是线程3

第四种:通过线程池创建线程

public class Test4 {

    private static int POOL_NUM=10;

    public static void main(String[] args) throws InterruptedException {
        //线程池的数量
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        for(int i=0;i<POOL_NUM;i++){
            RunnableThread thread = new RunnableThread();
            executorService.execute(thread);
        }
         //关闭线程池
        executorService.shutdown();
    }
}

class RunnableThread implements Runnable{
    public void run() {
        System.out.println("我是线程4:"+Thread.currentThread().getName());
    }
}

运行结果:

我是线程4:pool-1-thread-2
我是线程4:pool-1-thread-5
我是线程4:pool-1-thread-4
我是线程4:pool-1-thread-3
我是线程4:pool-1-thread-1
我是线程4:pool-1-thread-5
我是线程4:pool-1-thread-4
我是线程4:pool-1-thread-2
我是线程4:pool-1-thread-1
我是线程4:pool-1-thread-3

Executors类:提供了一系列工厂方法用于创建线程池,返回的线程池都实现了ExecutorService接口。

public static ExecutorService newFixedThreadPool(int nThreads)
创建固定数目线程的线程池。
public static ExecutorService newCachedThreadPool()
创建一个可缓存的线程池,调用execute 将重用以前构造的线程(如果线程可用)。如果现有线程没有可用的,则创建一个新线程并添加到池中。终止并从缓存中移除那些已有 60 秒钟未被使用的线程。
public static ExecutorService newSingleThreadExecutor()
创建一个单线程化的Executor。
public static ScheduledExecutorService newScheduledThreadPool(int
corePoolSize)
创建一个支持定时及周期性的任务执行的线程池,多数情况下可用来替代Timer类。
ExecutoreService提供了submit()方法,传递一个Callable,或Runnable,返回Future。如果Executor后台线程池还没有完成Callable的计算,这调用返回Future对象的get()方法,会阻塞直到计算完成。

二、Thread 和Runnable 的区别和联系

Thread其实本身就是实现了接口 Runnable的一个类;Thread类也是Runnable接口的子类。

因此 Thread中的方法和成员变量要比Runnable多,最典型地就是 Thread有start()方法,但是Runnable接口没有start()方法;

实际开发中我们通常采用Runnable接口来实现多线程。实现Runnable接口比继承Thread类有如下好处: 
1. 避免继承的局限,一个类可以继承多个接口,但是类只能继承一个类。 
2. Runnable接口实现的线程便于资源共享。而通过Thread类实现,各自线程的资源是独立的,不方便共享。 

  • 2
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值