JAVA并发编程学习笔记------线程的三种创建方式

创建线程一般有如下几个方式:

1. 通过继承Thread类来创建一个线程:

/**
 * 步骤1:定义一个继承Thread类的子类
 * 步骤2:构造子类的一个对象
 * 步骤3:启动线程:
 * */
public class ThreadTest{
	public static void main(String[] args) {
		//构造子类对象
		SubThread subThread = new SubThread();
		//启动线程
		subThread.start();
	}
}
//定义继承Thread的子类
class SubThread extends Thread{
	public void run(){
		System.out.println("Thread test.......");
	}
}

2. 通过实现Runnable接口来创建一个线程(静态代理模式)

/**
 * 静态代理模式的步骤
 * 步骤一: 创建真实角色
 * 步骤二: 创建代理角色,持有真实角色的引用
 * 步骤三: 两者实现相同的接口
 **/
public class RunnableTest {
	public static void main(String[] args) {
		//创建一个类对象
		Runnable subRunnable = new SubRunnable();
		//由Runnable创建一个Thread对象
		Thread  subThread = new Thread(subRunnable);
		//启动线程
		subThread.start();
	}
}

//创建实现Runnable接口的类
class SubRunnable implements Runnable{
	@Override
	public void run() {
		System.out.println("Runnable Test......");
	}
}

3、通过实现Callable接口来创建一个线程(可返回值,对外声明异常等)

/*
 * 步骤一:创建Callable接口的实现类,并重写call()方法
 * 步骤二:借助执行调度服务ExecutorService获取Future对象
 * 步骤三: get()方法获取值
 * 步骤四: shutdownNow()方法停止服务
 */
public class CallableTest {
	public static void main(String[] args) {
		//借助执行调度服务ExecutorService获取Future对象
		ExecutorService ser = Executors.newFixedThreadPool(2);
		Race test1 = new Race("LiLei");
		Race test2 = new Race("HanMeimei");
		Future<String> res1 = ser.submit(test1);
		Future<String> res2 = ser.submit(test2);

		try {
			//get()方法获取值
			System.out.println(res1.get());
			System.out.println(res2.get());
		} catch (InterruptedException e) {
			e.printStackTrace();
		} catch (ExecutionException e) {
			e.printStackTrace();
		}
		//shutdownNow()方法停止服务
		ser.shutdownNow();
	}
}

//创建Callable接口的实现类
class Race implements Callable<String> {
	private String name;

	public Race(String name) {
		super();
		this.name = name;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	@Override
	public String call() throws Exception {
		return "Hello "+ name +"......";
	}
}

总结:

1、推荐使用接口,原因如下:

  1)避免单继承的局限性;

  2)便于共享资源

2、两种实现接口的方式(Runnable和Callable)的区别:

  1)Callable的启动方法是call(),Runnable的启动方法是run()

  2)Callable的任务执行后可返回值,而Runnable的任务无返回值

  3)Callable的call方法可以抛出异常,Runnable的run方法不可以

  4)Callable任务可以获取Future对象,这个对象提供了检查计算是否完成的方法,以等待计算的完成,并检索计算的结果。

     通过Future对象可以了解任务执行情况,可取消任务的执行,还可获取执行结果。

   

时间: 2024-10-23 03:44:38

JAVA并发编程学习笔记------线程的三种创建方式的相关文章

Java并发编程学习笔记(一)线程安全性 1

什么是线程安全性: 要编写线程安全的代码,其核心在于要对状态访问操作进行管理,特别是对共享的和可变的状态的访问."共享"意味着变量可以由多个线程同时访问,而"可变"则意味着变量的值在其生命周期内可以发生变化. 一个对象是否需要线程安全的,取决于他是否被多个线程访问.这指的是在程序中访问对象的方式,而不是对象要实现的功能.要使得对象时线程安全的,需要采用同步机制来协同对对象可变状态的访问.如果无法实现协同,那么可能导致数据破坏以及其他不该出现的结果. 如果当多个线程访

Java并发编程学习笔记

Java编程思想,并发编程学习笔记. 一.基本的线程机制 1.定义任务:Runnable接口 线程可以驱动任务,因此需要一种描述任务的方式,这可以由Runnable接口来提供.要想定义任务,只需实现Runnable接口并编写run方法,使得该任务可以执行你的命令.   class MyTask implements Runnable {    private String mName;     public MyTask(String name) {    mName = name;   }  

Java并发编程学习:线程安全与锁优化

本文参考<深入理解java虚拟机第二版> 一.什么是线程安全? 这里我借<Java Concurrency In Practice>里面的话:当多个线程访问一个对象,如果不考虑这些线程在运行时环境下的调度和交替执行,也不需要额外的同步,或者调用方进行任何其他的协调操作,调用这个对象的行为都可以获得正确的结果,那么这个对象是线程安全的. 我的理解:多线程访问一个对象,任何情况下,都能保持正确行为,就是对象就是安全的. 我们可以将Java语言中各种操作共享的数据分为以下5类:不可变.

java多线程总结一:线程的两种创建方式及优劣比较

1.通过实现Runnable接口线程创建 (1).定义一个类实现Runnable接口,重写接口中的run()方法.在run()方法中加入具体的任务代码或处理逻辑. (2).创建Runnable接口实现类的对象. (3).创建一个Thread类的对象,需要封装前面Runnable接口实现类的对象.(接口可以实现多继承) (4).调用Thread对象的start()方法,启动线程 示例代码: [java] view plaincopy <span style="font-size:16px;&

java多线程总结一:线程的两种创建方式及比较

1.线程的概念:线程(thread)是指一个任务从头至尾的执行流,线程提供一个运行任务的机制,对于java而言,一个程序中可以并发的执行多个线程,这些线程可以在多处理器系统上同时运行.当程序作为一个应用程序运行时,java解释器为main()方法启动一个线程. 2.并行与并发: (1)并发:在单处理器系统中,多个线程共享CPU时间,而操作系统负责调度及分配资源给它们. (2)并行:在多处理器系统中,多个处理器可以同时运行多个线程,这些线程在同一时间可以同时运行,而不同于并发,只能多个线程共享CP

Java并发编程学习笔记(一)——线程安全性

1.当多个线程访问某个状态变量并且其中有一个献策灰姑娘执行写入操作时,必须采用同步机制来协同这些线程对变量的访问.Java中的主要同步机制是关键字synchronized,他提供了一种独占的加锁方式. 2.在任何情况下,只有当类中仅包含自己的状态时,线程安全类才是有意义的. 3.当多个线程访问某个类时,不管运行时环境采用何种调度方式或者这些献策灰姑娘讲如何交替执行,并且在主调代码中不需要任何额外的同步或协同,这个类都能表现出正确的行为,那么就称这个类是线程安全的. 4.无状态对象一定是线程安全的

JAVA并发编程学习笔记------对象的可见性及发布逸出

一.非原子的64位操作: 当线程在没有同步的情况下读取变量时,可能会得到一个失效值,但至少这个值是由之前某个线程设置的值,而不是一个随机值,这种安全性保证被称为最低安全性.最低安全性适用于绝大多数变量,但存在一个例外:非volatile类型的64位数值变量(double,long),Java内存模型要求,变量的读取和写入操作都必须是原子操作,但对于非volatile型的long,double变量,JVM允许将64位的读操作或写操作分解为两个32位的操作,当读取一个非volatile类型的long

Java并发编程学习笔记(二)——对象的共享

主要概念:可见性.重排序.失效数据.最低安全性.发布.逸出.线程封闭(Ad-hoc.栈封闭.ThreadLocal类).不变性.Final域.事实不可变对象. 1.在没有同步的情况下,编译器.处理器以及运行时等都可能对操作的执行顺序进行一些意想不到的调整.在缺乏足够同步的多线程程序中,要想对内存操作的执行顺序进行判断,几乎无法得出正确的结论. 2.在多线程中使用共享且可变的long和double等类型的变量是不安全的,除非用关键字volatile来声明它们,或者用锁来保护他们. 3.加锁的含义不

java并发编程实战笔记-线程安全性

什么是线程安全性 线程安全性定义中最核心的概念就是:**正确性**.我们将单线程的正确性近似 定义为"所见即所知",当多个线程访问这个类的时候,始终能表现出正确的行为, 那么这个类就是线程安全类. 当多个线程访问某个类时,不管运行时环境采用什么调度方式或者这些线程将如何 交替运行,并且调用代码时,不需要额外的同步,就可以产生正确的结果.这个类 就是线程安全类. 在线程安全类上执行任何串行或者并行的操作都不会使对象处于无效状态. 可重入代码:就是这段代码,和其他代码不存在共享状态,只包含