在Java中创建线程有三种方式:

  1. 继承 Thread 类
  2. 实现 Runnable 接口
  3. 实现 Callable 接口

继承 Thread 类创建线程

步骤:

1. 自定义线程类继承 Thread 类
2. 重写 run() 方法,编写线程执行体
3. 创建线程对象,调用 start() 方法启动线程

// 创建线程方式一:继承Thread类
//线程开启不一定马上执行,由CPU调度来调度执行
public class TestThread1 extends Thread {

    //重写 run 方法,调用start方法开启线程
    @Override
    public void run() {
        for (int i = 0; i < 20; i++) {
            System.out.println("我在看代码!===="+i);
        }
    }

    public static void main(String[] args) {
        //main线程,主线程

        //创建一个线程对象
        TestThread1 testThread1 = new TestThread1();

        //调用start() 方法开启线程
        testThread1.start();

        for (int i = 0; i < 20; i++) {
            System.out.println("我在学习多线程===="+i);
        }
    }
}

实现 Runnable 接口创建线程

步骤:

1. 自定义线程类实现 Runnable 接口
2. 实现 run() 方法
3. 通过 new Thread(线程对象).start() 启动线程

// 创建线程方式二:实现 Runnable 接口,重写run方法,执行线程需要丢入runnable接口的实现类,调用start方法
public class TestThread3 implements Runnable {

    @Override
    public void run() {
        for (int i = 0; i < 20; i++) {
            System.out.println("我在看代码!===="+i);
        }
    }

    public static void main(String[] args) {
        //main线程,主线程

        //创建runnable接口实现类
        TestThread3 testThread3 = new TestThread3();
        //创建线程对象,通过线程对象来开启我们的线程,代理模式
        new Thread(testThread3).start();

        for (int i = 0; i < 20; i++) {
            System.out.println("我在学习多线程===="+i);
        }
    }
}

继承 Thread 和实现 Runnable 接口对比

- 继承Thread类
  - 子类继承Thread类具备多线程能力
  - 启动线程:子类对象.start()
  - 不建议使用:避免OOP单继承局限性
- 实现Runnable接口
  - 实现接口Runnable具有多线程能力
  - 启动线程:传入目标对象 + Thread对象.start()
  - 推荐使用:避免单继承局限性,灵活方便,方便同一个对象被多个线程使用

实现 Callable 接口创建线程

步骤:

1. 创建线程类实现 Callable 接口,需要返回值类型
2. 重写 call 方法,需要抛出异常
3. 创建目标对象
4. 创建执行服务: ExecutorService ser = Executors.newFixedThreadPool(1);
5. 提交执行: Future<Boolean> result1 = ser.submit(t1);
6. 获取结果:boolean r1 = result1.get()
7. 关闭服务:ser.shutdownNow()

// 线程创建方式三:实现callable接口
/*
* callable的好处
* 1、可以定义返回值
* 2、可以抛出异常
* */
@AllArgsConstructor
@NoArgsConstructor
public class TestCallable implements Callable<Boolean> {

    private String url;  //网络图片地址
    private String fileName; //保存的文件名



    @Override
    public Boolean call() throws Exception {
        WebDownLoader webDownLoader = new WebDownLoader();
        webDownLoader.downloader(url,fileName);
        System.out.println("下载的文件名为:"+fileName);
        return true;
    }

    @SneakyThrows
    public static void main(String[] args) {
        String url = "https://upload-images.jianshu.io/upload_images/1689841-f622a468b2694253.png?imageMogr2/auto-orient/strip%7CimageView2/2";
        //创建多个线程
        TestCallable t1 = new TestCallable(url,"1.jpg");
        TestCallable t2 = new TestCallable(url,"2.jpg");
        TestCallable t3 = new TestCallable(url,"3.jpg");

        //1. 创建执行服务:
        ExecutorService ser = Executors.newFixedThreadPool(3);
        //2. 提交执行:
        Future<Boolean> r1 = ser.submit(t1);
        Future<Boolean> r2 = ser.submit(t2);
        Future<Boolean> r3 = ser.submit(t3);
        //3. 获取结果:
        boolean rs1 = r1.get();
        boolean rs2 = r2.get();
        boolean rs3 = r3.get();
        //4. 关闭服务:
        ser.shutdownNow();
    }


    //下载器
    class WebDownLoader{
        //下载方法
        public void downloader(String url,String fileName){
            try {
                FileUtils.copyURLToFile(new URL(url),new File(fileName));
            } catch (IOException e) {
                e.printStackTrace();
                System.out.println("IO异常,downloader方法出现问题");
            }
        }
    }
}

总结三个创建线程的方式!!~.~