Java基础系列-第一章 创建Thread的6种方式和线程常用方法

简介: 读完本章节,您将掌握如何创建线程和线程的常用方法。

前言

读完本章节,您将掌握如何创建线程和线程的常用方法。

一、创建线程

1. 通过Thread子类创建线程

   public class MyThread extends Thread {
        @Override
        public void run() {
            System.out.println("通过Thread子类创建线程");
        }
    }

    //通过Thread子类创建线程
    public void createThread1() {
        MyThread t = new MyThread();
        t.start();
    }

2. 使用runnable创建线程

2.1 通过函数式接口创建线程
    //使用runnable创建线程-函数式接口
    public void createThread2() {
        Thread t = new Thread(() -> System.out.println("使用runnable创建线程1"));
        t.start();
    }
2.2 通过内部匿名类创建线程
    //使用runnable创建线程-内部匿名类
    public void createThread3() {
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("使用runnable创建线程2");
            }
        });
        t.start();
    }
2.3 通过runnable子类创建线程
    public class MyRunnable implements Runnable {
        @Override
        public void run() {
            System.out.println("使用runnable创建线程3");
        }
    }

    //使用runnable创建线程-runnable子类
    public void createThread4() {
        MyRunnable runnable = new MyRunnable();
        Thread t = new Thread(runnable);
        t.start();
    }

3. 使用匿名内部类创建线程

    //使用匿名内部类创建线程
    public void createThread5() {
        Thread t = new Thread() {
            @Override
            public void run() {
                System.out.println("使用匿名内部类创建线程");
            }
        };
        t.start();
    }

4. 使用线程池创建线程

    @Resource
    private ThreadPoolExecutor threadPoolExecutor;

    @Bean
    public ThreadPoolExecutor executor() {
        int processors = Runtime.getRuntime().availableProcessors();
        return new ThreadPoolExecutor(
                Math.max(processors * 4, 6),
                Math.max(processors * 4, 8),
                0L,
                TimeUnit.MILLISECONDS,
                new LinkedBlockingDeque<>(100000),
                runnable -> {
                    Thread r = new Thread(runnable);
                    r.setName("thread-");
                    return r;
                }
        );
    }

    //使用线程池创建线程
    public void createThread6() {
        threadPoolExecutor.execute(() -> System.out.println("使用线程池创建线程"));
    }

二、线程常用静态方法

    //线程常用静态方法
    public void threadStaticMethod() throws InterruptedException {
        //将当前线程设为让步状态,表示当前线程愿意让出CPU资源使得其他线程执行。
        //将当前线程设为待运行就绪状态,再从线程池中取出可运行的线程执行后续逻辑,有可能取到刚放回去的当前线程。
        Thread.yield();
        //线程休眠(阻塞)100ms
        Thread.sleep(100L);
        //获取当前线程的信息
        Thread.currentThread();
        //返回当前线程的线程组中活动线程的数量。(返回的值只是一个估计值)
        Thread.activeCount();
        //获取线程中断标志位
        Thread.interrupted();
    }

三、线程实例常用方法

    //线程实例常用方法
    public void threadMethod() throws InterruptedException {
        Thread t = new Thread(() -> System.out.println("线程实例常用方法"));

        //当前线程等待t线程执行直到t执行完成
        //要让join方法正常生效,调用join方法的线程对象必须已经调用了start()方法并且未进入终止状态。
        //常见面试题:现在有T1、T2,你怎样保证T2在T1执行完之后执行,就可以在T2中调用t1.join()实现。
        t.join();

        //挂起当前线程和释放synchronized同步锁
        //wait()和notify()、notifyAll()都是java.lang.Object的方法。
        t.wait();

        //设为后台线程(守护线程)
        //java线程分为后台线程与前台线程,其中后台线程不会影响到进程的退出,而前台线程会影响进程的退出。
        //比如有线程t1与线程t2,当这两个线程为前台线程时,main方法执行完毕时,t1与t2不会立即退出,要等到线程执行完毕,整个进程才会退出,反之,当这两个线程为后台线程时,main方法执行完毕时,t1与t2线程被强制结束,整个进程也就结束了。
        t.setDaemon(true);

        //使用interrupt方法修改线程中断标志位为true,默认为false
        t.interrupt();

        //获取线程中断标志位
        //在线程run()方法里可以通过Thread.currentThread().isInterrupted()获取标志位。但是调用interrupt方法遇到线程run()方法里调用 wait/join/sleep 等方法而阻塞线程时会使sleep等方法抛出异常,并且中断标志位不会修改为true
        //可通过静态方法Thread.interrupted()代替,但注意:一个程序里Thread.interrupted()是共用的。
        t.isInterrupted();

        //以同步方式执行方法,不创建新线程
        t.run();

        //创建线程并执行run方法
        t.start();
    }

四、线程基本状态

NEW: 初始化状态。初始化线程对象,这时没有执行start方法,java内部的状态,与进程中的状态无关。
RUNNABLE: 就绪状态。分两种:RUNNABLE运行中,READY待运行。
BLOCKED: 阻塞状态。线程正在等待锁释放而引起的阻塞状态(synchronized加锁)。
WAITING: 等待状态。线程正在等待等待唤醒而引起的阻塞状态(wait方法使线程等待唤醒)。
TIMED_WAITING: 超时等待状态。在一段时间内处于阻塞状态,通常是使用sleep或者join(带参数)方法引起。
TERMINATED:终止状态。Thread对象还存在,但是关联的线程已经工作完成了,java内部的状态,与进程中的状态无关。

在这里插入图片描述

总结

读完本章节,您已经掌握如何创建线程和线程的常用方法。可以动手按照示例代码自己敲下进行测试和巩固。

目录
相关文章
|
7天前
|
存储 Java 程序员
优化Java多线程应用:是创建Thread对象直接调用start()方法?还是用个变量调用?
这篇文章探讨了Java中两种创建和启动线程的方法,并分析了它们的区别。作者建议直接调用 `Thread` 对象的 `start()` 方法,而非保持强引用,以避免内存泄漏、简化线程生命周期管理,并减少不必要的线程控制。文章详细解释了这种方法在使用 `ThreadLocal` 时的优势,并提供了代码示例。作者洛小豆,文章来源于稀土掘金。
|
14天前
|
算法 安全 Java
三种方法教你实现多线程交替打印ABC,干货满满!
本文介绍了多线程编程中的经典问题——多线程交替打印ABC。通过三种方法实现:使用`wait()`和`notify()`、`ReentrantLock`与`Condition`、以及`Semaphore`。每种方法详细讲解了实现步骤和代码示例,帮助读者理解和掌握线程间的同步与互斥,有效解决并发问题。适合不同层次的开发者学习参考。
35 11
|
9天前
|
Java Spring
运行@Async注解的方法的线程池
自定义@Async注解线程池
34 3
|
12天前
|
Java
用JAVA架建List集合为树形结构的代码方法
这段代码定义了一个表示树形结构的 `Node` 类和一个用于构建树形结构的 `TreeController`。`Node` 类包含基本属性如 `id`、`pid`、`name` 和 `type`,以及子节点列表 `children`。`TreeController` 包含初始化节点列表并将其转换为树形结构的方法。通过过滤和分组操作实现树形结构的构建。详情可见:[代码示例链接1](http://www.zidongmutanji.com/zsjx/43551.html),[代码效果参考链接2](https://www.257342.com/sitemap/post.html)。
25 5
|
15天前
|
安全 Java 开发者
【技术咖必看】Java异常处理新境界:throws关键字,打造万无一失的方法签名!
【技术咖必看】Java异常处理新境界:throws关键字,打造万无一失的方法签名!
29 3
|
11天前
|
Java 开发者
探索Java中的Lambda表达式:简化代码的现代方法
【8月更文挑战第31天】Lambda表达式在Java 8中首次亮相,为Java开发者提供了一种更简洁、灵活的编程方式。它不仅减少了代码量,还提升了代码的可读性和可维护性。本文将通过实际示例,展示Lambda表达式如何简化集合操作和事件处理,同时探讨其对函数式编程范式的支持。
|
16天前
|
存储 监控 Java
Java多线程优化:提高线程池性能的技巧与实践
Java多线程优化:提高线程池性能的技巧与实践
43 1
|
8天前
|
存储 Ubuntu Linux
C语言 多线程编程(1) 初识线程和条件变量
本文档详细介绍了多线程的概念、相关命令及线程的操作方法。首先解释了线程的定义及其与进程的关系,接着对比了线程与进程的区别。随后介绍了如何在 Linux 系统中使用 `pidstat`、`top` 和 `ps` 命令查看线程信息。文档还探讨了多进程和多线程模式各自的优缺点及适用场景,并详细讲解了如何使用 POSIX 线程库创建、退出、等待和取消线程。此外,还介绍了线程分离的概念和方法,并提供了多个示例代码帮助理解。最后,深入探讨了线程间的通讯机制、互斥锁和条件变量的使用,通过具体示例展示了如何实现生产者与消费者的同步模型。
|
16天前
|
监控 安全 Java
Java多线程调试技巧:如何定位和解决线程安全问题
Java多线程调试技巧:如何定位和解决线程安全问题
66 2