实现多线程的三种方式

p.MsoNormal,li.MsoNormal,div.MsoNormal { margin: 0cm; margin-bottom: .0001pt; text-align: justify; font-size: 10.5pt; font-family: "Calibri", "sans-serif" }
h1 { margin-right: 0cm; margin-left: 0cm; font-size: 24.0pt; font-family: 宋体 }
h2 { margin-right: 0cm; margin-left: 0cm; font-size: 18.0pt; font-family: 宋体 }
h3 { margin-right: 0cm; margin-left: 0cm; font-size: 13.5pt; font-family: 宋体 }
h4 { margin-right: 0cm; margin-left: 0cm; font-size: 12.0pt; font-family: 宋体 }
p.MsoHeader,li.MsoHeader,div.MsoHeader { margin: 0cm; margin-bottom: .0001pt; text-align: center; border: none; padding: 0cm; font-size: 9.0pt; font-family: "Calibri", "sans-serif" }
p.MsoFooter,li.MsoFooter,div.MsoFooter { margin: 0cm; margin-bottom: .0001pt; font-size: 9.0pt; font-family: "Calibri", "sans-serif" }
a:link,span.MsoHyperlink { color: blue; text-decoration: underline }
a:visited,span.MsoHyperlinkFollowed { color: #954F72; text-decoration: underline }
p.MsoAcetate,li.MsoAcetate,div.MsoAcetate { margin: 0cm; margin-bottom: .0001pt; text-align: justify; font-size: 9.0pt; font-family: "Calibri", "sans-serif" }
span.Char { font-family: "Calibri", "sans-serif" }
span.Char0 { font-family: "Calibri", "sans-serif" }
span.Char1 { font-family: "Calibri", "sans-serif" }
span.msoIns { text-decoration: underline; color: teal }
span.msoDel { text-decoration: line-through; color: red }
.MsoChpDefault { font-size: 10.0pt }
div.Section1 { }

 

Java 多线程实现方式主要有三种(区别):

1)继承Thread类:线程执行完后都没有返回值。

2)实现Runnable接口:线程执行完后都没有返回值。

3)使用ExecutorService、Callable、Future实现有返回结果的多线程。这一种是带有返回值的。

1、继承Thread类实现多线程
继承Thread类的方法尽管被我列为一种多线程实现方式,但Thread本质上也是实现了Runnable接口的一个实例,它代表一个线程的实例,并且,启动线程的唯一方法就是通过Thread类的start()实例方法。start()方法是一个native方法,它将启动一个新线程,并执行run()方法。这种方式实现多线程很简单,通过自己的类直接extend Thread,并复写run()方法,就可以启动新线程并执行自己定义的run()方法。

public class MyThread extends Thread {

  public void run() {

    System.out.println("MyThread.run()");

  }

}

在合适的地方启动线程如下:

MyThread myThread1 = new MyThread();

MyThread myThread2 = new MyThread();

myThread1.start();

myThread2.start();

2、实现Runnable接口方式实现多线程
如果自己的类已经extends另一个类,就无法直接extends Thread,此时,必须实现一个Runnable接口public class MyThread extends OtherClass implements Runnable {

  public void run() {

    System.out.println("MyThread.run()");

  }

}

为了启动MyThread,需要首先实例化一个Thread,并传入自己的MyThread实例:

MyThread myThread = new MyThread();

Thread thread = new Thread(myThread);

thread.start();

事实上,当传入一个Runnable target参数给Thread后,Thread的run()方法就会调用target.run(),参考JDK源代码:

public void run() {

  if (target != null) {

     target.run();

  }

}

3、使用ExecutorService、Callable、Future实现有返回结果的多线程
ExecutorService、Callable、Future这个对象实际上都是属于Executor框架中的功能类。(想要详细了解Executor框架的可以访问http://www.javaeye.com/topic/366591 ,这里面对该框架做了很详细的解释。)

返回结果的线程是在JDK1.5中引入的新特征,确实很实用,有了这种特征我就不需要再为了得到返回值而大费周折了,而且即便实现了也可能漏洞百出。
可返回值的任务必须实现Callable接口,类似的,无返回值的任务必须Runnable接口。执行Callable任务后,可以获取一个Future的对象,在该对象上调用get就可以获取到Callable任务返回的Object了,再结合线程池接口ExecutorService就可以实现传说中有返回结果的多线程了。下面提供了一个完整的有返回结果的多线程测试例子,在JDK1.5下验证过没问题可以直接使用。代码如下:

p.MsoNormal,li.MsoNormal,div.MsoNormal { margin: 0cm; margin-bottom: .0001pt; text-align: justify; font-size: 10.5pt; font-family: "Calibri", "sans-serif" }
p.MsoHeader,li.MsoHeader,div.MsoHeader { margin: 0cm; margin-bottom: .0001pt; text-align: center; border: none; padding: 0cm; font-size: 9.0pt; font-family: "Calibri", "sans-serif" }
p.MsoFooter,li.MsoFooter,div.MsoFooter { margin: 0cm; margin-bottom: .0001pt; font-size: 9.0pt; font-family: "Calibri", "sans-serif" }
span.Char { }
span.Char0 { }
span.keyword { }
span.comment { }
span.annotation { }
span.string { }
span.number { }
span.apple-converted-space { }
div.Section1 { }
ol { margin-bottom: 0cm }
ul { margin-bottom: 0cm }

1.   import java.util.concurrent.*;

2.   import java.util.Date;

3.   import java.util.List;

4.   import java.util.ArrayList;

5.

6.   /**

7.   * 有返回值的线程

8.   */

9.   @SuppressWarnings("unchecked")

10.  public class Test {

11.  public static void main(String[] args) throws ExecutionException,

12.      InterruptedException {

13.     System.out.println("----程序开始运行----");

14.     Date date1 = new Date();

15.

16.     int taskSize = 5;

17.     // 创建一个线程池

18.     ExecutorService pool = Executors.newFixedThreadPool(taskSize);

19.     // 创建多个有返回值的任务

20.     List<Future> list = new ArrayList<Future>();

21.     for (int i = 0; i < taskSize; i++) {

22.      Callable c = new MyCallable(i + " ");

23.      // 执行任务并获取Future对象

24.      Future f = pool.submit(c);

25.      // System.out.println(">>>" + f.get().toString());

26.      list.add(f);

27.     }

28.     // 关闭线程池

1.      pool.shutdown();

// 获取所有并发任务的运行结果

2.      for (Future f : list) {

3.       // 从Future对象上获取任务的返回值,并输出到控制台

4.       System.out.println(">>>" + f.get().toString());

5.      }

6.

7.      Date date2 = new Date();

8.      System.out.println("----程序结束运行----,程序运行时间【"

9.        + (date2.getTime() - date1.getTime()) + "毫秒】");

10.  }

11.  }

12.

13.  class MyCallable implements Callable<Object> {

14.  private String taskNum;

15.

16.  MyCallable(String taskNum) {

17.     this.taskNum = taskNum;

18.  }

19.

20.  public Object call() throws Exception {

21.     System.out.println(">>>" + taskNum + "任务启动");

22.     Date dateTmp1 = new Date();

23.     Thread.sleep(1000);

24.     Date dateTmp2 = new Date();

25.     long time = dateTmp2.getTime() - dateTmp1.getTime();

26.     System.out.println(">>>" + taskNum + "任务终止");

27.     return taskNum + "任务返回运行结果,当前任务时间【" + time + "毫秒】";

28.  }

29.  }

代码说明:

上述代码中Executors类,提供了一系列工厂方法用于创先线程池,返回的线程池都实现了ExecutorService接口。

public static ExecutorService
newFixedThreadPool(int nThreads)

创建固定数目线程的线程池。

public static ExecutorService newCachedThreadPool()

创建一个可缓存的线程池,调用execute 将重用以前构造的线程(如果线程可用)。如果现有线程没有可用的,则创建一个新线程并添加到池中。终止并从缓存中移除那些已有 60 秒钟未被使用的线程。

public static ExecutorService newSingleThreadExecutor()

创建一个单线程化的Executor。

public static ScheduledExecutorService
newScheduledThreadPool(int corePoolSize)

创建一个支持定时及周期性的任务执行的线程池,多数情况下可用来替代Timer类。

ExecutoreService提供了submit()方法,传递一个Callable,或Runnable,返回Future。如果Executor后台线程池还没有完成Callable的计算,这调用返回Future对象的get()方法,会阻塞直到计算完成。

时间: 2024-09-29 23:30:38

实现多线程的三种方式的相关文章

线程的状态以及创建多线程的三种方式

首先了解一下线程的五种状态: 新建状态: 新建状态是指new之后,即新创建了一个线程的时候,此时并未运行任何线程方法体内的程序代码. 就绪状态: 简单来说就是指程序调用了start()之后,线程就得到了启动,代表线程进入了就绪状态,但是此时并不代表它会立刻去执行run()方法体内的程序代码,而是随时等待cpu的调度. 运行状态: 获得cpu的时间后,调用run()方法,进入运行状态. 阻塞状态: 由于某种原因放弃了cpu的会用权力,暂时停止运行,等待再次被调用. 死亡状态: 线程正常执行完毕,或

java实现多线程的三种方式

java中实现多线程的方法有两种:继承Thread类和实现runnable接口 1.继承Thread类,重写父类run()方法   public class thread1 extends Thread {           public void run() {                 for (int i = 0; i < 10000; i++) {                         System.out.println("我是线程"+this.get

实现Java多线程的三种方式

1. JAVA多线程实现方式    JAVA多线程实现方式主要有三种: 继承Thread类 实现Runnable接口 使用ExecutorService.Callable.Future实现有返回结果的多线程. 其中前两种方式线程执行完后都没有返回值,只有最后一种是带返回值的. 2. 继承Thread类实现多线程 继承Thread类的方法尽管被我列为一种多线程实现方式,但 Thread本质上也是实现了Runnable接口的一个实例,它代表一个线程的实例,并且,启动线程的唯一方法就是通过Thread

多线程编程(三种方式)

一.NSThread DownloadFile是一个继承NSObject的类,当中的runDownload方法是需要放到线程中执行:ThreadDownload则是一个继承NSThread的子线程类,配合隐式创建线程的方式,执行threadD对象中的run:函数并传入字符串@"download3" // download DownloadFile *download = [[DownloadFile alloc] init]; // 动态创建线程 NSThread *downloadT

Java 实现多线程的三种方式

import java.util.concurrent.Callable; import java.util.concurrent.FutureTask; public class Main { public static void main(String[] args) { //方法一:继承Thread int i = 0; // for(; i < 100; i++){ // System.out.println(Thread.currentThread().getName() + "

【java多线程】多线程的创建三种方式--笔记

申明:线程的概念以及进程的相关概念,可以参考网络上其他资料,这里只讨论多线程是怎么实现. 一.多线程的简单理解 明白什么是多线程,小生通俗一点的理解为:在一个程序里,我想同时让这个程序完成多个任务. 比如:让主函数 main 在打印1~100之间的所有整数的时候,要求在主函数打印到 20 的时候,再运行另一个类里的程序,让它打印10~100之间的所有整数. 这里忽略同进程内的多线程之间的抢占时间问题,上面的举例需求是要求一个程序只要有发生同时运行俩个程序的情况就行,即不准出现无论程序跑多少次都是

java多线程开启的三种方式

1.继承Thread类,新建一个当前类对象,并且运行其start()方法 1 package com.xiaostudy.thread; 2 3 /** 4 * @desc 第一种开启线程的方式 5 * @author xiaostudy 6 * 7 */ 8 public class Demo1_Thread extends Thread { 9 10 public void run() { 11 for (int i = 0; i < 10; i++) { 12 System.out.pri

多线程的实现三种方式

多线程的实现三种方式:1 继承thread类,重写run方法 继承thread方法就可以i调用thread类的start方法,,start方法调用java natvie start0();这个是调用操作系统的方法,start方法 package com.cxy; class Mythread01 extends Thread{ @Override public void run(){ System.out.println("iii"); }} public class Mythread

IOS 多线程,线程同步的三种方式

一般情况下我们使用线程,在多个线程共同访问同一块资源.为保护线程资源的安全和线程访问的正确性. 在IOS中我们一般情况下使用以下三种线程同步代码方式: 第一种和第二种代码同步的使用方法,一般情况下我们只需要使用NSLock和NSCondition申明2个属性.然后给此属性赋对应的值.那么即可作为安全防控的线程手段. 同时也可以保证线程的资源安全. 1:NSLock方式 [xxxlock   lock] //上锁 同步代码块 [xxxlock   unlock]//解锁 2:NSCondition