文档章节

并发编程基础一--继承Thread,实现Runnable,实现Callable和Future.

Zhao-Qian
 Zhao-Qian
发布于 2013/01/22 17:18
字数 1589
阅读 428
收藏 3

一.继承Thread的线程

1.定义Thread类的子类,并重写该类的run()方法,该run()方法就是代表了线程要完成的任务.因此run方法是线程执行体.
2.创建Thread子类的实例,就是创建了线程对象.
3.调用线程对象的start()方法来启动该线程.

package org.credo.thread;

/**
 * <p>Description:继承Thread类创建线程类. </p>
 * @author <a href="zhaoqianjava@qq.com">Credo</a>
 */
public class FirstThread extends Thread{
	private int i;
	
	/* 
	 * 重写run()方法,run方法的方法体就是线程的执行体.
	 */
	public void run(){
		for(;i<10;i++){
			//当线程类继承Thread类时,直接使用this即可获取当前线程
			//Thread对象的getName()返回当前线程的名字
			//因此可以直接调用getName()方法返回当前线程的名字
			System.out.println(getName()+" "+i);
		}
	}
	
	public static void main(String[] args) {
		for(int i=0;i<10;i++){
			//调用Thread的currentThread()方法获得当前线程
			System.out.println(Thread.currentThread().getName()+" "+i);
			if(i==2){
				//创建并启动第一个线程,使用start方法启动线程.
				new FirstThread().start();
				//创建并启动第二个线程
				new FirstThread().start();
			}
		}
	}
}
上面的程序使用到了线程的两个方法:
  • Thread.currentThread():currentThread()是Thread类的静态方法,该方法总是返回当前正在执行的线程对象.
  • getName():该方法是Thread的实例方法,该方法返回调用该方法的线程名字.
程序也可以通过setName去为线程设定名字.
我们看下线程的输出情况.
main 0
main 1
main 2
Thread-0 0
Thread-0 1
Thread-0 2
Thread-0 3
Thread-0 4
Thread-1 0
main 3
Thread-1 1
Thread-0 5
Thread-1 2
main 4
Thread-1 3
main 5
Thread-0 6
main 6
Thread-1 4
main 7
Thread-0 7
main 8
Thread-1 5
main 9
Thread-0 8
Thread-0 9
Thread-1 6
Thread-1 7
Thread-1 8
Thread-1 9
通过上面的输出可以看得出,3个线程输出的"i 变量"是不连续的.注意,i变量是FirstThread的实例属性,而不是局部变量.但因为程序每次创建线程对象时都需要创建一个FirstThread对象,所以他们不能共享该实例属性,i.

因此就可以有个结论:使用继承Thread类的方法来创建线程类时,多个线程之间无法共享线程类的实例变量.

二.实现Runnable接口创建线程类

实现Runnable接口来创建并启动多线程的步骤如下:
1.定义Runnable接口的实现类,并重写该接口的run()方法,该run()方法的方法体同样是该线程的线程执行体.
2.创建Runnable实现类的实例,并以此实例作为Thread的target来创建Thread对象,该Thread对象才是真正的线程对象.

示例代码:

package org.credo.thread;

public class FirstRunnable implements Runnable{

	private int i;
	
	public void run(){
		for(;i<10;i++){
			//当线程类实现Runnable接口时,
			//如果想获取当前线程,只能用Thread.currentThread()方法.
			System.out.println(Thread.currentThread().getName()+" "+i);
		}
	}
	
	public static void main(String[] args) {
		for(int i=0;i<10;i++){
			System.out.println(Thread.currentThread().getName()+" "+i);
			if(i==2){
				FirstRunnable fr=new FirstRunnable();
				//通过new Thread(target,name)方法创建新线程
				new Thread(fr,"newThread11").start();
				new Thread(fr, "newThread22").start();
			}
		}
	}
}
输出让我有点额...不太理解,就那两个0....按理说应该是1,2线程都是0~9顺序的.为毛啊?有高手解释没?

main 0
main 1
main 2
main 3
main 4
newThread11 0
newThread22 0
newThread22 2
main 5
main 6
main 7
main 8
main 9
newThread22 3
newThread22 4
newThread11 1
newThread11 6
newThread22 5
newThread11 7
newThread22 8
newThread11 9
按理来说,i应该是连续的,也就是采用runnable接口的方式创建的多个线程可以共享线程类的实例属性.这是因为在这种方式下,程序所创建的Runnable对象只是线程的target,而多个线程可以共享同一个target,所以多个线程可以共享同一个线程类(实际上应该是线程的target类)的实例属性.

三:使用Callable和Future创建线程

前面可以看到,通过实现Runnable接口创建多线程的时候,Thread类的作用就是把run()方法包装成线程执行体.
那么是否可以把任意方法都包装成线程执行体呢?
Java貌似不可以,但C#貌似可以.
因此从java5开始,java提供了Callable接口,该接口其实就是Runnable接口的增强版.Callable接口提供了一个call()方法可以作为线程执行体,但call方法比run()方法功能更强大.

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

因此我们可以提供一个callable对象作为Thread的target.而该线程的线程执行体就是该Callable对象的call方法.但问题来了,callable是java5新增的一个并不是runnable接口的子接口,因此不能直接作为Thread的target,而且call方法还有一个返回值,call方法并不是直接调用.它是作为线程执行体被调用的.那么如果最后获取其值呢?

Java提供了Future接口来代表Callable接口里的call方法的返回值,并为future提供了一个FutureTask实现类.该类实现了Future接口,并且实现了Runnable接口--它可作为Thread类的target.

步骤基本如下:

1.创建Callable接口的实现类,并实现call()方法,该call方法将作为线程执行体.能抛异常,能有返回值.
2.创建Callable实现类的实例,使用FutureTask类来包装Callable对象,该FutureTask对象封装了该Callable对象的call()方法的返回值.
3.使用FutureTask对象作为Thread对象的target创建并启动线程.
4.调用FutureTask对象的get()方法获取子线程的返回值.

package org.credo.thread;

import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;

public class CallableAndFuture implements Callable<Integer>{

	public static void main(String[] args) {
		CallableAndFuture caf=new CallableAndFuture();
		FutureTask<Integer> future=new FutureTask<Integer>(caf);
		for(int i=0;i<100;i++){
			System.out.println(Thread.currentThread().getName()+"的I的值是:"+i);
			if(i==20){
				new Thread(future, "有返回值的线程").start();
			}
		}
		try {
			System.out.println("子线程的返回值:"+future.get());
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	@Override
	public Integer call() throws Exception {
		int i=0;
		for(;i<100;i++){
			System.out.println(Thread.currentThread().getName()+"的I的值为:"+i);
		}
		return i;
	}

}

总结:掌握主线,支线的API查阅也就知道了.一般情况下不去使用继承Thread.太蛋疼了那样.就用Runnable和callable.


© 著作权归作者所有

共有 人打赏支持
Zhao-Qian
粉丝 310
博文 158
码字总数 238167
作品 0
朝阳
技术主管
Java多线程、并发杂记

多线程涉及的类可以分为以下几类: 可执行对象:最基本的多线程 执行器:简化多线程编程 工具类 容器 并发控制 一、可执行对象: 1、Runnable: 执行单位:Thread 创建线程的两种方式(来自于...

rathan0
2016/02/17
60
0
Thread,Runnable,Callable. 多线程

编写多线程程序是为了实现多任务的并发执行,从而能够更好地与用户交互。一般有三种方法,Thread,Runnable,Callable.   Runnable和Callable的区别是,   (1)Callable规定的方法是call(...

千惊万喜
2016/06/30
12
0
Callable,Runnable比较及用法以及创建线程的4种方法

编写多线程程序是为了实现多任务的并发执行,从而能够更好地与用户交互。一般有三种方法,Thread,Runnable,Callable. Runnable和Callable的区别是, (1)Callable规定的方法是call(),Runnable...

geek_loser
2017/10/25
0
0
java多线程实现方法

编写多线程程序是为了实现多任务的并发执行,从而能够更好地与用户交互。一般有三种方法,Thread,Runnable,Callable. Runnable和Callable的区别是, (1)Callable规定的方法是call(),Runnable...

张欢19933
2016/02/23
31
0
Java多线程基础篇(02)-多线程的实现

1.概要 JAVA多线程实现方式主要有三种:继承Thread类、实现Runnable接口、使用ExecutorService、Callable、Future实现有返回结果的多线程。其中前两种方式线程执行完后都没有返回值,只有最后...

kukudeku
2016/09/19
41
0

没有更多内容

加载失败,请刷新页面

加载更多

jQuery学习笔记180923

jQuery 操作 CSS jQuery 拥有若干进行 CSS 操作的方法。我们将学习下面这些: addClass() - 向被选元素添加一个或多个类 removeClass() - 从被选元素删除一个或多个类 toggleClass() - 对被选...

颖伙虫
21分钟前
0
0
[python] colorama 模块 - 改变控制台输出文本的颜色

除了使用 PyQt 这样的图形化开发框架外,基本上 python 程序都是跑在控制台中的。很多时候,单纯使用黑白的文字不能很好地突出我们要显示的信息。有时候我们需要将错误的提示使用红色标注,而...

cometeme
25分钟前
0
0
Makefile 学习 2 - 基于若干 Blog 的汇总

基于若干 Blog 汇总的 makefile 教程 陈皓 https://blog.csdn.net/haoel/article/details/2886 Makefile 进阶 1. Makefile 中的内容 显式规则。显式规则说明了,如何生成一个或多的的目标文件...

公孙衍
41分钟前
0
0
NIO与BIO的区别、NIO的运行原理和并发使用场景

NIO(Non-blocking I/O,在Java领域,也称为New I/O),是一种同步非阻塞的I/O模型,也是I/O多路复用的基础,已经被越来越多地应用到大型应用服务器,成为解决高并发与大量连接、I/O处理问题的...

Java干货分享
59分钟前
1
0
72.告警系统邮件引擎 运行告警系统

20.23/20.24/20.25 告警系统邮件引擎 20.26 运行告警系统 20.23/20.24/20.25 告警系统邮件引擎 邮件首先要有一个mail.py,以下。 因为我们之前zabbix的时候做过,就可以直接拷贝过来 mail.s...

王鑫linux
今天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部