浅析 Java多线程

什么是进程

  当一个程序进入内存中运行起来它就变为一个进程。因此,进程就是一个处于运行状态的程序。同时进程具有独立功能,进程是操作系统进行资源分配和调度的独立单位。

什么是线程

  线程是进程的组成部分。通常情况下,一个进程可拥有多个线程,而一个线程只能拥有一个父进程。

  线程可以拥有自己的堆栈、自己的程序计数器及自己的局部变量,但是线程不能拥有系统资源,它与其父进程的其他线程共享进程中的全部资源,这其中包括进程的代码段、数据段、堆空间以及一些进程级的资源(例如,打开的文件等)。

  线程是进程的执行单元,是CPU调度和分派的基本单位,当进程被初始化之后,主线程就会被创建。同时如果有需要,还可以在程序执行过程中创建出其他线程,这些线程之间也是相互独立的,并且在同一进程中并发执行。因此一个进程中可以包含多个线程,但是至少要包含一个线程,即主线程。

Java中的线程

  Java 中使用Thread类表示一个线程。所有的线程对象都必须是Thread或其子类的对象。Thread 类中的 run 方法是该线程的执行代码。让我们来看一个实例:

public class Ticket extends Thread{
  // 重写run方法
  public void run() {
    for (int i = 0; i < 20; i++) {
      System.out.println(getName() + ": " + i);
    }
  }
}
public class TestThread {
  public static void main(String[] args) {
    // 1.创建线程
    Thread thread1 = new Ticket();
    Thread thread2 = new Ticket();

    // 2.启动线程
    thread1.start();
    thread2.start();
  }
}

运行结果如下:

 通过上面的代码和运行结果,我们可以得到:

线程运行的几个特点

  1.同一进程下不同线程的调度不由程序控制。线程的执行是抢占式的,运行的顺序和线程的启动顺序是无关的,当前运行的线程随时都可能被挂起,然后其他进程抢占运行。

  2.线程独享自己的堆栈程序计数器和局部变量。两个进程的局部变量互不干扰,各自的执行顺序也是互不干扰。

  3.两个线程并发执行。两个线程同时向前推进,并没有说执行完一个后再执行另一个。

start()方法和run()方法

  启动一个线程必须调用Thread 类的 start()方法,使该线程处于就绪状态,这样该线程就可以被处理器调度。

   run()方法是一个线程所关联的执行代码,无论是派生自 Thread类的线程类,还是实现Runnable接口的类,都必须实现run()方法,run()方法里是我们需要线程所执行的代码。

  实现多线程必须调用Thread 类的 start()方法来启动线程,使线程处于就绪状态随时供CPU调度。如果直接调用run()方法的话,只是调用了Thread类的一个普通方法,会立即执行该方法中的代码,并没有实现多线程技术。

Java中多线程的实现方法

  在Java中有三种方法实现多线程。

    第一种方法:使用Thread类或者使用一个派生自Thread 类的类构建一个线程。

    第二种方法:实现Runnable 接口来构建一个线程。(推荐使用)

    第三种方法:实现Callable 接口来构建一个线程。(有返回值)

第一种方法

  使用Thread类或者使用一个派生自Thread 类的类构建一个线程。

public class Ticket extends Thread{
  // 重写run方法
  public void run() {
    for (int i = 0; i < 20; i++) {
      System.out.println(getName() + ": " + i);
    }
  }
}
public class TestThread {
  public static void main(String[] args) {
    // 1.创建线程
    Thread thread1 = new Ticket();
    Thread thread2 = new Ticket();

    // 2.启动线程
    thread1.start();
    thread2.start();
  }
}

  看上面的代码,我们创建了一个Ticket类,它继承了Thread类,重写了Thread类的run方法。然后我们用Ticket类创建了两个线程,并且启动了它们。但我们不推荐使用这种方法,因为一个类继承了Thread类,那它就没有办法继承其他类了,这对较为复杂的程序开发是不利的。

第二种方法

  实现Runnable 接口来构建一个线程。

public class Ticket implements Runnable{
  // 重写run方法
  public void run() {
    for (int i = 0; i < 20; i++) {
      System.out.println(Thread.currentThread().getName() + ": " + i);
    }
  }
}
public class TestThread {
  public static void main(String[] args) {
    // 1.创建线程
    Ticket t1 = new Ticket();
    Ticket t2 = new Ticket();
    Thread thread1 = new Thread(t1, "买票1号");
    Thread thread2 = new Thread(t2, "买票2号");

    // 2.启动线程
    thread1.start();
    thread2.start();
  }
}

  我们创建了一个Ticket类,实现了Runnable接口,在该类中实现了run方法。在启动线程前,我们要创建一个线程对象,不同的是我们要将一个实现了Runnable接口的类的对象作为Thread类构造方法的参数传入,以构建线程对象。构造方法Thread的第二个参数用来指定该线程的名字,通过Thread.currentThread().getName()可获取当前线程的名字。

  在真实的项目开发中,推荐使用实现Runnable接口的方法进行多线程编程。因为这样既可以实现一个线程的功能,又可以更好地复用其他类的属性和方法。

第三种方法

  实现Callable 接口来构建一个线程。

public class TestThread {
  public static void main(String[] args) {
    // 1.创建Callable的实例
    Callable<String> callable = new Callable<String>() {
      @Override
      public String call() throws Exception {
        Thread.sleep(7000);
        return "我结束了";
      }
    };

    // 2.通过FutureTask接口的实例包装Callable的实例
    FutureTask<String> futureTask = new FutureTask<String>(callable);

    // 3.创建线程并启动
    new Thread(futureTask).start();

    // 4.获得结果并打印
    try {
      System.out.println(futureTask.get());
    } catch (Exception e) {
      e.printStackTrace();
    }
  }
}

  首先我们用匿名内部类创建了一个实现Callable接口的类的对象,然后通过FutureTask 的实例包装了Callable的实例,这样我们就可以通过一个Thread 对象在新线程中执行call()方法,同时又可以通过get方法获取到call()的返回值。然后创建线程并启动它,最后在线程执行完执行完call()方法后得到返回值并打印。

  我们来看一下Callable的源码:

public interface Callable<V> {
  /**
   * Computes a result, or throws an exception if unable to do so.
   *
   * @return computed result
   * @throws Exception if unable to compute a result
   */
  V call() throws Exception;
}

  从Callable 的定义可以看出,Callable接口是一个泛型接口,它定义的call()方法类似于Runnable 的run()方法,是线程所关联的执行代码。但是与run()方法不同的是,call()方法具有返回值,并且泛型接口的参数V指定了call()方法的返回值类型。同时,如果call()方法得不到返回值将会抛出一个异常,而在Runnable的run()方法中不能抛出异常。

如何获得call()方法的返回值

  通过Future接口来获取。Future接口定义了一组对 Runnable 或者Callable 任务的执行结果进行取消、查询、获取、设置的操作。其中get方法用于获取call()的返回值,它会发生阻塞,直到call()返回结果。

这样的线程调用与直接同步调用函数有什么差异

  在上面的例子中,通过future.get()获取 call()的返回值时,由于call方法中会 sleep 7s,所以在执行future.get()的时候主线程会被阻塞而什么都不做,等待call()执行完并得到返回值。但是这与直接调用函数获取返回值还是有本质区别的。

  因为call()方法是运行在其他线程里的,在这个过程中主线程并没有被阻塞,还是可以做其他事情的,除非执行future.get()去获取 call()的返回值时主线程才会被阻塞。所以当调用了Thread.start()方法启动 Callable 线程后主线程可以执行别的工作,当需要call()的返回值时再去调用future.get()获取,此时call()方法可能早已执行完毕,这样就可以既确保耗时操作在工作线程中完成而不阻挡主线程,又可以得到线程执行结果的返回值。而直接调用函数获取返回值是一个同步操作,该函数本身就是运行在主线程中,所以一旦函数中有耗时操作,必然会阻挡主线程。

以上就是浅析 Java多线程的详细内容,更多关于Java多线程的资料请关注我们其它相关文章!

(0)

相关推荐

  • Java中的多线程一定就快吗?

    并发编程与多线程编程 要了解并发编程,首先要懂得与并行这个概念进行区分.并行是指两个事件同时进行,并发是CPU切换速度快,看起来像是每个任务同时进行一样.多线程是实现并发编程的一种方式,假设一个场景,在广州地铁高峰时段,一群人涌进地铁里,在不同的闸机口刷卡进去.在这个场景里,进地铁就是任务,每个人可以看出是并发的,而多个刷卡闸机口就是多线程.   并发编程的本质目的是为了充分利用CPU,让程序运行得更快.然而,并不是启动更多的线程就能让程序最大限度地并发执行.在进行并发编程时,如果希望通过多线程

  • Java实现多线程轮流打印1-100的数字操作

    首先打印1-100数字如果用一个单线程实现那么只要一个for循环即可,那么如果要用两个线程打印出来呢?(一个线程打印奇数,一个线程打印偶数)于是大家会想到可以通过加锁实现,但是这样的效率是不是不高?这里我用一个变量来控制两个线程的输出 public class ThreadTest { volatile int flag=0; public void runThread() throws InterruptedException{ Thread t1=new Thread(new Thread1

  • Java多线程中Lock锁的使用总结

    多核时代 摩尔定律告诉我们:当价格不变时,集成电路上可容纳的晶体管数目,约每隔18个月便会增加一倍,性能也将提升一倍.换言之,每一美元所能买到的电脑性能,将每隔18个月翻两倍以上.然而最近摩尔定律似乎遇到了麻烦,目前微处理器的集成度似乎到了极限,在目前的制造工艺和体系架构下很难再提高单个处理器的速度了,否则它就被烧坏了.所以现在的芯片制造商改变了策略,转而在一个电路板上集成更多的处理器,也就是我们现在常见的多核处理器. 这就给软件行业带来麻烦(也可以说带来机会,比如说就业机会,呵呵).原来的情况

  • Java多线程CAS操作原理代码实例解析

    CAS操作号称无锁优化,也叫作自旋:对于一些常见的操作需要加锁,然后jdk就提供了一些以Atomic开头的类,这些类内部自动带了锁,当然这里的锁并非是用synchronized来实现的,而是通过CAS操作来实现的: 一.下面是 AtomicInteger 的使用: package com.designmodal.design.juc01; import java.util.ArrayList; import java.util.List; import java.util.concurrent.

  • Java 多线程传值的四种方法

    其实大家都知道多线程传值有三种方式: 1:通过构造方法传递数据 2:通过变量和方法传递数据 3:通过回调函数传递数据 那么博主有个非常变态的需求,所以找出了第四种实现方式,先看效果图: 动态Cron4j调度器,我曾经发过类似的文章,可以去搜索一下. 点击执行走下边的代码,然后根据类名反编译 public static void executeCron4j(String packageClass){ try { Object taskObj = classNewInstance(packageCl

  • Java多线程锁机制相关原理实例解析

    上下文:程序运行需要的环境(外部变量) 上下文切换:将之前的程序需要的外部变量复制保存,然后切换到新的程序运行环境 系统调用:(用户态陷入操作系统,通过操作系统执行内核态指令,执行完回到用户态)用户态--内核态--用户态:两次上下文切换 线程wait()方法:将自身加入等待队列,发生了一次上下文切换 notify()方法:将线程唤醒,也发生了上下文切换 Java线程中的锁:偏向锁.轻量级锁.重量级锁. 注意:偏向锁和轻量级锁都没有发生竞争,重量级锁发生了竞争. 偏向锁:可重入和经常使用某一个线程

  • Java Lambda表达式原理及多线程实现

    1.使用Lambda表达式实现多线程 public static void main(String[] args) { //使用匿名内部类的方式,实现多线程 new Thread(new Runnable() { @Override public void run() { System.out.println(Thread.currentThread().getName() + "新线程创建了!"); } }).start(); //使用Lambda表达式,实现多线程 new Thre

  • 浅谈java多线程编程

    一.多线程的优缺点 多线程的优点: 1)资源利用率更好 2)程序设计在某些情况下更简单 3)程序响应更快 多线程的代价: 1)设计更复杂 虽然有一些多线程应用程序比单线程的应用程序要简单,但其他的一般都更复杂.在多线程访问共享数据的时候,这部分代码需要特别的注意.线程之间的交互往往非常复杂.不正确的线程同步产生的错误非常难以被发现,并且重现以修复. 2)上下文切换的开销 当CPU从执行一个线程切换到执行另外一个线程的时候,它需要先存储当前线程的本地的数据,程序指针等,然后载入另一个线程的本地数据

  • java利用多线程和Socket实现猜拳游戏

    本文实例为大家分享了利用多线程和Socket实现猜拳游戏的具体代码,供大家参考,具体内容如下 实例:猜拳游戏 猜拳游戏是指小时候玩的石头.剪刀.布的游戏.客户端与服务器的"较量",服务器会自动随机产生出石头.剪刀或者布,客户端则由用户输入石头.剪刀.布之一,然后服务器通过比较告知比较结果.比如,客户端出石头,而服务器出的是剪刀,那么,服务器就会告知客户端,用户赢了.猜拳游戏也要制作一个多线程的程序,让多个客户端能够共同参与游戏.在该程序中,要包括猜拳的业务逻辑的线程类.客户端类.服务器

  • Java多线程通信:交替打印ABAB实例

    使用wait()和notify()实现Java多线程通信:两个线程交替打印A和B,如ABABAB public class Test { public static void main(String[] args) { final PrintAB print = new PrintAB(); new Thread(new Runnable() { public void run(){ for(int i=0;i<5;i++) { print.printA(); } } }).start(); n

随机推荐