线程创建的5种方式(面试)

线程创建的5种方式(面试)

线程创建有几种方式这个可以说是经典面试题了,我之前看到一个有趣的说法:如果你要说有 1种、2种、3种、4种 其实也是可以的。重要的是,你要能说出你的依据,讲出它们各自的不同点和共同点。讲得头头是道,让面试官对你频频点头。

接下来我来说说创建线程的5种方式

1.继承Thread类(包含了匿名内部类的方式)

上面演示快速上手一个线程就属于继承Thread类

  • 通过继承Thread类,并重写它的run方法,我们就可以创建一个线程。
  • 首先定义一个类来继承Thread类,重写run方法。
  • 然后创建这个子类对象,并调用start方法启动线程。
class MyThread extends Thread{
    @Override
    public void run() {
        System.out.println("hello,world..");
    }
}
public class ThreadDemo1 {
    public static void main(String[] args) {
        Thread t=new MyThread();
        t.start();
    }
}


用匿名内部类的方式:

public class ThreadDemo4 {
    public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(){
            @Override
            public void run() {
                    System.out.println("hello,world");
               }
        };
        t.start();
    }
}

2.实现Runnable接口(包含了匿名内部类的方式)

  • 通过实现Runnable,并实现run方法,也可以创建一个线程
  • 首先定义一个类实现Runnable接口,并实现run方法。
  • 然后创建Runnable实现类对象,并把它作为target传入Thread的构造函数中。
  • 最后调用start方法启动线程。
class MyThread3 implements Runnable {
    @Override
    public void run() {
        System.out.println("hello,线程");
    }
}
public class ThreadDemo3 {
    public static void main(String[] args) throws InterruptedException {

        Thread t= new Thread(new MyThread3());
        t.start();
    }
}

用匿名内部类的方式:

public class ThreadDemo5 {
    public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(new Runnable() {
            @Override
            public void run() {
                    System.out.println("hello,world");
            }
        });
        t.start();
    }
}

3.实现 Callable 接口,并结合 Future 实现

  • 首先定义一个 Callable 的实现类,并实现 call 方法。call 方法是带返回值的。
  • 然后通过 FutureTask 的构造方法,把这个 Callable 实现类传进去。
  • 把 FutureTask 作为 Thread 类的 target ,创建 Thread 线程对象。
  • 通过 FutureTask 的 get 方法获取线程的执行结果。
public class ThreadDemo36 {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Callable<Integer> callable=new Callable<Integer>() {
            @Override
            public Integer call() throws Exception {
                int result=0;
                for(int i=0;i<500;i++){
                    result++;
                }
                return result;
            }
        };
        FutureTask<Integer>futureTask=new FutureTask<>(callable);
        Thread t=new Thread(futureTask);
        t.start();
        System.out.println(futureTask.get());

    }
}

Runnable关注的是这个过程,不关注执行结果。Runnable提供的run方法,返回值类型是void

Callable要关注执行结果,Callable提供的call方法,返回值就是线程执行任务得到的结果

Callable是一个“锦上添花”的东西,它能干的事情,使用Runnable也能干。对于这种带有返回值的任务,使用Callable会更好一些。代码更直观,更简单。

4.基于lambda

使用java8的Lambda表达式创建一个匿名的Runnable对象作为线程

public class ThreadDemo6 {
    public static void main(String[] args) throws InterruptedException {
        Thread t=new Thread(()->{
               System.out.println("hello,world");
           }
        });
        t.start();
}
}

5.基于线程池

此处用 JDK 自带的 Executors 来创建线程池对象。

  • 首先,用一个 Runnable 的匿名内部类,重写 run 方法。
  • 然后创建一个拥有固定线程数的线程池。
  • 最后通过 ExecutorService 对象的 execute 方法传入线程对象。
public class ThreadDemo32 {
    public static void main(String[] args) {
        // 使用Executors工具类创建一个线程池,这里创建的是一个固定大小的线程池
        // newFixedThreadPool方法接受一个整数参数,表示线程池中的线程数量
        // 在本示例中,创建了一个包含4个线程的固定大小线程池
        ExecutorService service = Executors.newFixedThreadPool(4);
        // 向刚刚创建的线程池提交一个任务,任务以匿名内部类的形式实现了Runnable接口
        service.submit(new Runnable() {
            // 实现Runnable接口中的run方法,这个方法定义了任务具体要执行的内容
            @Override
            public void run() {
                System.out.println("hello");
            }
        });
        // 注意:这里缺少了关闭线程池的操作

    }
}

为什么说创建线程的方式只有一种也是可以的呢?

因为所有的方式的本质都是 new Thread()

这里不作过多解释,只因为博主也没有深入了解,所以不敢瞎说

大家可以去搜索一番

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值