转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/6560057.html 

一:继承Thread类创建线程

1:继承Thread类定义线程子类;

2:重写run()方法,定义线程的操作;

3:通过创建的线程子类对象.start() 启动线程。

package com.thread;
public class FirstThreadTest extends Thread{
public void run()
{
System.out.println(Thread.currentThread().getName());
}
public static void main(String[] args)
{
new FirstThreadTest().start();
} }

二:实现Runnable接口创建线程

1:实现Runnable接口定义线程类;

2:重写run()方法;

3:创建Thread对象:把上面 实现runnable接口的线程类 的对象作为构造参数,创建出线程对象;

4:由thread对象的start()方法启动线程;

public class RunnableThreadTest implements Runnable
{
public void run()
{
System.out.println(Thread.currentThread().getName())
}
public static void main(String[] args)
{
new Thread(new RunnableThreadTest()).start();
} }

三:通过Callable接口和Future创建线程

1:创建Callable接口的实现类,并指明返回值类型;

2:在实现类中重写call()方法,并返回结果;

3:创建Future<V>类型的list接收线程的预期结果: List<Future<String>> results = new ArrayList<Future<String>>();

4:通过线程池启动线程,并且把线程返回结果add到list中;

5:遍历list时,通过每个元素的 get() 方法获取每个线程的实际运行结果值

//实现callable接口,定义线程类
class TaskWithResult implements Callable<String>{
private int id;
public TaskWithResult(int id){
this.id = id;
}
//重写call()方法,切记抛出异常,并返回值
public String call() throws Exception {
return "result of task:" + id;
}
public class TestCallable {
public static void main(String[] args) {
//创建线程池
ExecutorService es = Executors.newCachedThreadPool();
//创建预期结果集合
List<Future<String>> results = new ArrayList<Future<String>>(); for(int i=0;i<5;i++){
//通过线程池启动线程,把线程结果保存到预期结果集合
results.add(es.submit(new TaskWithResult(i)));
}
//遍历结果集合
for(Future<String> fs : results){
try {
//通过 结果.get() 方法获取每个线程运行结束后的返回值。并处理异常
System.out.println(fs.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
}
}
}

另一种启动callable实现类线程的办法:

Callable<Result> ca = new Callable<Result>() {
public Result call() throws Exception {
System.out.println("当前线程名称是:" + Thread.currentThread().getName());
Thread.sleep(2000L);
return new Result("callable 执行完毕");
}
};
//包装对象
FutureTask<Result> ft = new FutureTask<Result>(ca);
try {
//启动线程执行
new Thread(ft).start();
//获取结果
System.out.println(ft.get().getMsg());
} catch (Exception e) {
e.printStackTrace();
}
}

四:对比

1:继承Thread类创建线程:其run()方法没有返回值;通过start方法启动;由于Java不能允许多继承,一个类如果需要继承其他类就不能再定义为线程类了;run方法中的异常必须捕获并处理;

2:实现runnable接口创建线程:其run()方法没有返回值;通过作为thread类的构造函数的参数被包装成thread类对象,再通过start方法启动(由此,可以同一个任务对象交给多个thread对象来运行,实现资源的共享以及并行处理);由于接口可以多实现,一个类可以继承其他类的同时实现runnable接口成为线程类;run方法中的异常必须捕获并处理;

3:实现Callable接口创建线程:其call()方法有返回值;一般通过线程池来启动线程,也可以先包装成为FutureTask对象,然后再由futuretask对象包装成Thread对象通过start方法启动;call()方法可以抛出异常,直到在结果处通过get()方法获取结果时再处理异常;在主线程启动实现callable创建的线程时可以获得一个Future对象作为异步处理的预期结果,在通过future.get()方法时检测启动的线程是否已完成并返回结果,是则得到结果,否则阻塞主线程等待任务线程完成并返回结果;

最新文章

  1. Oracle中改变表的Owner和tablespace
  2. hadoop学习笔记之倒排索引
  3. Junit的最简单样例:Hello world!
  4. 百度之星资格赛,hdu 4825 XOR SUM
  5. 环境配置与JBoss安装-EJB3.0入门经典学习笔记(1)
  6. Recursive - leetcode [递归]
  7. css书写规范以及如何写出赏心悦目的代码
  8. 字符串转码中文乱码问题的进一步理解 UTF-8 GBK转码
  9. jxl 读取xls,并转为二维数组可进行保存
  10. CentOS 安装Mosquitto及测试
  11. tpshop全局公共方法
  12. ajax 请求发出了,数据更改了,但是没进入success 函数 把success 换成 complete
  13. [转]How to nest transactions nicely - &quot;begin transaction&quot; vs &quot;save transaction&quot; and SQL Server
  14. Android实例剖析笔记(四)
  15. IntentService介绍
  16. Vuex 基本概念
  17. 前端资源管理工具sourcetree
  18. SQL查询某一字段重复的数据
  19. 【问题记录】mysql设置任意ip访问
  20. 使用SpringMVC参数传递时,解决get请求时中文乱码的问题

热门文章

  1. junit5了解一下
  2. ASP.NET MVC:无法向会话状态服务器发出会话状态请求
  3. Go语言之进阶篇获取文件属性
  4. Easyui1.3.4+IIS6.0+IE8兼容问题解决
  5. VS单元测试中Assert类的用法
  6. .Net(c#)打印--多页打印
  7. 汇总c#.net常用函数和方法集
  8. 弹出层框架layer快速使用
  9. 【Scala】Java-Scala-单例模式实现方法
  10. jquery操作select(取值,设置选中)(转)