大家都说Java有三种创建线程的方式,并发编程中的惊天骗局

news2024/11/25 12:58:52

在Java中,创建线程是一项非常重要的任务。线程是一种轻量级的子进程,可以并行执行,使得程序的执行效率得到提高。Java提供了多种方式来创建线程,但许多人都认为Java有三种创建线程的方式,它们分别是继承Thread类、实现Runnable接口和使用线程池。

但是,你们知道吗?其实在创建线程的过程中,除了上述描述的方法还有很多种方式可以选择哦。今天,我们就来揭开这个惊天秘密,一起来了解一下Java并发编程中创建线程的八股文。

一. 创建线程的方法:

1. 继承Thread类创建线程

这是最基本的创建线程的方式,我们可以通过继承Thread类来创建一个自定义的线程类,然后重写run()方法,实现线程的逻辑。

public class MyThread extends Thread {
    @Override
    public void run() {
        // 线程逻辑
    }
}

// 创建并启动线程
MyThread myThread = new MyThread();
myThread.start();

2. 实现Runnable接口创建线程

这是另一种常见的创建线程的方式,我们可以通过实现Runnable接口来创建一个自定义的线程类,然后将该类实例化并传递给Thread类的构造方法中,最后调用start()方法启动线程。

public class MyRunnable implements Runnable {

@Override

public void run() {

// 线程逻辑

}

}

// 创建并启动线程

MyRunnable myRunnable = new MyRunnable();

Thread thread = new Thread(myRunnable);

thread.start();

3. 实现Callable接口创建线程

Callable接口与Runnable接口类似,但是它可以返回一个结果并抛出异常。我们可以通过实现Callable接口来创建一个自定义的线程类,然后将该类实例化并传递给FutureTask类的构造方法中,最后调用start()方法启动线程。

public class MyCallable implements Callable {

@Override

public String call() throws Exception {

// 线程逻辑

return "result";

}

}

// 创建并启动线程

MyCallable myCallable = new MyCallable();

FutureTask futureTask = new FutureTask<>(myCallable);

Thread thread = new Thread(futureTask);

thread.start();

// 获取线程返回结果

String result = futureTask.get();

4. 使用线程池创建线程

线程池是一种重用线程的机制,可以减少线程创建和销毁的开销。我们可以通过Executors类提供的静态方法来创建不同类型的线程池,然后将任务提交给线程池执行。

ExecutorService executorService = Executors.newFixedThreadPool(10);

// 提交任务并执行

executorService.submit(new Runnable() {

@Override

public void run() {

// 线程逻辑

}

});

// 关闭线程池

executorService.shutdown();

5. 使用定时器创建线程

定时器可以用来定时执行某个任务。我们可以通过Timer类来创建一个定时器,然后将任务添加到定时器中执行。
请添加图片描述

6. 使用ScheduledExecutorService创建线程

ScheduledExecutorService是一种可以调度任务执行的线程池。我们可以通过它来创建一个定时任务,也可以创建一个周期性任务。

ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(10);

// 创建定时任务并执行
scheduledExecutorService.schedule(new Runnable() {
    @Override
    public void run() {
        // 线程逻辑
    }
}, 1, TimeUnit.SECONDS);

// 创建周期性任务并执行
scheduledExecutorService.scheduleAtFixedRate(new Runnable() {
    @Override
    public void run() {
        // 线程逻辑
    }
}, 1, 1, TimeUnit.SECONDS);

// 关闭线程池
scheduledExecutorService.shutdown();

7. 使用Fork/Join框架创建线程

Fork/Join框架是Java 7中引入的一种并行执行任务的机制。它可以将一个大任务拆分成多个小任务并行执行,最后将结果合并。

public class MyTask extends RecursiveTask {
    private int start;
    private int end;

    public MyTask(int start, int end) {
        this.start = start;
        this.end = end;
    }

    @Override
    protected Integer compute() {
        if (end - start <= 1000) {
            // 执行小任务
            return 0;
        } else {
            // 拆分大任务
            int mid = (start + end) / 2;
            MyTask leftTask = new MyTask(start, mid);
            MyTask rightTask = new MyTask(mid + 1, end);
            leftTask.fork();
            rightTask.fork();

            // 合并结果
            return leftTask.join() + rightTask.join();
        }
    }
}

// 创建并执行任务
ForkJoinPool forkJoinPool = new ForkJoinPool();
MyTask myTask = new MyTask(1, 10000);
int result = forkJoinPool.invoke(myTask);

8. 使用Semaphore创建线程

Semaphore是一种计数器,用来控制同时访问某个资源的线程数量。我们可以通过Semaphore来创建一个有限的线程池。

Semaphore semaphore = new Semaphore(10);

// 创建并执行任务
Runnable runnable = new Runnable() {
    @Override
    public void run() {
        try {
            semaphore.acquire();
            // 线程逻辑
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            semaphore.release();
        }
    }
};

for (int i = 0; i < 100; i++) {
    new Thread(runnable).start();
}

二. 线程和线程体的关系

说到线程大家都知道,它是一种重要的多任务处理机制。可以让我们同时执行多个任务,并且可以提高程序的效率。 上述也给大家带来了很多种创建线程的方式,但是说到这里,我发现经常有朋友问:线程体和线程是什么关系?其实,简单说:线程体就是是线程的具体执行代码。那么接下来就让我们具体来看看线程和线程体的关系吧!

1. 线程和线程体的关系

在Java中,线程和线程体是两个不同的概念。

线程是一条执行路径,线程体是线程的具体执行代码。 每个线程都有一个与之相关的线程体。线程体是实现了Runnable接口的类的实例。线程体可以是一个独立的类,也可以是一个内部类。线程创建之后,它的run()方法就会被调用,run()方法中的代码就是线程体的执行代码。

2. 案例说明

可能会有朋友觉得上述的解释过于书面化,那么如果通过生活实例来说明线程体和线程的关系的话,我们可以将线程理解成为一个人,线程体则是这个人所做的事情。

比如,在上班的路上,我们可以同时进行多个任务,比如听音乐、看书、发短信等等。这些任务就可以看做是这个人的线程体。线程就是这个人同时进行多个任务的机制。如果我们只有一个人,那么这个人必须先做完一个任务才能开始做下一个任务。这样就会浪费很多时间。但是如果我们有两个人,那么一个人可以看书,另一个人可以听音乐,这样就可以同时进行多个任务,提高了效率。

3. 代码案例

下面给大家带来一个:创建多个线程来同时执行多个任务的简单的代码案例,示例如下:

public class MyThread implements Runnable {
    private String taskName;

    public MyThread(String taskName) {
        this.taskName = taskName;
    }

    @Override
    public void run() {
        for (int i = 0; i < 5; i++) {
            System.out.println(taskName + "执行了第" + (i + 1) + "次任务");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) {
        MyThread thread1 = new MyThread("线程1");
        MyThread thread2 = new MyThread("线程2");
        Thread t1 = new Thread(thread1);
        Thread t2 = new Thread(thread2);
        t1.start();
        t2.start();
    }
}

在这个案例中,我们创建了两个线程,分别执行不同的任务。每个线程的线程体是MyThread类的run()方法。在run()方法中,每个线程会执行5次任务,并在每次任务之间暂停1秒钟。通过创建多个线程,我们可以同时执行多个任务,提高程序的效率。

4. 小结

通过上述介绍的Java线程和线程体的关系,大家应该都清楚了。线程是一种多任务处理机制,线程体是线程的具体执行代码。在Java中,我们可以通过创建多个线程来同时执行多个任务,提高程序的效率。希望上述可以帮助大家更好地理解Java中线程和线程体的概念哦。

三. 总结

现在大家都知道了在Java并发编程中,创建线程是一个非常重要的操作。

本文给大家介绍了八种不同的创建线程的方式,包括继承Thread类、实现Runnable接口、实现Callable接口、使用线程池、使用定时器、使用ScheduledExecutorService、使用Fork/Join框架和使用Semaphore。

每种方式都有各自的优缺点,我们需要根据实际情况选择合适的方式来创建线程。同时,我们还需要注意线程安全和性能等问题,确保程序的正确性和高效性。

在生活中,我们也可以找到很多与多线程编程相关的例子。比如,在厨房里做饭的时候,我们可以将不同的任务分配给不同的人来完成,比如一个人负责洗菜,一个人负责切菜,一个人负责烧菜等等。这样可以提高效率,也可以避免出现混乱和错误。同样,在程序中,我们也可以将不同的任务分配给不同的线程来完成,以提高程序的响应速度和吞吐量。因此,多线程编程是非常重要的,值得我们深入学习和掌握。希望能对大家有所帮助哦。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/630172.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

论文浅尝 | Dually Distilling KGE for Faster and Cheaper Reasoning

笔记整理&#xff1a;张津瑞&#xff0c;天津大学硕士&#xff0c;研究方向为知识图谱 链接&#xff1a;https://dl.acm.org/doi/10.1145/3488560.3498437 动机 知识图谱已被证明可用于各种 AI 任务&#xff0c;如语义搜索&#xff0c;信息提取和问答等。然而众所周知&#xff…

【C++】C++11常用新特性

✍作者&#xff1a;阿润菜菜 &#x1f4d6;专栏&#xff1a;C 目录 一、统一的列表初始化二、 简化声明2.1 auto2.2 decltype2.3 nullptr 三、右值引用和移动语义 -- 重要3.1 区分左值引用和右值引用3.2 对比左值引用看看右值引用使用价值3.3 万能引用和完美转发&#xff08;st…

基于word文档,使用Python输出关键词和词频,并将关键词的词性也标注出来

点击上方“Python爬虫与数据挖掘”&#xff0c;进行关注 回复“书籍”即可获赠Python从入门到进阶共10本电子书 今 日 鸡 汤 移船相近邀相见&#xff0c;添酒回灯重开宴。 大家好&#xff0c;我是Python进阶者。 一、前言 前几天在有个粉丝问了个问题&#xff0c;大概意思是这样…

一道北大强基题背后的故事(三)——什么样的题是好题?

早点关注我&#xff0c;精彩不错过&#xff01; 上回我们针对这道北大强基题[((1 sqrt(5)) / 2) ^ 12]在答案的基础上给出了出题的可能思路&#xff0c;想一探究竟&#xff0c;相关内容请戳&#xff1a; 一道北大强基题背后的故事&#xff08;二&#xff09;——出题者怎么想的…

【Kubernetes入门】Service四层代理入门实战详解

文章目录 一、Service四层代理概念、原理1、Service四层代理概念2、Service工作原理3、Service原理解读4、Service四种类型 二、Service四层代理三种类型案例1、创建ClusterIP类型Service2、创建NodePort类型Service3、创建ExternalName类型Service 三、拓展1、Service域名解析…

Latex中图片排版(多个子图、横排、竖排、添加小标题)

1、两个子图横排 \begin{figure}[!t] \centering %\includegraphics[width3in]{fig5} \subfloat[subfig figure title]{\includegraphics[scale0.5]{superd2}} \subfloat[subfig figure title]{\includegraphics[scale0.5]{superd2}} \caption{title} \label{fig_6} \end{figu…

阿里发布的百亿级高并发系统(全彩版小册),涵盖了所有的高并发操作

高并发 提到“高并发”相信你们应该都不会感到陌生&#xff01;此时你脑中应该会浮现好多有关高并发的&#xff1a;业务急剧增长、电商购物、电商秒杀、12306抢票、淘宝天猫各种活动等&#xff1b;都是需要用到高并发的&#xff0c;那么如何去设计一个高并发系统抵挡这些冲击呢…

Django的app里面的视图函数

我之前说过需要重点去了解view和model&#xff0c;下面是我的总结。 视图函数是存在view.py里面的&#xff0c;视图函数的主要功能是接收请求、返回响应。在建立应用程序后&#xff0c;先在URL配置文件中加一条配置项指明URL与视图函数的对应关系。然后按照实际需求在视图函数…

三次握手四次挥手过程剖析

【一】预备知识&#xff1a; 1.三次握手并不一定非得成功&#xff0c;最担心得其实就是最后一个akc&#xff08;应答&#xff09;丢失&#xff0c;但是还是有配套得解决方案&#xff0c;比如超时重传机制。 2.连接是需要被保存下来得&#xff0c;是需要被os管理起来得&#xf…

被一个gpio口搞死的一天

今天是新项目调试的第一天。 我起的很早&#xff0c;起早的原因很简单&#xff0c;我家楠哥要我送他上学&#xff0c;他说爸爸没有起到一个当爸爸的责任&#xff0c;他也想让爸爸送他上学&#xff0c;然后我就送了。 7点30起来&#xff0c;8点出发&#xff0c;然后回来看了一下…

IDEA把css/js压缩成一行min文件,idea实现右键压缩css和js文件

前言 发布时有些css和js文件较长多行&#xff0c;导致加载的时候略慢&#xff0c;所以想把指定的css或js压缩 实现 整合 yuicompressor-2.4.8.jar 下载地址1&#xff1a;https://github.com/yui/yuicompressor/releases 下载地址2&#xff1a;https://github.com/yui/yuicom…

小学生开“卷”AIGC,绝不能输在起跑线上

图片来源&#xff1a;由无界AI生成 OpenAI的研究报告称&#xff0c;未来&#xff0c;大量工作岗位将受到AI冲击&#xff0c;首当其冲的岗位是作家、数学家、网页设计师、记者、律师…… 自从ChatGPT问世以来&#xff0c;人类会被AI替代的讨论甚嚣尘上&#xff0c;焦虑情绪无处不…

chatgpt赋能python:Python中的倒序遍历:如何使用Python倒序遍历?

Python中的倒序遍历&#xff1a;如何使用Python倒序遍历&#xff1f; Python是一种高级编程语言&#xff0c;它非常适合数据科学、机器学习和人工智能等领域。Python的强大之处在于它有很多内置功能&#xff0c;其中包括倒序遍历。 在本篇文章中&#xff0c;我们将介绍如何使…

EmbodiedGPT|具身智能或将成为实现AGI的最后一公里

卷友们好&#xff0c;我是穆尧。 最近由Chatgpt所引爆的新一代人工智能的革命正在如火如荼的进行&#xff0c;几乎重塑了所有的互联网产品&#xff0c;如办公软件、浏览器插件、搜索引擎、推荐系统等。这样巨大的改变&#xff0c;让大家对通用人工智能又燃起了新的希望&#xf…

CTPN文本检测详解 面试版本

二.关键idea 1.采用垂直anchor回归机制&#xff0c;检测小尺度的文本候选框 2.文本检测的难点在于文本的长度是不固定&#xff0c;可以是很长的文本&#xff0c;也可以是很短的文本&#xff0e;如果采用通用目标检测的方法&#xff0c;将会面临一个问题&#xff1a;**如何生成好…

Autosar诊断实战系列01-手把手教你增加一路31Routine服务

本文框架 1.系列概述2. UDS Routine服务添加3. DcmDspRoutine配置3.1 DcmDspRoutineInfos配置3.2 DcmDspRoutines配置1.系列概述 在本系列笔者将结合工作中对诊断实战部分的应用经验进一步介绍常用UDS服务的进一步探讨及开发中注意事项, Dem/Dcm/CanTp/Fim模块配置开发及注意…

编译tolua——2、基础编译tolua

目录 1、编译工具和环境说明 2、基础编译tolua 大家好&#xff0c;我是阿赵。 继续来讲tolua的各个常用平台的编译。 这里使用官方的tolua_runtime-master项目来做编译 具体需要的编译软件和源码地址&#xff0c;在上一篇文章已经介绍过了&#xff0c;先把环境准备好&#xff…

飞桨AI4S污染物扩散快速预测模型,亮相全国数据驱动计算力学研讨会

5月19-21日&#xff0c;第一届全国数据驱动计算力学研讨会在大连召开。本次研讨会由中国力学学会主办&#xff0c;大连理工大学运载工程与力学学部承办&#xff0c;北京理工大学先进结构技术研究院协办。 会议共吸引了400多位来自全国各地高校与企业的老师与学生参会&#xff0…

DNSPod十问林洪祥:顶级带货主播,其实是数字人?

本期嘉宾 林洪祥 风平智能CEO 林洪祥&#xff0c;风平智能CEO。风平智能拥有全球领先的数字人AIGC预训练大模型技术&#xff0c;利用数字人AI知识大模型打造视频版ChatGPT&#xff0c;实现数字人名师、数字人医生、数字人保险客服、数字广告模特、数字人AI直播等&#xff0c;…

百度大模型算法实习岗上岸经验!

Datawhale干货 作者&#xff1a;J同学&#xff0c;学校&#xff1a;天津大学 写在前面 大家好&#xff0c;我是天大J同学&#xff0c;2023找暑期实习期间也算是历经坎坷&#xff0c;最后去了百度做大模型相关工作&#xff0c;虽然本硕均为计算机科班但本身实力尚浅&#xff0c…