多线程陷阱(不要调用run方法;静态的同步方法)
2011-10-23 21:18
323 查看
1. 从JDK1.5开始,Java提供了3种方式来创建,启动多线程: Ø 继承Thread类来创建线程类,重写run()方法作为线程执行体。 Ø 实现Runnable接口来创建线程类,重写run()方法作为线程执行体。 Ø 实现Callable接口来创建线程类,重写run()方法作为线程执行体。 其中第一种方式效果最差,它有2点坏处: l 线程类继承了Thread类,无法在继承其他父类。 l 因为每条线程都是一个Thread子类的实例,因此多个线程之间共享数据比较麻烦。 对于第二三种方式,它们的本质是一样的,只是Callable接口里包含的call()方法既可以声明抛出异常,也可以拥有返回值。 2.此外启动线程应该使用start()方法,而不是run()方法。如果程序从未调用线程对象的start()方法来启动它,那么这个线程对象将一直处于”新建”状态(1.新建 2.就绪 3.运行 4.阻塞 5.死亡总共5个状态),它永远也不会作为线程获得执行的机会,它只是一个普通的Java对象。当程序调用线程对象的run()方法时,与调用普通Java对象的普通方法并无任何区别,因此绝对不会启动一条新线程的。 3. 静态的同步方法: Java语言规定:任何线程进入同步方法,同步代码块之前,必须先获取同步方法,同步代码块对应的同步监视器。对于同步代码块而言,程序必须显示为它指定同步监视器;对于同步非静态方法而言,该方法的同步监视器是this-即调用该方法的Java对象;对于静态的同步方法而言,该方法的同步监视器不是this,而是该类本身。 如以下代码:class SynchronizedStatic implements Runnable { static boolean flag = true; public static synchronized void test0() {//同步监视器是该类本身 for (int i = 0; i < 1000; i++) { System.out.println("test0: " + Thread.currentThread().getName() + " " + i); } } public void test1() { synchronized (this) {//同步监视器是this,即调用该方法的Java对象。 for (int i = 0; i < 1000; i++) { System.out.println("test1: " + Thread.currentThread().getName() + " " + i); } } } public void run() { if (flag) { flag = false; test0(); } else { flag = true; test1(); } } public static void main(String args[]) throws InterruptedException { SynchronizedStatic ss = new SynchronizedStatic(); new Thread(ss).start(); Thread.sleep(1); new Thread(ss).start(); } }
运行结果:
test0: Thread-0 244
test1: Thread-1 7
test0: Thread-0 245
test1: Thread-1 8
test0: Thread-0 246
test1: Thread-1 9
test0: Thread-0 247
test1: Thread-1 10
test1: Thread-1 11
test1: Thread-1 12
test0: Thread-0 248
test1: Thread-1 13
test0: Thread-0 249
test0: Thread-0 250
从运行结果可以看出:静态同步方法可以和以this为同步监视器的同步代码块同时执行。因为两者的同步监视器不一样,前者是对SynchronizedStatic类的锁定,后者是对ss变量所引用的对象的锁定,因此程序可以在两个线程间相互切换。
若将test1()方法做以下更改:
[code]public void test1() {
synchronized (SynchronizedStatic.class) {//和静态方法具有了相同的同步监视器
for (int i = 0; i < 1000; i++) {
System.out.println("test1: " + Thread.currentThread().getName()
+ " " + i);
}
}
}
运行结果:
test0: Thread-0 995
test0: Thread-0 996
test0: Thread-0 997
test0: Thread-0 998
test0: Thread-0 999
test1: Thread-1 0
test1: Thread-1 1
test1: Thread-1 2
test1: Thread-1 3
test1: Thread-1 4
结果说明:静态同步方法和以当前类为同步监视器的同步代码块不能同时执行。
[/code]
相关文章推荐
- Java:多线程,线程同步,synchronized关键字的用法(同步代码块、非静态同步方法、静态同步方法)
- (Java多线程)分析静态方法所适用的同步监视器对象是什么?
- 同步调用异步方法how-would-i-run-an-async-taskt-method-synchronously
- JavaSE8基础 多线程synchronized 同步静态方法的锁
- 有空帮我看一下,多线程(问题:从不同步的代码块中调用了对象同步方法)
- 多线程:创建Thread为什么要调用start启动,而不能直接调用run方法
- java 多线程10:synchronized锁机制 之 锁定类静态方法 和锁定类.Class 和 数据String的常量池特性 同步静态方法
- 多线程中调用run()方法和start()方法的简单区别
- 多线程静态方法同步
- 多线程执行时为什么调用的是start()方法而不是run()方法
- Java多线程 6 静态同步方法的锁
- 多线程有几种实现方法?同步有几种实现方法?静态同步方法和非静态有何不同?—— Java经典面试题系列
- 从头认识多线程-2.16 同步静态方法和静态代码块
- 多线程为什么调用start而不是调用run方法
- 多线程之静态同步函数/方法
- 多线程静态方法同步
- paip.多线程调用静态方法错乱的解决
- 静态方法在多线程下的调用
- java多线程并发去调用一个类的静态方法,会有问题吗?
- java多线程之-----静态同步synchronized方法与synchronized(class) 代码块