java开启线程的四种方法

简介: 这篇文章介绍了Java中开启线程的四种方法,包括继承Thread类、实现Runnable接口、实现Callable接口和创建线程池,每种方法都提供了代码实现和测试结果。

文章目录

  • 1、继承Thread类
    • 1.1 代码实现
    • 1.2 测试结果
  • 2、实现Runnable接口
    • 2.1 方式一:直接实现Runnable接口
      • 2.1.1 代码实现
      • 2.1.2 测试结果
    • 2.2 方式二:匿名内部类
      • 2.2.1 代码实现
      • 2.2.2 测试结果
  • 3、实现Callable接口
    • 3.1 代码实现
    • 3.2 测试结果
  • 4、创建线程池
    • 4.1 代码实例
    • 4.2 测试结果

1、继承Thread类

/*
 * 创建步骤如下:
 * 1,定义Thread类的子类,并重写该类的run()方法,该run()方法的方法体就代表了线程需要完成的任务。因此把run方法称为线程执行体。
 * 2,创建Thread子类了的实例,即创建线程对象。本实例中是new一个ExtendThread,即可创建线程对象,也就是开启了一个线程
 * 3,调用线程对象的start()方法来启动该线程。
 *
 * 调用示例:
 * //循环10次即开启10个线程
 * for (int i = 0; i < 10; i++) {
 *     ExtendThread extendThread = new ExtendThread();
 *     extendThread.start();
 * }
 * */

1.1 代码实现

package com.zyz.mynative.demo03;

/**
 * @author zyz
 * @version 1.0
 * @data 2023/2/15 15:51
 * @Description:继承Thread类,重写run方法(不推荐,因为java的单继承局限性)
 */
public class ExtendThread extends Thread {

    public static void main(String[] args) {
        ExtendThread thread = new ExtendThread();
        thread.start();
    }

    /**
     * 重写Thread类的run(),这个方法称为线程执行体
     */
    @Override
    public void run() {
        doSomething();
    }

    /**
     * 需要处理的任务
     */
    public void doSomething() {
        for (int i = 0; i < 10; i++) {
            System.out.println(Thread.currentThread().getName() + "执行" + i);
        }
    }
}

1.2 测试结果

在这里插入图片描述


2、实现Runnable接口

2.1 方式一:直接实现Runnable接口

避免单继承的局限性,方便共享资源,推荐使用

/*
 * 创建步骤如下:
 * 1,定义Runnable接口的实现类,并且实现run方法,这个方法同样是线程执行体
 * 2,创建Runnable实现类的实例,并以此实例对象作为Thread的target来创建Thread类,这个新创建的Thread对象才是真正的线程对象,即开启了新的线程
 * 3,调用线程对象的start()方法来开启该线程
 *
 * 调用示例:
 * //开启10个线程
 * for (int i = 0; i < 10; i++) {
 *     Thread thread = new Thread(new RunnableImpl());
 *     thread.start();
 * }
 * */

2.1.1 代码实现

package com.zyz.mynative.demo03;

/**
 * @author zyz
 * @version 1.0
 * @data 2023/2/15 15:57
 * @Description:
 */
public class RunnableImpl implements Runnable {


    public static void main(String[] args) {
        RunnableImpl runnable = new RunnableImpl();
        Thread thread = new Thread(runnable);
        thread.start();

        /**
         * 简写
         * new Thread(runnable).start();
         */

    }

    /**
     * 实现Runnable接口的run方法,这个方法称为线程执行体
     * */
    @Override
    public void run() {
        doSomething();
    }

    /**
     * 需要处理的任务
     * */
    private void doSomething(){
        for (int i = 0; i < 10; i++) {
            System.out.println(Thread.currentThread().getName() + "执行" + i);
        }
    }
}

2.1.2 测试结果

在这里插入图片描述

2.2 方式二:匿名内部类

/*
 * 创建步骤如下:
 * 匿名内部类本质上也是一个类实现了Runnable接口,重写了run方法,只不过这个类没有名字,直接作为参数传入Thread类
 *
 * 调用示例:
 * //开启10个线程
 * for (int i = 0; i < 10; i++) {
 *     Anonymous anonymous =new Anonymous();
 *     anonymous.myRun();
 * }
 *
 * */

2.2.1 代码实现

package com.zyz.mynative.demo03;

/**
 * @author zyz
 * @version 1.0
 * @data 2023/2/15 15:57
 * @Description:
 */
public class RunnableImpl2 {


    public static void main(String[] args) {
        RunnableImpl2 test = new RunnableImpl2();
        test.myRun();
    }

    public void myRun(){
        new Thread(new Runnable() {
            @Override
            public void run() {
                doSomething();
            }
        }).start();
    }

    /**
     * 需要处理的任务
     * */
    private void doSomething(){
        for (int i = 0; i < 10; i++) {
            System.out.println(Thread.currentThread().getName() + "执行" + i);
        }
    }
}

2.2.2 测试结果

在这里插入图片描述


3、实现Callable接口

/*
 * 创建步骤如下:
 * 1,定义实现Callable<V>接口的实现类,实现call方法,这个方法是线程执行体
 * 2,创建Callable<V>实现类的实例,借助FutureTask得到线程执行的返回值
 * 3,将FutureTask的实例,作为Thread的target来创建Thread类
 * 4,调用start方法,开启线程
 *
 * 调用示例:
 * Callable<String> tc = new CallableImpl();
 * FutureTask<String> task = new FutureTask<>(tc);
 * new Thread(task).start();
 * try {
 *     System.out.println(task.get());
 * } catch (InterruptedException | ExecutionException e) {
 *     e.printStackTrace();
 * }
 *
 * 说明:
 * 1.与使用Runnable相比, Callable功能更强大些
 * 2.实现的call()方法相比run()方法,可以返回值
 * 3.方法可以抛出异常
 * 4.支持泛型的返回值
 * 5.需要借助FutureTask类,比如获取返回结果
 * Future接口可以对具体Runnable、Callable任务的执行结果进行取消、查询是否完成、获取结果等。
 * FutureTask是Futrue接口的唯一的实现类
 * FutureTask 同时实现了Runnable, Future接口。它既可以作为Runnable被线程执行,又可以作为Future得到Callable的返回值
 *
 * */

3.1 代码实现

package com.zyz.mynative.demo03;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

/**
 * @author zyz
 * @version 1.0
 * @data 2023/2/15 16:08
 * @Description:
 */
public class CallableImpl implements Callable<String> {

    public static void main(String[] args) {
        Callable<String> tc = new CallableImpl();
        FutureTask<String> task = new FutureTask<>(tc);
        new Thread(task).start();
        try {
            System.out.println(task.get());
        } catch (InterruptedException | ExecutionException e) {
            e.printStackTrace();
        }
    }

    private int ticket = 5;

    @Override
    public String call() throws Exception {
        for (int i = 0; i < 10; i++) {
            System.out.println(doSomething());
        }

        return "出票任务完成";
    }

    public String doSomething() {
        String result = "";
        if (this.ticket > 0) {
            result = "出票成功,ticket=" + this.ticket--;
        } else {
            result = "出票失败,ticket=" + this.ticket;
        }
        return result;
    }
}

3.2 测试结果

在这里插入图片描述


4、创建线程池

/*
 * 创建步骤如下:
 * 1,定义Runnable接口的实现类,或者定义(继承Runnable接口的类)的实现类,并且实现run方法,这个方法是线程执行体
 * 2,创建一个自定义线程个数的线程池
 * 3,实例化Runnable接口的实现类
 * 4,将3步的实例,作为线程池实例的execute方法的command参数,开启线程
 * 5,关闭线程池
 *
 * 调用示例:
 * ExecutorService pool = Executors.newFixedThreadPool(2);
 * ThreadPool threadPool = new ThreadPool("AA");
 * ThreadPool threadPoo2 = new ThreadPool("BB");
 * pool.execute(threadPool);
 * pool.execute(threadPoo2);
 * pool.shutdown();
 *
 * 说明:
 * 示例中创建的是2个线程的线程池
 * execute方法是开启线程方法,实参要求是实现Runnable的类。所以,继承Thread类的子类也可以以线程池的方式开启线程
 *
 * */

4.1 代码实例

package com.zyz.mynative.demo03;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * @author zyz
 * @version 1.0
 * @data 2023/2/15 16:11
 * @Description:
 */
public class ThreadPool implements Runnable {



    public static void main(String[] args) {
      ExecutorService pool = Executors.newFixedThreadPool(2);
      ThreadPool threadPool = new ThreadPool("AA");
      ThreadPool threadPoo2 = new ThreadPool("BB");
      pool.execute(threadPool);
      pool.execute(threadPoo2);
      pool.shutdown();
    }

    String name;
    public ThreadPool(String name) {
        this.name = name;
    }

    @Override
    public void run() {
        doSomething();
    }

    /**
     * 需要处理的任务
     * */
    private void doSomething() {
        for (int i = 0; i < 10; i++) {
            System.out.println(Thread.currentThread().getName() + "执行" + i + ",name=" + this.name);
        }
    }
}

4.2 测试结果

在这里插入图片描述

资料参考:创建线程池的实现方法

相关文章
|
5天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
44 17
|
16天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
1天前
|
缓存 安全 算法
Java 多线程 面试题
Java 多线程 相关基础面试题
|
18天前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。
|
18天前
|
消息中间件 缓存 安全
Java多线程是什么
Java多线程简介:本文介绍了Java中常见的线程池类型,包括`newCachedThreadPool`(适用于短期异步任务)、`newFixedThreadPool`(适用于固定数量的长期任务)、`newScheduledThreadPool`(支持定时和周期性任务)以及`newSingleThreadExecutor`(保证任务顺序执行)。同时,文章还讲解了Java中的锁机制,如`synchronized`关键字、CAS操作及其实现方式,并详细描述了可重入锁`ReentrantLock`和读写锁`ReadWriteLock`的工作原理与应用场景。
|
18天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
42 3
|
18天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
105 2
|
26天前
|
安全 Java API
java如何请求接口然后终止某个线程
通过本文的介绍,您应该能够理解如何在Java中请求接口并根据返回结果终止某个线程。合理使用标志位或 `interrupt`方法可以确保线程的安全终止,而处理好网络请求中的各种异常情况,可以提高程序的稳定性和可靠性。
48 6
|
1月前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
16天前
|
缓存 安全 Java
【JavaEE】——单例模式引起的多线程安全问题:“饿汉/懒汉”模式,及解决思路和方法(面试高频)
单例模式下,“饿汉模式”,“懒汉模式”,单例模式下引起的线程安全问题,解锁思路和解决方法