【Java系列】详解多线程(二)——Thread类及常见方法(上篇)

这篇具有很好参考价值的文章主要介绍了【Java系列】详解多线程(二)——Thread类及常见方法(上篇)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

个人主页:兜里有颗棉花糖
欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创
收录于专栏【Java系列专栏】【JaveEE学习专栏】
本专栏旨在分享学习Java的一点学习心得,欢迎大家在评论区交流讨论💌

一、前文回顾

我们先来回顾一下线程与进程之间的联系。

我们知道多进程可以帮助我们完成并发编程,即可以把多个cpu核心充分利用起来以完成同时执行多任务的场景。但是进程有一个问题就是进程的创建和销毁的开销是比较大的,如果我们需要频繁的创建和销毁进程的话,那么多进程这种方式就是比较低效的。所以就衍生出了轻量级进程——线程。线程之所以比进程更轻量级主要有两方面的原因

  • 一方面是线程共享进程的资源,不要忘记,线程是进程的一部分,线程可以共享那些属于进程的资源,但是进程之间是相对独立的,线程创建时不需要像进程那样需要分配独立的资源,因此线程创建和销毁的开销会比进程小很多
  • 另一方面线程调度切换要比进程调度切换要快上很多,由于多线程使用的调度算法比单进程要复杂很多,加上线程对系统资源的共享(线程在同一个进程内共享资源,线程切换只需要切换线程的上下文,而不需要切换整个进程的上下文;而相比之间进程切换需要保存和恢复进程的所有上下文信息,包括内存映像、打开的文件、进程状态等。),所以线程调度之间的切换要比进程调度的切换快上很多,也更容易实现多任务的处理。

线程在同一进程内共享内存资源,所以当进程创建完成时,线程已经可以访问和共享这些内存资源(主线程是在进程创建时默认创建的一个线程)。如果后续需要我们创建其它的线程,我们就可以重用之前的字眼就可以了。

进程包含了线程,一个进程内部至少有一个线程,进程是系统中分配资源的基本单位,而线程是cpu调度执行的基本单位。我们可以这样说线程的引入最主要的就是为了实现更高效的并发处理和资源共享。

由于线程可以共享同一进程内的内存资源,因此资源分配的主要任务是为进程分配足够的内存空间,对线程的资源分配比在进程级别上的资源分配要少得多。

现在问题来了,一个进程内既然可以存在多个线程,同时线程可以共享同一个进程内的资源,这也意味着这多个线程之间的相互干扰会比较大(这一点就不如进程之间那样相对独立一些,一个进程挂掉知道后不会对其它进程产生影响。),一旦某个出现异常之后就有可能导致整个进程都会挂掉,这当然会对其它线程产生影响。另外,多个线程去访问同一块公共资源的时候也可能会出现冲突带来线程安全的问题。

所以,总的来说使用多线程的方式完成任务比多进程的方式完成任务会有优势,但同时存在一些缺点。但尽管如此,我们依然可以利用多线程的方式来很好的完成并发编程的任务。

好了,友友们,回顾到此结束,接下来我们一起学习新的内容吧!!!

二、创建线程的几种方式。

Java标准库中,提供了Thread()类来表示线程,同时Java创建线程的方式有很多种。最常见的两种写法:继承Thread()类重写Runnable方法()

继承Thread类

方式一:通过继承thread类来重写run方法是一种创建线程的方式(这里我们需要自己创建一个类来继承thread方法);方式二:还有一种方式我们也可以创建线程:基于匿名内部类的形式来继承thread类,并重run方法。

现在我们通过匿名内部类的方式来创建线程(基于匿名内部类的形式来继承thread类,并重写run方法,即方式二):

代码如下:

public class Demo03 {
    public static void main(String[] args) {
        Thread t = new Thread() {
            @Override
            public void run() {
                while(true) {
                    System.out.println("hello thread!!!");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        };
        t.start();
    }
}

解释:上述代码中我们创建了一个子类,这个子类(此类就是我们说的匿名内部类)继承自Thread类,同时我们在这个子类重对run方法进行了重写。
另外我们还创建了该子类(也就是之前说的匿名内部类)的实例,用引用t类指向这个实例。
最后就是通过引用t来调用start方法来调用系统API,再从内核中把线程创建出来。
【Java系列】详解多线程(二)——Thread类及常见方法(上篇),Java系列,JavaEE学习专栏,java,开发语言,java-ee
最后我们来看一下运行结果:
【Java系列】详解多线程(二)——Thread类及常见方法(上篇),Java系列,JavaEE学习专栏,java,开发语言,java-ee

实现runnable方法

方式三通过实现runnable来重写run方法(这里是自己创建一个类)的方式可以创建一个线程。方式四:另外我们这里依然可以用其它方式来创建一个线程:即基于匿名内部类的形式来实现runnable并重写run方法。

现在我们通过匿名内部类的形式来实现runnable并重写run方法(即方式四)来创建线程,代码如下:

public class Demo04 {
    public static void main(String[] args) {
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                while(true) {
                    System.out.println("hello world");
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        });
        t.start();
    }
}

解释,上述代码中,我们创建了runnable的子类,并重写了run方法,然后创建出了runnable的子类的实例,把这个实例传给Thread的构造方法。

lambda表达式

我们还可以通过lambda表达式来表示run方法的内容,从而创建出线程。

代码如下:

public class Demo05 {
    public static void main(String[] args) {
        Thread t = new Thread(() -> {
            while(true) {
                System.out.println("hello world");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        });
        t.start();
    }
}

什么是lambda表达式:lambda表达式本质上就是一个匿名函数,主要可以用来作为回调函数来进行使用。
先来回顾一下回调函数吧:回调函数不需要我们自己去进行调用,因为回调函数会在特定的时机自动地被调用。
此时的回调函数就是在线程创建成功之后才会真正执行。

关于线程的创建这里,也有其它的方式,比如基于callable的方式创建线程,基于线程池的方式创建线程。

三、Thread类及常见方法

Thread类是JVM用来管理线程的一个类,换句话说,每个线程都有一个唯一的Thread对象与之关联。我们对线程的各种操作都是根据Thread类进行展开的。

Thread类的常见构造方法

这里我们可以参照Java官方文档:https://docs.oracle.com/javase/8/docs/api/index.html
【Java系列】详解多线程(二)——Thread类及常见方法(上篇),Java系列,JavaEE学习专栏,java,开发语言,java-ee

【Java系列】详解多线程(二)——Thread类及常见方法(上篇),Java系列,JavaEE学习专栏,java,开发语言,java-ee

如下图的代码中,我们给线程起了个名字mythread
【Java系列】详解多线程(二)——Thread类及常见方法(上篇),Java系列,JavaEE学习专栏,java,开发语言,java-ee
我们可以通过JVM中的jconsole.exe来看到这个线程(mythread):
【Java系列】详解多线程(二)——Thread类及常见方法(上篇),Java系列,JavaEE学习专栏,java,开发语言,java-ee
现在问题来了,为什么我们没有看到主线程(main)呢?在上述代码中,主线程创建并启动了一个新线程,而新线程负责打印"hello thread"的消息。由于新线程的循环中有一个线程休眠的操作(Thread.sleep(1000)),所以您将会在不同时间间隔内看到"hello thread"的输出。但这个输出是由新线程负责的,而不是主线程。
所以,我们没有看到主线程的原因是主线程没有显示任何输出或执行其他代码来表明它的存在。它仅仅完成了创建和启动新线程的任务,然后退出。
对于主线程来说,main方法就是主线程的入口;而对于其它线程来说,lambda或者run就是线程的入口。所以执行完线程的入口函数,线程就算是结束了。

Thread类的属性

下面是Thread类的几个常见属性:

属性 获取方法
ID getId()
名称 getName()
状态 getState()
优先级 getPriority()
是否后台线程 isDasmon()
是否存活 isAlive()
是否被中断 isInterrupted()
  • ID:这里是JVM给线程设定的身份标识。对于一个线程来说,身份标识可以有好多个,就像一个人有大名也有小名。比如JVM给某一线程设定了一个身份标识,phread库(系统给开发人员提供的操作线程的API)也有一个线程的身份标识,内核中还有一个线程的身份标识,这几个身份标识之间相互独立互不干扰。
  • 名称:设置线程名称方便我们知道这个线程是干什么用的,同时也方便我们去进行调试。
  • 状态:Java中的线程状态和操作系统中的有所区别,Java中的线程状态更加细化。
  • 优先级:关于获取、设置线程的优先级其实并没有太大意义,因为系统内核进行线程调度的速度极快(线程调度是由系统内核负责的),快到我们根本无法感知,所有我们一般使用默认的线程优先级即可。
  • 是否是后台线程:后台线程又称为守护线程,后台线程不会影响线程结束;而前台线程会影响线程结束,如果前台线程没有执行完的话,进程是不会结束的;一个进程中如果所有的前台线程都执行完此时进程退出,如果此时依然存在后台线程没有执行完的话,后台线程依然会随着进程的退出而退出。我们创建的线程默认是前台线程。如下代码进行演示:
// 这里我们创建的线程是前台线程
public class Demo07 {
    public static void main(String[] args) {
        Thread t = new Thread(()->{
            while(true) {
                System.out.println("hello thread!!!");
            }
        });
        t.start();
    }
}

// 这里将我们创建的线程设置为了后台线程
public class Demo07 {
    public static void main(String[] args) {
        Thread t = new Thread(()->{
            while(true) {
                System.out.println("hello thread!!!");
            }
        });
        t.setDaemon(true);
        t.start();
    }
}

当我们把创建的线程设置为后台线程之后,程序运行起来之后就会立即结束。原因:由于我们把自己手动的线程创建成了后台线程,所以此时就只剩下main主线程了,而main线程这里不需要执行代码,所以执行时间极短main线程(前台线程)就结束了,而我们设置的后台线程还没有来得及执行就随着进程的退出而退出了。

  • 是否存活(isAlive()):这里的存活指的并不是thread对象是否存活,而是指的thread对象(我们也可以称为线程对象)对应的线程(即系统内核中的线程)是否存活。Thread对象的声明周期并不是和系统内核中的线程的生命周期完全一致。一般来说都是先把thread对象创建好,然后手动调用start方法,此时内核才真正创建出线程。对于线程,当它的run方法执行完后,线程的生命周期也会自然结束,就算线程对象还存在于内存中。此时该线程会释放占用的资源并进入死亡状态;还有另外一种情况就是线程对象生命周期的结束(即没有引用指向该线程对象了),此时线程也会结束,这种情况下,线程的run方法执行与否并不影响线程的结束状态。垃圾回收器会在适当时候回收无引用的线程对象,并释放相关资源。

好了,本文到这里就结束了,希望友友们可以支持一下一键三连哈。嗯,就到这里吧,再见啦!!!

【Java系列】详解多线程(二)——Thread类及常见方法(上篇),Java系列,JavaEE学习专栏,java,开发语言,java-ee文章来源地址https://www.toymoban.com/news/detail-755947.html

到了这里,关于【Java系列】详解多线程(二)——Thread类及常见方法(上篇)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • JavaEE之多线程编程:2.创建线程及Thread类常见方法(超全!!!)

    Java中创建线程的写法有很多种!!!这里介绍其中5种。 方法1:继承Thread类,重写run 创建一个类,让这个类继承自Thread父类,再重写我们的run方法就可以了。 使用Thread类,不需要import别的包,因为它是再Java.lang下面的。 注意: start() 是创建了一个新的线程,由新的线程来执

    2024年02月04日
    浏览(33)
  • 多线程系列(二) -Thread类使用详解

    在之前的文章中,我们简单的介绍了线程诞生的意义和基本概念,采用多线程的编程方式,能充分利用 CPU 资源,显著的提升程序的执行效率。 其中 java.lang.Thread 是 Java 实现多线程编程最核心的类,学习 Thread 类中的方法,是学习多线程的第一步。 下面我们就一起来看看,创

    2024年02月19日
    浏览(42)
  • Jmeter系列-阶梯加压线程组Stepping Thread Group详解(6)

    tepping Thread Group是第一个自定义线程组但,随着版本的迭代,已经有更好的线程组代替Stepping Thread Group了【Concurrency Thread Group】,所以说Stepping Thread Group已经是过去式了,但还是介绍一下 有预览图显示估计的负载; 可延迟启动线程组; 可持续增加线程负载; 可设置最大负载

    2024年02月09日
    浏览(43)
  • Jmeter学习系列之七:并发线程组Concurrency Thread Group详解

    Concurrency Thread Group提供了用于配置多个线程计划的简化方法 该线程组目的是为了 保持并发水平 ,意味着如果并发线程不够,则在运行线程中启动额外的线程 和Standard Thread Group不同,它不会预先创建所有线程,因此不会使用额外的内存 对于上篇讲到的Stepping Thread Group来说,

    2024年03月12日
    浏览(42)
  • Java 线程池(Thread Pools)详解

    目录 1、线程池介绍 2、线程池执行原理 3、线程池中的阻塞队列 4、Java 线程池中的拒绝策略 5、Java 提供的创建线程池的方式 6、线程池的使用示例 7、ForkJoinPool 和 ThreadPool 的区别 1、线程池介绍          线程池是一种重用线程的机制 ,用于提高线程的利用率和管理线程的

    2024年02月05日
    浏览(40)
  • Jmeter系列- test plan【测试计划】详细讲解 、 测试计划参数详解 、基础线程组Thread Group

    测试计划描述了 Jmeter 在执行时,一系列的步骤 一个完整的测试计划包含了一个或多个【线程组、逻辑控制器、采样器、监听器、定时器、断言和配置元素】测试计划添加or删除元件 通过右键点击树中的元件,选中要添加的元件 也可以通过合并(merge)或打开(open)从文件中

    2024年02月22日
    浏览(45)
  • 【Java中的Thread线程的简单方法介绍和使用详细分析】

    提示:若对Thread没有基本的了解,可以先阅读以下文章,同时部分的方法已经在如下两篇文章中介绍过了,本文不再重复介绍!! 【Java中Tread和Runnable创建新的线程的使用方法】 【Java中的Thread线程的七种属性的使用和分析】 提示:以下是本篇文章正文内容,下面案例可供参

    2024年02月15日
    浏览(39)
  • JAVA深化篇_29—— 线程使用之线程联合以及Thread类中的其他常用方法【附有详细说明及代码案例】

    线程联合 当前线程邀请调用方法的线程优先执行,在调用方法的线程执行结束之前,当前线程不能再次执行。线程A在运行期间,可以调用线程B的join()方法,让线程B和线程A联合。这样,线程A就必须等待线程B执行完毕后,才能继续执行。 join方法的使用 join()方法就是指调用该

    2024年02月05日
    浏览(44)
  • 【Java系列】详解多线程(三)—— 线程安全(下篇)

    个人主页:兜里有颗棉花糖 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创 收录于专栏【Java系列专栏】【JaveEE学习专栏】 本专栏旨在分享学习Java的一点学习心得,欢迎大家在评论区交流讨论💌 我们先来看一下什么是内存可见性问题,通过一段代码来进

    2024年02月04日
    浏览(40)
  • 【Java系列】详解多线程(一)

    个人主页:兜里有颗棉花糖 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创 收录于专栏【Java系列专栏】【JaveEE学习专栏】 本专栏旨在分享学习Java的一点学习心得,欢迎大家在评论区交流讨论💌 在引入多线程之前, 我们先来看一下进程是为了干什么的,

    2024年02月05日
    浏览(36)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包