1、进程与线程的概念

现在几乎所有操作系统都支持多任务,通常一个任务就是一个程序,一个运行中的程序就是一个进程。当一个程序行时,其内部也可能在执行多个任务,进程内每一个任务的执行流,就是一个线程。

所以线程也被称作轻量级进程。

总而言之,线程是进程的组成部分,可以独立、并发的执行任务

2、线程的创建和启动

Java中有两种方式来创建和启动线程。

2.1继承Thread类创建和启动线程

通过继承Thread类创建并启动多线程的步骤如下:

1、创建Thread的子类,并重写run方法。run方法中就是线程要执行的任务,所以也把run方法称为线程执行体。

2、创建该子类的实例,即线程对象。

3、使用线程对象的start方法启动线程。

线程类代码如下:

 //通过继承Thread类来创建线程类

 public class ThreadOne extends Thread{

     private int i;

     //重写run方法,方法体就是线程要执行的任务

     @Override

     public void run() {

        for (i = 0; i < 20; i++) {

            //继承Thread类时,可以直接调用getName()方法来返回当前线程的名字

            //如果想获取当前线程,直接使用this即可

            System.out.println(getName()+" "+i);

        }

     }

 }

测试代码如下:

 public class TestThreadOne {

     public static void main(String[] args) {

        for (int i = 0; i < 10; i++) {

            //打印主线线程的信息

            System.out.println(Thread.currentThread().getName()+" "+i);

            if (i==3) {

               //创建并启动第一条线程

               new ThreadOne().start();

               //创建并启动第二条线程

               new ThreadOne().start();

            }

        }

     }

 }

部分结果如下:

main 0
main 1
main 2
main 3
Thread-0 0
Thread-1 0
main 4
Thread-1 1
Thread-0 1
Thread-1 2
main 5

从上面的运行结果可以发现,测试类只显式的创建了两条线程,但实际上有三条线程在运行,即主线程、线程0和线程1。其中主线程的线程体是main方法里的内容。

此外,Thread-0和Thread-1两条线程输出的变量i的值是不连续的,而i并非是局部变量,而是实例成员变量,从这一点可以看出,每次创建线程都创建了一个ThreadOne对象。

2.2实现Runnable接口创建线程

实现Runnable接口创建并启动线程的步骤如下:

1、创建Runnable接口的实现类,重写run方法。run方法的内容即是线程要执行的任务。

2、创建Runnable实现类的实例,并以此实例作为Thread的target来创建Thread对象。该Thread对象才是真正的线程类。

线程类代码如下:

 //通过实现Runnable接口创建线程类

 public class ThreadTwo implements Runnable{

     private int i;

     //run方法同样是线程的执行体

     @Override

     public void run() {

        for (i = 0; i < 5; i++) {

            //实现Runnable接口创建线程类时,只能使用Thread.currentThread()来获取当前线程

            System.out.println(Thread.currentThread().getName()+" "+i);

        }

     }

 }

测试代码如下:

 public class TestThreadTwo {

     public static void main(String[] args) {

        for (int i = 0; i < 5; i++) {

            //打印主线线程的信息

            System.out.println(Thread.currentThread().getName()+" "+i);

            if (i==3) {

               ThreadTwo threadTwo = new ThreadTwo();

               //创建并启动第一条线程

               new Thread(threadTwo).start();

               //创建并启动第二条线程

               new Thread(threadTwo).start();

            }

        }

     }

 }

运行结果如下:

 main 0

 main 1

 main 2

 main 3

 Thread-0 0

 Thread-1 0

 main 4

 Thread-1 2

 Thread-0 1

 Thread-0 4

 Thread-1 3

从运行结果可以看出,Thread-0和Thread-1打印的i值是连续的,说明这两条线程是共享一个实例的。这是因为,我们所创建的两个Thread类使用同一个Runnable对象作为target。

在实际开发中,推荐使用实现Runnable接口的方式来创建线程,这样还可以实现或继

2.3有返回值的线程

从JDK1.5开始,java提供Callable接口和Future接口以获得线程的返回值。

Callable类似Runnable的加强版,提供一个call()方法作为线程的执行体。与Runnable的run()方法相比,call()方法更强大:

  • call()方法可以有返回值
  • call()可以声明抛出异常

不过Callable对象并不能直接作为Thread的target,必须使用Future对象包装一下。具体使用步骤如下:

  • 创建Callable的实现类,并实现call方法,注意有泛型限制。
  • 使用FutureTask包装Callable对象。
  • 把FutureTask作为target传入Thread,创建启动线程。
  • 调用FutureTask对象的方法获取返回值。

线程代码:

 public class CallableOne implements Callable<Integer>{

     @Override

     public Integer call() throws Exception {

        int i = 0;

        for(i=0;i<10;i++){

            System.out.println(Thread.currentThread().getName()+" "+i);

        }

        //线程返回值,模拟返回前阻塞主线程

        Thread.sleep(5000);

        return i;

     }

 }

测试代码:

 public class TestCallable {

     public static void main(String[] args) {

        //创建Callable对象

        CallableOne callableOne = new CallableOne();

        //使用FutureTask包装Callable对象

        FutureTask<Integer> futureTask = new FutureTask<Integer>(callableOne);

        for(int i=0;i<100;i++){

            System.out.println(Thread.currentThread().getName()+" "+i);

            //i等于3时启动子线程,此时可见主线程与子线程交替执行

            if (i==3) {

               //FutureTask是Runnable的子类,可传入Thread当作target

               new Thread(futureTask).start();

            }

            //i等于80的时候来获取子线程的返回值,此时主线程会阻塞,直到返回了结果

            if (i==80) {

               try {

                   System.out.println("线程的返回值:"+futureTask.get());

               } catch (Exception e) {

                   e.printStackTrace();

               }

            }

        }

     }

 }

以上代码中,在i等于3时启动子线程,此时可见主线程与子线程交替执行。在i等于80的时候来获取子线程的返回值,此时主线程会阻塞,直到返回结果。

java多线程回顾1:线程的概念与创建的更多相关文章

  1. 廖雪峰Java11多线程编程-1线程的概念-2创建新线程

    Java语言内置多线程支持: 一个Java程序实际上是一个JVM进程 JVM用一个主线程来执行main()方法 在main()方法中又可以启动多个线程 1.创建新线程 1.1 方法一:使用Thread ...

  2. java多线程总结:线程的两种创建方式及优劣比较

    1.通过实现Runnable接口线程创建 (1).定义一个类实现Runnable接口,重写接口中的run()方法.在run()方法中加入具体的任务代码或处理逻辑. (2).创建Runnable接口实现 ...

  3. Java多线程系列--“JUC线程池”04之 线程池原理(三)

    转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509960.html 本章介绍线程池的生命周期.在"Java多线程系列--“基础篇”01之 基 ...

  4. Java多线程系列--“JUC线程池”06之 Callable和Future

    概要 本章介绍线程池中的Callable和Future.Callable 和 Future 简介示例和源码分析(基于JDK1.7.0_40) 转载请注明出处:http://www.cnblogs.co ...

  5. Java多线程系列--“JUC线程池”02之 线程池原理(一)

    概要 在上一章"Java多线程系列--“JUC线程池”01之 线程池架构"中,我们了解了线程池的架构.线程池的实现类是ThreadPoolExecutor类.本章,我们通过分析Th ...

  6. Java多线程系列--“JUC线程池”03之 线程池原理(二)

    概要 在前面一章"Java多线程系列--“JUC线程池”02之 线程池原理(一)"中介绍了线程池的数据结构,本章会通过分析线程池的源码,对线程池进行说明.内容包括:线程池示例参考代 ...

  7. Java多线程系列--“JUC线程池”05之 线程池原理(四)

    概要 本章介绍线程池的拒绝策略.内容包括:拒绝策略介绍拒绝策略对比和示例 转载请注明出处:http://www.cnblogs.com/skywang12345/p/3512947.html 拒绝策略 ...

  8. 转:java多线程CountDownLatch及线程池ThreadPoolExecutor/ExecutorService使用示例

    java多线程CountDownLatch及线程池ThreadPoolExecutor/ExecutorService使用示例 1.CountDownLatch:一个同步工具类,它允许一个或多个线程一 ...

  9. Java多线程——进程和线程

    Java多线程——进程和线程 摘要:本文主要解释在Java这门编程语言中,什么是进程,什么是线程,以及二者之间的关系. 部分内容来自以下博客: https://www.cnblogs.com/dolp ...

  10. Java多线程之守护线程

    Java多线程之守护线程 一.前言 Java线程有两类: 用户线程:运行在前台,执行具体的任务,程序的主线程,连接网络的子线程等都是用户线程 守护线程:运行在后台,为其他前台线程服务 特点:一旦所有用 ...

随机推荐

  1. Vue-CLI项目-vue-cookie与vue-cookies处理cookie

    08.31自我总结 Vue-CLI项目-vue-cookie与vue-cookies处理cookie vue-cookie 一.模块的安装 npm install vue-cookie --save ...

  2. postman常用断言

    1.Code is 200 断言状态码是200 2.contains string 断言respoonse body中包含string 3.json value check (检查JSON值)

  3. Bran的内核开发教程(bkerndev)-02 准备工作

    准备工作   内核开发是编写代码以及调试各种系统组件的漫长过程.一开始这似乎是一个让人畏惧的任务,但是并不需要大量的工具集来编写自己的内核.这个内核开发教程主要涉及使用GRUB将内核加载到内存中.GR ...

  4. shark恒破解笔记6-摆脱NAG

    1.打开软件后,发现是未注册,然后点击关闭按钮,会弹出窗口 我们的目的就是为了能够去掉这个弹窗. 2.对这个程序进行查壳,没有什么发现 3.载入OD里面,F9运行起来,随后切换到程序主界面点击关闭按钮 ...

  5. [插件化开发] Poc之后,我选择放弃OSGI

    Poc之后,我选择放弃OSGI TIPS: 如贵司允许重构老系统或者允许使用OSGI的第三方框架改造所带来的投入成本,并且评估之后ROI乐观,那么还是可以使用的. Runtime Version 以下 ...

  6. luogu P3380 【模板】二逼平衡树(分块实现)

    题目描述 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 查询 \(k\) 在区间内的排名 查询区间内排名为 \(k\) 的值 修改某一位值上的数值 查询 \(k\ ...

  7. Python 爬取豆瓣TOP250实战

    学习爬虫之路,必经的一个小项目就是爬取豆瓣的TOP250了,首先我们进入TOP250的界面看看. 可以看到每部电影都有比较全面的简介.其中包括电影名.导演.评分等. 接下来,我们就爬取这些数据,并将这 ...

  8. 关于Linux中的 localhost 默认地址简单介绍

    大家都知道localhost指的是本机的IP地址:127.0.0.1 用于回路测试,那能不能修改localhost呢,答案肯定是可以的 打开终端--->输入: vim /etc/host  然后 ...

  9. mysql数据备份之 xtrabackup

    上一篇简单介绍了一下mysqldump进行数据库的备份和恢复,这一篇说一下另一种备份工具xtrabackup,在InnoDB事务引擎泛滥的时代,xtrabackup可以很好的支持数据库的热备份,这就很 ...

  10. mybatis 使用redis实现二级缓存(spring boot)

    mybatis 自定义redis做二级缓存 前言 如果关注功能实现,可以直接看功能实现部分 何时使用二级缓存 一个宗旨---不常变的稳定而常用的 一级是默认开启的sqlsession级别的. 只在单表 ...