第一种:  通过继承Thread类创建线程

第二种: 通过实现Runnable接口创建线程

这两种早已烂记于心,这里就不作过多的介绍, 主要介绍其源码

Thread类 implements Runnable

thread空线程的run方法 是判断target是否存在实,再执行target实例中的run方法

  1. public void run(){
  2. if(this.target != null){
  3. this.target.run();
  4. }
  5. }

通过实现Runnable接口, 并且通过thread构造函数创建线程的方法

  1. Runnable runnable = new Runnable(){...重写run方法};
  2.  
  3. Thread thread_one = new Thread(runnable);
  4. Thread thread_two = new Thread(runnable);

如果创建多个线程, 实质是多个线程引用同一个target 实例

对比两种方式的区别:

  通过继承Thread类实现多线程的方式由于单继承的局限性, 不能再继承其他类, 只能完成各自的任务

  通过实现Runnable接口实现多线程的方式能更好的做到并发完成同一个任务, 因为访问的是同一个target, 实现了共享数据

总之,在大多数情况下,都偏向于通过实现Runnable接口创建多线程

第三种: 使用Callable接口 和 FutureTask类创建线程

由于Thread类和Runnbale接口中的run方法没有返回值, 所以这两种方式不能获取异步执行的结果

Callable接口

  1. package java.util.concurrent;
  2. @FunctionalInterface
  3. public interface Callable<V>{
  4. V call() throws Exception;
  5. }

@FunctionalInterface 注解标注在接口上, 表示此接口为"函数式接口"

函数式接口:  只有一个抽象方法的接口

此注解只是方便编译器进行检查, 不加也不会影响. 如果加了注解但该接口不是函数式接口,编译器会报错

Callable接口是泛型接口 ,也是函数式接口

call()抽象方法还有一个Exception的异常声明, 容许方法内的异常直接抛出,并且可以不予捕获

Future接口, RunnableFuture接口,以及FutureTask实现类都是位于 java.util.concurrent包下

V get()  用于阻塞性得到异步执行的结果.  此方法是阻塞性的,异步未执行完会处于阻塞状态

Object outcome  用于保存call()方法的异步执行结果.  get()会获取

创建线程的具体步骤

  1. class ReturnableTask implements Callable<Long>{
  2. public long call() throws Exception{
  3. //线程要执行的代码
  4. }
  5. }
  6.  
  7. public static void main(String args[]) throws InterruptedException{
  8. ReturnableTask task = new new ReturnableTask();
  9. Futuretask<Long> FutureTask = new Futuretask<Long>(task);
  10. Thread thread = new Thread(FutureTask,"returnableThread");
  11. thread.start();
  12. System.out.println(FutureTask.get()); //得到异步执行结果
  13. }

下图为具体实现过程

两个线程处于并发状态, 默认异步执行

看起来两个线程是同时进行,实质上是不是, 单个进程在同一时间只能执行一个进程,由于分给线程的时间片非常短(线程切换毫秒级),所以以为是同时

并发执行的消息通信机制分为同步和异步,  这些就不做过多解释

总之,图上的两个线程不是同时运行

第四种: 通过线程池创建线程

通过Thread创建线程在执行完就被销毁了, 不可服用. 在高并发场景中, 频繁创建线程是非常消耗资源的, 通过线程池创建线程可以对已经创建好的线程进行复用

Executors 静态工厂类  用于创建不同的线程池 java.util.concurrent;

  1. //创建一个包含三个线程的线程池
  2. private static ExecutorService pool = Executors.newFixedThreadPool(3);

Executor接口

  void executo(Runnable command);    //执行Runnable类型

ExecutorService<T> extends Executor

  <T> Future<T> submit(Callable<T> task)   //提交callable类型以执行

  Future<?> submit(Runnable task)      //提交Runnable类型以执行

具体实现

  1. main
  2. Future future = pool.submit(new RuturnableTask());
  3. Long result = (Long) future.get(); //得到异步执行的结果

execute() 与 submit() 执行线程方法的区别

execute()只能执行Runnable, 并且无返回值

submit() 既能执行Runnable又能执行callable, 并且有返回值

注: 实际生产环境中禁止使用Executors创建线程池

java创建线程的四种方法的更多相关文章

  1. Java创建线程的四种方式

    Java创建线程的四种方式 1.继承Thread类创建线程 定义Thread类的子类,并重写该类的run方法,run()方法的内容就是该线程执行的内容 创建Thread子类的实例,即创建了线程对象. ...

  2. java创建线程的三种方法

    这里不会贴代码,只是将创建线程的三种方法做个笼统的介绍,再根据源码添加上自己的分析. 通过三种方法可以创建java线程: 1.继承Thread类. 2.实现Runnable接口. 3.实现Callab ...

  3. Java多线程——之一创建线程的四种方法

    1.实现Runnable接口,重载run(),无返回值 package thread; public class ThreadRunnable implements Runnable { public ...

  4. Java 创建线程的两种方法

    Java提供了线程类Thread来创建多线程的程序.其实,创建线程与创建普通的类的对象的操作是一样的,而线程就是Thread类或其子类的实例对象.每个Thread对象描述了一个单独的线程.要产生一个线 ...

  5. Java创建线程的两种方法

    大多数情况,通过实例化一个Thread对象来创建一个线程.Java定义了两种方式: 实现Runnable 接口: 可以继承Thread类. 下面的两小节依次介绍了每一种方式. 实现Runnable接口 ...

  6. Java 创建线程的3种方法及各自优势

    1. 继承 Thread 类,然后调用 start 方法. class MyThread extends Thread { //重写run方法,线程运行后,跑的就是run方法 public void ...

  7. java 创建线程的三种方法Callable,Runnable,Thread比较及用法

    转自:http://www.chinaitlab.com/Java/line/942440.html 编写多线程程序是为了实现多任务的并发执行,从而能够更好地与用户交互.一般有三种方法,Thread, ...

  8. Java创建线程的三种方法比较

    一般有三种方法,Thread,Runnable,Callable. Runnable和Callable的区别 (1)Callable规定的方法是call(),Runnable规定的方法是run(). ...

  9. 区分JAVA创建线程的几种方法

    1. start()和run()       通过调用Thread类的start()方法来启动一个线程,这时此线程是处于就绪状态,并没有运行.然后 通过此Thread类调用方法run()来完成其运行操 ...

随机推荐

  1. golang中结构体当做函数参数或函数返回值都会被拷贝

    1. 结构体做函数的参数或返回值时,都会被重新拷贝一份如果不想拷贝,可以传递结构体指针 package main import "fmt" type Person struct { ...

  2. ansible roles实践——安装nginx

    1.创建roles 在/etc/ansible/roles目录下 1.1 手动创建需要的目录 1.2 使用命令创建,用不到的目录可以创建为空目录,但不可以不创建. 创建目录[root@master] ...

  3. mongodb中oplogsize大小设置

    1 oplogsize大小配置: 2 mongodb副本集构建的高可用方案,最少需要三个节点,一个主节点master,一个从节点slave,一个选举仲裁节点arbiter.当主节点奔溃的时候,仲裁节点 ...

  4. Vue之 css3 样式重置 代码

    reset.css @charset "utf-8";html{background-color:#fff;color:#000;font-size:12px} body,ul,o ...

  5. File 类的 getPath()、getAbsolutePath()、getCanonicalPath() 的区别【转】

    File 类的 getPath().getAbsolutePath().getCanonicalPath() 的区别 感谢大佬:https://blog.csdn.net/zsensei/articl ...

  6. docker容器编排 (4)

    容器编排 我们的项目可能会使用了多个容器,容器多了之后管理容器的工作就会变得麻烦.如果要对多个容器进行自动配置使得容器可以相互协作甚至实现复杂的调度,这就需要进行容器编排.Docker原生对容器编排的 ...

  7. NSMutableString基本概念

    1.NSMutableString 基本概念 NSMutableString 类 继承NSString类,那么NSString 供的方法在NSMutableString中基本都可以使用,NSMutab ...

  8. Docker容器和宿主机互传文件

    1.docker容器向宿主机传送文件 格式: docker cp container_id:<docker容器内的路径> <本地保存文件的路径> 例: docker cp 10 ...

  9. postman项目接口文档和登录步骤原理

    培训内容 实训项目:非常果岭-发现模块接口测试,单接口.流程脚本编写: 使用工具:postman 培训方式 1)postman使用说明 2)项目接口文档和登录步骤原理   一.首先了解postman使 ...

  10. IT职业技能图谱:架构师、H5、DBA、移动、大数据、运维...

    转载 作者:StuQ 文章收藏自微信:InfoQ   时隔近5个月,StuQ的小伙伴们再次出品了IT职业技能图谱更新版.这回除更新之前版本外,还添加了架构师.HTML 5.DBA等新的职业技能图谱.正 ...