多线程陷阱(不要调用run方法;静态的同步方法)
来源:互联网 发布:富士通打印软件安装 编辑:程序博客网 时间:2024/06/03 13:37
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()方法做以下更改:
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
结果说明:静态同步方法和以当前类为同步监视器的同步代码块不能同时执行。
- 多线程陷阱(不要调用run方法;静态的同步方法)
- 多线程静态方法同步
- 多线程静态方法同步
- 多线程调用静态方法
- 静态方法在多线程下的调用
- 多线程同步的方法
- 多线程同步的方法
- Java多线程 6 静态同步方法的锁
- 多线程之静态同步函数/方法
- 多线程中调用run()方法和start()方法的简单区别
- 多线程执行时为什么调用的是start()方法而不是run()方法
- 多线程为什么调用start而不是调用run方法
- 静态方法的调用
- paip.多线程调用静态方法错乱的解决
- 静态方法的同步测试
- 静态方法的同步代码
- 静态同步方法与非静态同步方法的区别
- JNI native多线程调用Java静态方法
- jsp输出中文页面乱码,中文提交后文本框里乱码,后台中文变乱码,全解决
- Android中ListView分页加载数据
- UML设计模式笔试题
- linux和 windows多线程开发应注意的区别
- Data-Intensive Text Processing with MapReduce 第三章(2)——PAIRS AND STRIPES
- 多线程陷阱(不要调用run方法;静态的同步方法)
- Linux远程控制之VNC (server ,viewer)安装教程 || chkconfig
- Exuberant Ctags
- 一道笔试题和UML思想
- Action与Command区别
- ARM-Linux移植之(四)——根文件系统构建
- java书籍
- sctp仿真的相关解读
- 读书笔记_windows的APIHook技术_part 2