什么是线程

说到线程就不得不说下进程了, 大家都知道,许许多多的线程组合在一起就成了一个进程,进程是由操作系统进行资源操作的一个最小的单位,线程则是比进程更小的实际执行操作的单位;每个线程都有自己的堆栈及变量空间。

就好比回家,各种回家线路组成了一个回家的进程,每条线路则代表一个单一的线程

线程的生命周期

  • 新创建(New)
  • 就绪(Ready)
  • 运行(Running)
  • 阻塞(Blocked)
  • 销毁(Dead)

线程的实现方式

  1. 继承Thread重写run方法

    public class MyThread extends Thread{
    
        private String threadName;
    // 控制循环的数量
    private int num = 1000; // 定义商品总数
    private int goodsNum = 15; MyThread(String name) {
    this.threadName = name;
    } @Override
    public void run() {
    for (int i = 0; i< num; i++) {
    if (this.goodsNum > 0) {
    this.goodsNum -- ;
    System.out.println(threadName + "购买了一个,剩余" + this.goodsNum);
    }
    }
    } public static void main (String[] args) { MyThread mt = new MyThread("线程1");
    MyThread mt1 = new MyThread("线程2");
    mt.start();
    mt1.start();
    }
    }

    继承Thread需要重写run方法,方法体中就是线程具体的运行逻辑,下面就是代码运行的结果:

    ![](https://img2018.cnblogs.com/blog/1777612/201908/1777612-20190826111234175-898122334.png)

    有趣的是2个线程都执行了15次,为啥会出现这样的问题?回到代码中

    我们是通过new了2个类对象来创建的线程;创建的时候,2个线程都被分配了一样的资源就会导致出现这样 的问题。

    ​优化如下:

    public class MyThread2{
    class TestThread extends Thread { private int goodsNumber = 15; @Override
    public void run() {
    for (int i = 0; i <= 16; i++) {
    if (this.goodsNumber > 0) {
    System.out.println(Thread.currentThread().getName()
    + "购买了一个,剩余:" + (this.goodsNumber --));
    }
    }
    }
    } public static void main(String[] args) throws InterruptedException {
    // 开启一个线程任务
    TestThread tt = new MyThread2().new TestThread();
    // 将任务分配给2个线程去执行
    Thread t1 = new Thread(tt, "线程1");
    Thread t2 = new Thread(tt, "线程2");
    t1.start();
    t2.start();
    }
    }

    结果:

    ![](https://img2018.cnblogs.com/blog/1777612/201908/1777612-20190826143441586-2109088274.png)

    备注:直接new一个线程调用run方法并不会创建一个新线程去执行,只是普通对象的方法调用。切记使用start来启动线程

  2. 实现Runnable接口

    根据查看Runnable的源码,Runnable只提供一个run方法,并没有start方法,所以Runnable的启动执行还是需要借助Thread类

    public class MyThread3 implements Runnable {
    
        private int num = 5;
    
        @Override
    public void run() {
    for (int i = 0; i< 10; i++) {
    if (this.num > 0) {
    System.out.println(Thread.currentThread().getName() + "num left:" + this.num --);
    }
    }
    } public static void main(String[] args) {
    // 创建线程任务
    MyThread3 m = new MyThread3();
    // 将任务分配给3个线程去执行
    new Thread(m, "Thread1").start(); // 线程start后,会放置线程等待队列,等待CPU调度,通过Thread调用run方法
    new Thread(m, "Thread2").start();
    new Thread(m, "Thread3").start();
    }
    }
  3. Callable和Future的使用

    Callable和Runnable有点类似,都是只提供了一个对外接口,不同的是Callable有返回值。

    一般Callable搭配ExecutorService来使用;

    通过查看源码可以知道ExecutorService可以用来启动Runnable和Callable

    // 启动Callable线程
    <T> Future<T> submit(Callable<T> task);
    // 启动Runnable线程
    Future<?> submit(Runnable task);

    Future就是对于具体的Runnable或者Callable任务的执行结果进行取消、查询是否完成、获取结果。必要时可以通过get方法获取执行结果,该方法会阻塞直到任务返回结果。

    package com.hervey.thread;
    
    import java.util.concurrent.*;
    
    /**
    * @program: thread
    * @description: Callable线程
    * @author: hetangyuese
    * @create: 2019-08-26 14:51
    **/
    public class ThreadCallable { private int num = 10; /**
    * 实现Callable的线程
    */
    class TestCallable implements Callable<String> {
    @Override
    public String call() throws Exception {
    Thread.sleep(5000);
    return "hello";
    }
    } /**
    * 实现Runnable的线程
    */
    class TestRunnable implements Runnable {
    @Override
    public void run() {
    int sleepTime = 1000;
    try {
    Thread.sleep(sleepTime);
    } catch (InterruptedException e) {
    e.printStackTrace();
    } System.out.println("start");
    }
    } public static void main(String[] args) throws ExecutionException, InterruptedException {
    // 创建线程池 数量为10
    ExecutorService es = Executors.newFixedThreadPool(10);
    // Callable线程可以有返回值
    Future<String> future = es.submit(new ThreadCallable().new TestCallable());
    // 在调用get方法的时候会阻塞线程池知道获取到了才会继续执行
    System.out.println(future.get());
    try {
    // 用于取消线程任务,传参表示是否取消正在执行的线程
    future.cancel(true);
    if (!future.isCancelled()) {
    // 此处通过设置超时时间,在一秒后如果future还未获取到返回值则直接结束
    System.out.println(future.get(10000, TimeUnit.MILLISECONDS));
    }
    } catch (TimeoutException e) {
    e.printStackTrace();
    }
    // isDone方法判断future是否完成即有没有获取到返回值
    System.out.println(future.isDone()); // 执行Runnable线程
    es.submit(new ThreadCallable().new TestRunnable()); if (!es.isShutdown()) {
    es.shutdown();
    }
    }
    }

java基础-多线程一的更多相关文章

  1. java基础---->多线程之Runnable(一)

    java线程的创建有两种方式,这里我们通过简单的实例来学习一下.一切都明明白白,但我们仍匆匆错过,因为你相信命运,因为我怀疑生活. java中多线程的创建 一.通过继承Thread类来创建多线程 pu ...

  2. java基础---->多线程之synchronized(六)

    这里学习一下java多线程中的关于synchronized的用法.我来不及认真地年轻,待明白过来时,只能选择认真地老去. synchronized的简单实例 一. synchronized在方法上的使 ...

  3. java基础---->多线程之wait和notify(八)

    这里学习一下java多线程中的关于wait方法和notify方法的用法.命运不是风,来回吹,命运是大地,走到哪你都在命运中. wait和notify方法的使用 一.wait与notify的简单实例 i ...

  4. java基础---->多线程之ThreadLocal(七)

    这里学习一下java多线程中的关于ThreadLocal的用法.人时已尽,人世还长,我在中间,应该休息. ThreadLocal的简单实例 一.ThreadLocal的简单使用 package com ...

  5. java基础---->多线程之interrupt(九)

    这里我们通过实例来学习一下java多线程中关于interrupt方法的一些知识.执者失之.我想当一个诗人的时候,我就失去了诗,我想当一个人的时候,我就失去了我自己.在你什么也不想要的时候,一切如期而来 ...

  6. java基础---->多线程之yield(三)

    yield方法的作用是放弃当前的CPU资源,将它让给其它的任务去占用CPU执行时间.但放弃的时间不确定,有可能刚刚放弃,马上又获得CPU时间片.今天我们通过实例来学习一下yield()方法的使用.最是 ...

  7. java基础---->多线程之Daemon(五)

    在java线程中有两种线程,一种是用户线程,另一种是守护线程.守护线程是一种特殊的线程,当进程中不存在非守护线程了,则守护线程自动销毁.今天我们通过实例来学习一下java中关于守护线程的知识.我是个平 ...

  8. java基础---->多线程之priority(四)

    线程的priority能告诉调度程序其重要性如何,今天我们通过实例来学习一下java多线程中的关于优先级的知识.我从没被谁知道,所以也没被谁忘记.在别人的回忆中生活,并不是我的目的. java多线程的 ...

  9. Java基础-进程与线程之Thread类详解

    Java基础-进程与线程之Thread类详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.进程与线程的区别 简而言之:一个程序运行后至少有一个进程,一个进程中可以包含多个线程 ...

随机推荐

  1. Tips 14:思维导图读书笔记法

    Tips 14:思维导图读书笔记法作读书笔记不仅能提高阅读书.文的效率,而且能提高科学研究和写作能力.读书笔记一般分为摘录.提纲.批注.心得几种,这里特别推荐思维导图式的读书笔记. 通过思维导图先大概 ...

  2. Python秒算24点,行还是不行?

    周末闲来无事,看到隔壁家的老王在和隔壁家的媳妇玩24点,就进屋看了看.发现老王是真不行啊,那不行,这也不行. 就连个24点都玩不过他媳妇,给他媳妇气的,啥都不能满足,这不能,那也不能. 我坐下来和他媳 ...

  3. WPF 动态添加按钮以及样式字典的引用(Style introduction)

    我们想要达到的结果是,绑定多个Checkbox然后我们还可以获取它是否被选中,其实很简单,我们只要找到那几个关键的对象就可以了. 下面是Ui,其中定义了一个WrapPanel来存放CheckBox,还 ...

  4. vue教程二 vue组件(2)

    每个组件都可以有自己的data.methods.computed和您之前看到的所有内容—就像Vue实例本身一样. 您可能已经注意到了组件和Vue实例之间的一个细微差别:Vue实例上的数据属性是一个对象 ...

  5. 作为前端的你,CC游戏开发可以上车

    1. 初来乍到 打开 Cocos Creator 点击新建空白项目,在默认布局的左下区域,一个黄黄assets文件夹映入眼帘.作为前端的你对这个文件是不是再熟悉不过了.是的,和你想象的一样,开发游戏中 ...

  6. ssm 搭建项目各项配置

    首先配置 pom.xml <?xml version="1.0" encoding="UTF-8"?> <project xmlns=&quo ...

  7. 0R电阻在PCB布线中对布线畅通的一个小妙用

    在PCB布线中,我们都会尽量节约板子空间,将元器件排布的紧密一些,难免会遇到布线不通的时候. 博主下面就来说一个关于0R电阻在PCB布线使之畅通的一个小妙用. 使用0R电阻前 假设我们这个TXD的线周 ...

  8. 如何创建Github创库

    重点:利用Markdown语言写简单的日常使用的文本 基础写作和语法格式 本篇文章的内容来源于Github的基础写作帮助.如果在观看时有什么问题,可以直接查阅源文件.另外需要说明的是Git对Markd ...

  9. Spring Cloud微服务接口这么多怎么调试

    导读 我们知道在微服务架构下,软件系统会被拆分成很多个独立运行的服务,而这些服务间需要交互通信,就需要定义各种各样的服务接口.具体来说,在基于Spring Cloud的微服务模式中,各个微服务会基于S ...

  10. .Net Core in Docker - 使用阿里云Codepipeline及阿里云容器镜像服务实现持续集成(CI)

    前面已经介绍过了 .Net Core 程序发布到 Docker 容器的内容.但是每次通过 SSH 链接到服务器敲命令,运行脚本也是挺麻烦的一件事.程序员是最懒的,能让电脑解决的问题绝不手动解决,如果当 ...