线程池启动线程的方式(java启动线程的方法)
admin
2023-09-17 19:04:41
0

我们在学习软件开发时,多线程,高并发是一个必不可少的知识点,也是在面试时必会问到的内容,为了让大家对多线程,高并发编程有个清晰认识,特地组织了一个专栏来专门介绍一下,希望能对大家有一些帮助。

线程简介

线程是程序运行的基本执行单元。当操作系统(不包括单线程的操作系统,如微软早期的DOS)在执行一个程序时,会在系统中建立一个进程,而在这个进程中,必须至少建立一个线程(这个线程被称为主线程)来作为这个程序运行的入口点。因此,在操作系统中运行的任何程序都至少有一个主线程。

进程和线程是现代操作系统中两个必不可少的运行模型。在操作系统中可以有多个进程,这些进程包括系统进程(由操作系统内部建立的进程)和用户进程(由用户程序建立的进程);一个进程中可以有一个或多个线程。进程和进程之间不共享内存,也就是说系统中的进程是在各自独立的内存空间中运行的。而一个进程中的线可以共享系统分派给这个进程的内存空间。

在进一步介绍之前,我们先来了解一些基本概念,以帮助大家更快速的理解。

基本概念

进程

进程是操作系统中正在执行的不同的应用程序,例如:我们可以同时打开微信和QQ,甚至更多的程序。

在操作系统中运行的程序就是进程,进程就是执行程序的一次执行过程,它是一个动态的概念式系统资源分配的单位

通常在一个进程中可以包含若干个线程,当然一个进程中至少有一个线程,不然没有存在的意义,线程是CPU调度和执行的单位

线程

线程是一个应用程序进程中不同的执行路径,例如:我们的WEB服务器,能够为多个用户同时提供请求服务。

进程是不活泼的,进程从来不执行任何东西,它只是线程的容器。线程总是在某个进程环境中创建的,而且它的整个生命周期都在该进程中。

在一个进程中,如果创建了多个线程,线程的运行是由调度器安排调度的,调度器是与操作系统紧密相关的,先后顺序是不能人为干预的

多线程

多线程拥有多条执行路径,「主线程与子线程并行交替执行」(普通方法只有主线程一条路径),对同一份资源操作时,会存在资源抢夺的问题,这时就需要加入并发控制了。

一个Java应用程序,至少有三个线程: main()主线程, gc()垃圾回收线程,异常处理线程。当然如果发生异常,会影响主线程。

多线程程序的优点:

  • 提高应用程序的响应。对图形化界面更有意义,可增强用户体验。同时做多个事情。比如:一边听歌、一边写代码。
  • 提高计算机系统CPU的利用率。不过线程也会带来额外的开销,如CPU调度时间,并发控制带来的系统开销。
  • 改善程序结构。将既长又复杂的进程分为多个线程,独立运行,利于理解和修改。

何时需要多线程?

程序需要同时执行两个或多个任务。

需要一些后台运行的程序时,比如:Java后台运行的GC线程。

创建线程

Java中创建线程有四种方式,我们下面依次介绍一下。

1、继承 Thread 类

(1)定义Thread类的子类,并重写该类的run方法,该run方法的方法体就代表了线程要完成的任务。因此把run()方法称为执行体。

(2)创建Thread子类的实例对象,即创建了一个线程对象。

(3)调用该线程对象的start()方法来启动该线程。

示例代码:

public class MyThread extends Thread { // 总票数 public int count = 10; @Override public void run() { // 当还有票时就继续售卖 while (count > 0) { // 剩余票数 count--; System.out.println( Thread.currentThread().getName() + "售卖第 " + (10 - count) + " 张票,当前剩余票数: " + count); } } public static void main(String[] args) { MyThread myThread = new MyThread(); myThread.start(); } }

2、实现Runnable接口

(1)定义Runnable接口的实现类,并重写该接口的run()方法,该run()方法的方法体同样是该线程的线程执行体。

(2)创建 Runnable实现类的实例对象,并将该实例作为Thread的target来创建一个Thread对象,该Thread对象才是真正的线程运行对象。

(3)调用该线程对象的start()方法来启动线程。

示例代码:

public class MyRunableThread implements Runnable { // 总票数 public int count = 10; @Override public void run() { // 当还有票时就继续售卖 while (count > 0) { // 剩余票数 count--; System.out.println( Thread.currentThread().getName() + "售卖第 " + (10 - count) + " 张票,当前剩余票数: " + count); } } public static void main(String[] args) { MyRunableThread myRunableThread = new MyRunableThread(); Thread myThread = new Thread(myRunableThread); myThread.start(); } }

「Thread 和 Runnable 的区别」

上述两种方法是大家最常见到的两种创建线程的方法,也常常会被问到两种方式创建线程的区别,下面简单总结了一下:

「继承 Thread 类」

子类继承 Thread 类具备多线程能力

启动线程:子类线程对象调用 .start()方法

不建议使用:避免 OOP 单继承局限性

「实现接口 Runnable」

  • 具有多线程能力
  • 启动线程:传入目标对象 + Thread对象调用.start()方法
  • 推荐使用:避免单继承局限性,方便同一个对象被多个线程使用

另外,在使用线程池时只能放入实现Runable或Callable类线程,不能直接放入继承Thread的类

3、实现Callable接口

(1)创建Callable接口的实现类,并实现call()方法,该call()方法将作为线程执行体,并且有返回值。

(2)创建Callable实现类的实例,使用FutureTask类来包装Callable对象,该FutureTask对象封装了该Callable对象的call()方法的返回值。

(3)使用FutureTask对象作为Thread对象的target创建并启动新线程。

(4)调用FutureTask对象的get()方法来获得子线程执行结束后的返回值。

示例代码:

public class MyCallableThread implements Callable { // 总票数 private int count = 10; @Override public String call() throws Exception { // 当还有票时就继续售卖 while (count > 0) { // 剩余票数 count--; System.out.println( Thread.currentThread().getName() + "售卖第 " + (10 - count) + " 张票,当前剩余票数: " + count); } return "票已售完"; } public static void main(String[] args) throws InterruptedException, ExecutionException { Callable callable = new MyCallableThread(); FutureTask futureTask = new FutureTask<>(callable); Thread myThread = new Thread(futureTask); myThread.start(); // 打印返回结果 System.out.println(futureTask.get()); } }

「Runnable和Callable的区别:」

Callable规定的方法是call(),Runnable规定的方法是run()。

Callable的任务执行后可返回值,而Runnable的任务是不能有返回值。

call方法可以抛出异常,run方法不可以。

4、线程池

Java默认提供了五种线程池,通过Executors创建,分别为:

  • 「newCachedThreadPool」 创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。
  • 「newFixedThreadPool」 创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。
  • 「newScheduledThreadPool」 创建一个定长线程池,支持定时及周期性任务执行。
  • 「newSingleThreadExecutor」 创建一个单线程化的线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行。
  • 「newWorkStealingPool」 创建一个具有抢占式操作的线程池,由于能够合理的使用CPU进行对任务操作(并行操作),所以适合使用在很耗时的任务中。

示例代码:

public class MyThreadPool implements Runnable { // 总票数 public int count = 10; @Override public void run() { // 当还有票时就继续售卖 while (count > 0) { // 剩余票数 count--; System.out.println( Thread.currentThread().getName() + "售卖第 " + (10 - count) + " 张票,当前剩余票数: " + count); } } public static void main(String[] args) { ExecutorService ex = Executors.newFixedThreadPool(5); MyThreadPool t = new MyThreadPool(); ex.submit(t); ex.shutdown(); } }

关于线程池,后续会有单独文章给大家详细介绍。

通过以上的内容,希望大家可以对线程有个初步的认识,相关示例代码,稍后整理后我会上传到GitHub上,也请大家留意我们的后续文章。

相关内容

热门资讯

“驴友”庐山野游被困获救:追偿... 11月15日,记者从江西省应急管理厅了解到,庐山“9·27”驴友野游救援事件处理有了最新进展,当地政...
惜败!张伟丽不敌舍甫琴科 无缘... 北京时间11月16日中午,在美国纽约进行的UFC(终极格斗冠军赛)322联合主赛中,中国选手张伟丽经...
美联邦官员证实执法人员在夏洛特... 15日,美国联邦官员证实,已加强了在北卡罗来纳州夏洛特市针对非法移民的执法力度。当天,联邦执法人员在...
日本前首相:高市早苗应坚持“无... △日本首相高市早苗(资料图)日本前首相、立宪民主党党首野田佳彦15日说,首相高市早苗应当坚持“无核三...
招商局集团有限公司原副总经理李... 招商局集团有限公司原党委委员、副总经理李百安涉嫌严重违纪违法,目前正接受中央纪委国家监委纪律审查和监...
教育部发布留学预警:中国公民谨... 今日,教育部发布留学预警。近段时间以来,日本社会治安不靖,针对中国公民的违法犯罪案件多发,治安形势和...
南部战区组织轰炸机编队位南海进... 南部战区新闻发言人田军里空军大校表示,11月14日,中国人民解放军南部战区组织轰炸机编队位南海进行例...
高市早苗涉台错误言论引发多方批... 日本首相高市早苗日前在国会答辩时公然发表涉台露骨挑衅言论,暗示可能武力介入台海问题,性质影响极其恶劣...
中国海警舰艇编队11月16日在... 11月16日,中国海警1307舰艇编队在我钓鱼岛领海内巡航。这是中国海警依法开展的维权巡航活动。
御寒“神器”电加热衣走红 专家... 随着入冬气温不断降低,一些具有加热保暖功能的电加热衣也逐渐走红。这到底是一种怎样的御寒神器?记者在网...
全国冬小麦播种过八成 各地抢抓... 农业农村部最新农情调度显示,目前,全国冬小麦播种已过八成。其中黄淮海近八成半。分省看,河南近八成,安...
俄称控制一定居点 乌军称打击俄... 当地时间15日,俄罗斯国防部发布战报称,俄军对保障乌克兰国防工业体系运行的能源设施、军用机场、乌国家...
加拿大本拿比市就歧视华裔历史正... 当地时间11月15日,加拿大不列颠哥伦比亚省大温哥华地区本拿比市就历史上针对华裔的歧视进行正式道歉。...
美国加州南部海域疑似偷渡船倾覆... 当地时间15日凌晨,美国加利福尼亚州南部海域发生一起疑似偷渡船倾覆事故。警方说,事故已导致4人死亡、...
乌克兰总统宣布启动国有能源企业... 当地时间11月15日,乌克兰总统泽连斯基通过社交媒体宣布,启动国有能源企业全面改革,同步推进财务审计...
继三大航司后,多家航司同日发布... 14日晚,外交部和中国驻日本使领馆就中国公民前往日本发布郑重提醒。15日,多家航司发出通知,公布了涉...
神舟二十二号飞船发射任务已启动... 神舟二十号乘组已平安返回地球,根据计划安排,后续将择机发射神舟二十二号飞船。据介绍,目前已启动神舟二...
航行警告!黄海中部连续三天24... 据中国海事局网站消息,盐城海事局发布航行警告,11月17日至19日,每日0时到24时,黄海中部部分海...
中国南极考察队成功救援1名俄南... 应俄罗斯方面请求,中国第42次南极考察队“雪鹰601”飞机,于11月14日成功将一名俄罗斯和平站病员...
王济生诗歌 紫竹院菊韵(诗三首... 1.双清秋尽黄花艳冷香朔风中紫竹邀仙子高洁见双清2.七彩碧波渺渺过旧船,紫竹浓处听潺潺。行宫百菊正浓...