`
newleague
  • 浏览: 1492486 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类

Java多线程编程

阅读更多

编写具有多线程能力的程序经常会用到的方法有:
run(),start(),wait(),notify(),notifyAll(),sleep(),yield(),join()
还有一个重要的关键字:synchronized
本文将对以上内容进行讲解。


一:run()和start()
示例1:
public class ThreadTest extends Thread
{
public void run()
{
for(int i=0;i<10;i++)
{
System.out.print(" " + i);
}
}
public static void main(String[] args)
{
new ThreadTest().start();
new ThreadTest().start();
}
}
这是个简单的多线程程序。run()和start()是大家都很熟悉的两个方法。把希望并行处理的
代码都放在run()中;stat()用于自动调用run(),这是JAVA 的内在机制规定的。并且run()
的访问控制符必须是public,返回值必须是void(这种说法不准确,run()没有返回值),run()
不带参数。
这些规定想必大家都早已知道了,但你是否清楚为什么run 方法必须声明成这样的形式?这涉
及到JAVA 的方法覆盖和重载的规定。这些内容很重要,请读者参考相关资料。


二:关键字synchronized
有了synchronized 关键字,多线程程序的运行结果将变得可以控制。synchronized 关键
字用于保护共享数据。请大家注意"共享数据",你一定要分清哪些数据是共享数据,JAVA 是面向
对象的程序设计语言,所以初学者在编写多线程程序时,容易分不清哪些数据是共享数据。请看下
面的例子:
示例2:
public class ThreadTest implements Runnable
{ public synchronized void run()
{
for(int i=0;i<10;i++)
{
System.out.print(" " + i);
}
}
public static void main(String[] args)
{
Runnable r1 = new ThreadTest();
Runnable r2 = new ThreadTest();
Thread t1 = new Thread(r1);
Thread t2 = new Thread(r2);
t1.start();
t2.start();
}
}
在这个程序中,run()被加上了synchronized 关键字。在main 方法中创建了两个线程。你
可能会认为此程序的运行结果一定为:0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9。但你
错了!这个程序中synchronized 关键字保护的不是共享数据(其实在这个程序中synchronized
关键字没有起到任何作用,此程序的运行结果是不可预先确定的)。这个程序中的t1,t2 是两个   对象(r1,r2)的线程。JAVA
是面向对象的程序设计语言,不同的对象的数据是不同的,r1,r2 有
各自的run()方法,而synchronized 使同一个对象的多个线程,在某个时刻只有其中的一个线程
可以访问这个对象的synchronized 数据。
每个对象都有一个"锁标志",当这个对象的一个线程
访问这个对象的某个synchronized 数据时,这个对象的所有被synchronized 修饰的数据将被
上锁(因为"锁标志"被当前线程拿走了),只有当前线程访问完它要访问的synchronized 数据
时,当前线程才会释放"锁标志",这样同一个对象的其它线程才有机会访问synchronized 数据。
示例3:
public class ThreadTest implements Runnable
{
public synchronized void run()
{
for(int i=0;i<10;i++)
{
System.out.print(" " + i);
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r);
Thread t2 = new Thread(r);
t1.start(); t2.start();
}
}
如果你运行1000 次这个程序,它的输出结果也一定每次都是:0 1 2 3 4 5 6 7 8 9 0 1
2 3 4 5 6 7 8 9。因为这里的synchronized 保护的是共享数据。t1,t2 是同一个对象(r)
的两个线程,当其中的一个线程(例如:t1)开始执行run()方法时,由于run()受synchronized
保护,所以同一个对象的其他线程(t2)无法访问synchronized 方法(run 方法)。只有当t1
执行完后t2 才有机会执行。
示例4:
public class ThreadTest implements Runnable
{
public void run()
{
synchronized(this)
{
for(int i=0;i<10;i++)
{
System.out.print(" " + i);
}
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r);
Thread t2 = new Thread(r);
t1.start();
t2.start();
}
}
这个程序与示例3 的运行结果一样。在可能的情况下,应该把保护范围缩到最小,可以用示例
4 的形式,this 代表"这个对象"。没有必要把整个run()保护起来,run()中的代码只有一个for 循环,所以只要保护for 循环就可以了。
示例5:
public class ThreadTest implements Runnable
{
public void run()
{
for(int k=0;k<5;k++)
{
System.out.println(Thread.currentThread().getName()
+ " : for loop : " + k); }
synchronized(this)
{
for(int k=0;k<5;k++)
{
System.out.println(Thread.currentThread().getName()
+ " : synchronized for loop : " + k);
}
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r,"t1_name");
Thread t2 = new Thread(r,"t2_name");
t1.start();
t2.start();
}
}
运行结果:
t1_name : for loop : 0
t1_name : for loop : 1
t1_name : for loop : 2
t2_name : for loop : 0
t1_name : for loop : 3
t2_name : for loop : 1
t1_name : for loop : 4
t2_name : for loop : 2
t1_name : synchronized for loop : 0
t2_name : for loop : 3
t1_name : synchronized for loop : 1
t2_name : for loop : 4
t1_name : synchronized for loop : 2
t1_name : synchronized for loop : 3
t1_name : synchronized for loop : 4
t2_name : synchronized for loop : 0
t2_name : synchronized for loop : 1
t2_name : synchronized for loop : 2
t2_name : synchronized for loop : 3
t2_name : synchronized for loop : 4
第一个for 循环没有受synchronized 保护。对于第一个for 循环,t1,t2 可以同时访问。
运行结果表明t1 执行到了k=2 时,t2 开始执行了。t1 首先执行完了第一个for 循环,此时t2
还没有执行完第一个for 循环(t2 刚执行到k=2)。t1 开始执行第二个for 循环,当t1 的第二
个for 循环执行到k=1 时,t2 的第一个for 循环执行完了。t2 想开始执行第二个for 循环,但
由于t1 首先执行了第二个for 循环,这个对象的锁标志自然在t1 手中(synchronized 方法的
执行权也就落到了t1 手中),在t1 没执行完第二个for 循环的时候,它是不会释放锁标志的。所
以t2 必须等到t1 执行完第二个for 循环后,它才可以执行第二个for 循环。


三:sleep()
示例6:
public class ThreadTest implements Runnable
{
public void run()
{
for(int k=0;k<5;k++)
{
if(k == 2)
{
try
{
Thread.currentThread().sleep(5000);
}
catch(Exception e)
{}
}
System.out.print(" " + k);
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t = new Thread(r);
t.start();
}
}
sleep 方法会使当前的线程暂停执行一定时间(给其它线程运行机会)。读者可以运行示例6,
看看结果就明白了。sleep 方法会抛出异常,必须提供捕获代码。
示例7:
public class ThreadTest implements Runnable
{
public void run()
{ for(int k=0;k<5;k++)
{
if(k == 2)
{
try
{
Thread.currentThread().sleep(5000);
}
catch(Exception e)
{}
}
System.out.println(Thread.currentThread().getName()
+ " : " + k);
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r,"t1_name");
Thread t2 = new Thread(r,"t2_name");
t1.setPriority(Thread.MAX_PRIORITY);
t2.setPriority(Thread.MIN_PRIORITY);
t1.start();
t2.start();
}
}
t1 被设置了最高的优先级,t2 被设置了最低的优先级。t1 不执行完,t2 就没有机会执行。
但由于t1 在执行的中途休息了5 秒中,这使得t2 就有机会执行了。读者可以运行这个程序试试看。
示例8:
public class ThreadTest implements Runnable
{
public synchronized void run()
{
for(int k=0;k<5;k++)
{
if(k == 2)
{
try
{
Thread.currentThread().sleep(5000);
}
catch(Exception e)
{} }
System.out.println(Thread.currentThread().getName()
+ " : " + k);
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r,"t1_name");
Thread t2 = new Thread(r,"t2_name");
t1.start();
t2.start();
}
}
请读者首先运行示例8 程序,从运行结果上看:一个线程在sleep 的时候,并不会释放这个对
象的锁标志。


四:join()
示例9:
public class ThreadTest implements Runnable
{
public static int a = 0;
public void run()
{
for(int k=0;k<5;k++)
{
a = a + 1;
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t = new Thread(r);
t.start();
System.out.println(a);
}
}
请问程序的输出结果是5 吗?答案是:有可能。其实你很难遇到输出5 的时候,通常情况下都
不是5。这里不讲解为什么输出结果不是5,我要讲的是:怎样才能让输出结果为5!其实很简单,
join()方法提供了这种功能。join()方法,它能够使调用该方法的线程在此之前执行完毕。
把示例9 的main()方法该成如下这样: public static void main(String[] args) throws Exception
{
Runnable r = new ThreadTest();
Thread t = new Thread(r);
t.start();
t.join();
System.out.println(a);
}
这时,输出结果肯定是5!join()方法会抛出异常,应该提供捕获代码。或留给JDK 捕获。
示例10:
public class ThreadTest implements Runnable
{
public void run()
{
for(int k=0;k<10;k++)
{
System.out.print(" " + k);
}
}
public static void main(String[] args) throws Exception
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r);
Thread t2 = new Thread(r);
t1.start();
t1.join();
t2.start();
}
}
运行这个程序,看看结果是否与示例3 一样?


五:yield()
yield()方法与sleep()方法相似,只是它不能由用户指定线程暂停多长时间。按照SUN 的
说法:sleep
方法可以使低优先级的线程得到执行的机会,当然也可以让同优先级和高优先级的线程有执行的机会。而yield()方法只能使同优先级的线程有执行的机会。
示例11:
public class ThreadTest implements Runnable
{
public void run()
{ for(int k=0;k<10;k++)
{
if(k == 5 && Thread.currentThread().getName().equals("t1"))
{
Thread.yield();
}
System.out.println(Thread.currentThread().getName()
+ " : " + k);
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r,"t1");
Thread t2 = new Thread(r,"t2");
t1.setPriority(Thread.MAX_PRIORITY);
t2.setPriority(Thread.MIN_PRIORITY);
t1.start();
t2.start();
}
}
输出结果:
t1 : 0
t1 : 1
t1 : 2
t1 : 3
t1 : 4
t1 : 5
t1 : 6
t1 : 7
t1 : 8
t1 : 9
t2 : 0
t2 : 1
t2 : 2
t2 : 3
t2 : 4
t2 : 5
t2 : 6
t2 : 7
t2 : 8
t2 : 9
多次运行这个程序,输出也是一样。这说明:yield()方法不会使不同优先级的线程有执行的
机会。


六:wait(),notify(),notifyAll()
首先说明:wait(),notify(),notifyAll()这些方法由java.lang.Object 类提供,而上面
讲到的方法都是由java.lang.Thread 类提供(Thread 类实现了Runnable 接口)。
wait(),notify(),notifyAll()这三个方法用于协调多个线程对共享数据的存取,所以必
须在synchronized 语句块内使用这三个方法。先看下面了例子:
示例12:
public class ThreadTest implements Runnable
{
public static int shareVar = 0;
public synchronized void run()
{
if(shareVar == 0)
{
for(int i=0;i<10;i++)
{
shareVar++ ;
if(shareVar == 5)
{
try
{
this.wait();
}
catch(Exception e)
{}
}
}
}
if(shareVar != 0)
{
System.out.print(Thread.currentThread().getName());
System.out.println(" shareVar = " + shareVar);
this.notify();
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Thread t1 = new Thread(r,"t1"); Thread t2 = new Thread(r,"t2");
t1.start();
t2.start();
}
}
运行结果:
t2 shareVar = 5
t1 shareVar = 10
t1线程最先执行。由于初始状态下shareVar 为0,t1 将使shareVar 连续加1,当shareVar
的值为5 时,t1 调用wait()方法,t1 将处于休息状态,同时释放锁标志。这时t2 得到了锁标志
开始执行,shareVar 的值已经变为5,所以t2 直接输出shareVar 的值,然后再调用notify()
方法唤醒t1。t1 接着上次休息前的进度继续执行,把shareVar 的值一直加到10,由于此刻
shareVar 的值不为0,所以t1 将输出此刻shareVar 的值,然后再调用notify()方法,由于
此刻已经没有等待锁标志的线程,所以此调用语句不起任何作用。
这个程序简单的示范了wait(),notify()的用法,读者还需要在实践中继续摸索。


七:关于线程的补充
编写一个具有多线程能力的程序可以继承Thread 类,也可以实现Runnable 接口。在这两个
方法中如何选择呢?从面向对象的角度考虑,作者建议你实现Runnable 接口。有时你也必须实现
Runnable 接口,例如当你编写具有多线程能力的小应用程序的时候。
线程的调度:
一个Thread 对象在它的生命周期中会处于各种不同的状态,上图形象地说明了这点。wa in
调用start()方法使线程处于可运行状态,这意味着它可以由JVM 调度并执行。这并不意味
着线程就会立即运行。
实际上,程序中的多个线程并不是同时执行的。除非线程正在真正的多CPU 计算机系统上执行,
否则线程使用单CPU 必须轮流执行。但是,由于这发生的很快,我们常常认为这些线程是同时执行
的。
JAVA 运行时系统的计划调度程序是抢占性的。如果计划调度程序正在运行一个线程并且来了另
一个优先级更高的线程,那么当前正在执行的线程就被暂时终止而让更高优先级的线程执行。
JAVA 计划调度程序不会为与当前线程具有同样优先级的另一个线程去抢占当前的线程。但是,
尽管计划调度程序本身没有时间片(即它没有给相同优先级的线程以执行用的时间片),但以
Thread 类为基础的线程的系统实现可能会支持时间片分配。这依赖具体的*作系统,Windows 与
UNIX 在这个问题上的支持不会完全一样。
由于你不能肯定小应用程序将运行在什么*作系统上,因此你不应该编写出依赖时间片分配的
程序。就是说,应该使用yield 方法以允许相同优先级的线程有机会执行而不是希望每一个线程都自动得到一段CPU 时间片。
Thread 类提供给你与系统无关的处理线程的机制。但是,线程的实际实现取决于JAVA 运行
所在的*作系统。因此,线程化的程序确实是利用了支持线程的*作系统。
当创建线程时,可以赋予它优先级。它的优先级越高,它就越能影响运行系统。JAVA 运行系统
使用一个负责在所有执行JAVA 程序内运行所有存在的计划调度程序。该计划调度程序实际上使用
一个固定优先级的算法来保证每个程序中的最高优先级的线程得到CPU――允许最高优先级的线程
在其它线程之前执行。
对于在一个程序中有几个相同优先级的线程等待执行的情况,该计划调度程序循环地选择它们,
当进行下一次选择时选择前面没有执行的线程,具有相同优先级的所有的线程都受到平等的对待。
较低优先级的线程在较高优先级的线程已经死亡或者进入不可执行状态之后才能执行。
继续讨论wait(),notify(),notifyAll():
当线程执行了对一个特定对象的wait()调用时,那个线程被放到与那个对象相关的等待池中。
此外,调用wait()的线程自动释放对象的锁标志。
可以调用不同的wait():wait() 或wait(long timeout)
对一个特定对象执行notify()调用时,将从对象的等待池中移走一个任意的线程,并放到锁
标志等待池中,那里的线程一直在等待,直到可以获得对象的锁标志。notifyAll()方法将从对象
等待池中移走所有等待那个对象的线程并放到锁标志等待池中。只有锁标志等待池中的线程能获取
对象的锁标志,锁标志允许线程从上次因调用wait()而中断的地方开始继续运行。
在许多实现了wait()/notify()机制的系统中,醒来的线程必定是那个等待时间最长的线程。
然而,在Java 技术中,并不保证这点。
注意,不管是否有线程在等待,都可以调用notify()。如果对一个对象调用notify()方法,
而在这个对象的锁标志等待池中并没有线程,那么notify()调用将不起任何作用。
在JAVA 中,多线程是一个神奇的主题。之所以说它"神奇",是因为多线程程序的运行结果不
可预测,但我们又可以通过某些方法控制多线程程序的执行。要想灵活使用多线程,读者还需要大
量实践。
另外,从JDK 1.2 开始,SUN 就不建议使用resume(),stop(),suspend()了。

分享到:
评论

相关推荐

    JAVA多线程编程技术PDF

    这份“JAVA多线程编程技术PDF”是学习和掌握这一领域的经典资料,涵盖了多线程的全部知识点。 首先,多线程的核心概念包括线程的创建与启动。在Java中,可以通过实现Runnable接口或继承Thread类来创建线程。创建后...

    汪文君JAVA多线程编程实战(完整不加密)

    《汪文君JAVA多线程编程实战》是一本专注于Java多线程编程的实战教程,由知名讲师汪文君倾力打造。这本书旨在帮助Java开发者深入理解和熟练掌握多线程编程技术,提升软件开发的效率和质量。在Java平台中,多线程是...

    java多线程编程

    Java多线程编程是Java开发中的重要组成部分,它允许程序同时执行多个任务,极大地提高了程序的效率和响应性。在Java中,多线程主要通过继承Thread类或实现Runnable接口来实现。本教程将深入探讨Java多线程的各个方面...

    Java多线程编程实战指南(核心篇)

    Java多线程编程实战指南(核心篇) 高清pdf带目录 随着现代处理器的生产工艺从提升处理器主频频率转向多核化,即在一块芯片上集成多个处理器内核(Core),多核处理器(Multicore Processor)离我们越来越近了――如今...

    java 多线程编程实战指南(核心 + 设计模式 完整版)

    《Java多线程编程实战指南》这本书深入浅出地讲解了Java多线程的核心概念和实战技巧,分为核心篇和设计模式篇,旨在帮助开发者掌握并应用多线程技术。 1. **线程基础** - **线程的创建**:Java提供了两种创建线程...

    Java多线程编程实战指南-核心篇

    《Java多线程编程实战指南-核心篇》是一本深入探讨Java并发编程的书籍,旨在帮助读者掌握在Java环境中创建、管理和同步线程的核心技术。Java的多线程能力是其强大之处,使得开发者能够在同一时间执行多个任务,提高...

    Java多线程编程核心技术_完整版_java_

    Java多线程编程是Java开发中的重要组成部分,它允许程序同时执行多个任务,极大地提高了程序的效率和响应性。在Java中,多线程主要通过继承Thread类或实现Runnable接口来实现。本教程《Java多线程编程核心技术》将...

    Java多线程编程技术

    《Java多线程编程核心技术》建议猿友们读两遍,因为其写得没有那么抽象,第一遍有些概念不是很理解,可以先跳过并记录起来,第一遍阅读的目的主要是了解整个架构。第二遍再慢慢品味,并贯穿全部是指点来思考,并将...

    java 多线程编程指南

    这份“Java多线程编程指南”深入探讨了这一主题,为中级到高级的Java开发者提供了宝贵的资源。 首先,多线程的基础概念是理解整个主题的关键。线程是程序执行的最小单元,每个线程都有自己的程序计数器、虚拟机栈、...

    深入学习:Java多线程编程

    《深入学习:Java多线程编程》是一本专注于Java并发技术的专业书籍,旨在帮助开发者深入理解和熟练运用Java中的多线程编程。Java多线程是Java编程中的核心部分,尤其在现代高性能应用和分布式系统中不可或缺。理解并...

    Java多线程编程实例

    本书“Java多线程编程实例”深入浅出地讲解了如何在Java环境中实现多线程操作,尽管出版时间较早,但其内容的经典性和实用性使其在现代开发中仍具有极高的参考价值。 首先,我们要理解Java中的线程是如何创建的。...

    Java多线程编程核心技术.zip

    Java多线程编程是Java开发中的重要组成部分,它允许程序同时执行多个任务,提升系统效率。在Java中,实现多线程主要有两种方式:继承Thread类和实现Runnable接口。本资料"Java多线程编程核心技术.zip"深入探讨了这些...

    《Java多线程编程实例》随书源码

    《Java多线程编程实例》这本书深入浅出地探讨了Java中的多线程编程,通过丰富的实例帮助读者理解和掌握这一复杂主题。随书源码提供了实际操作的机会,以便读者能够亲手实践书中的示例,加深理解。 1. **线程创建...

    java多线程编程实例

    从给定的文件信息中,我们可以提取出关于Java多线程编程的重要知识点,涉及线程创建、线程生命周期以及线程间的同步与通信等核心概念。 ### Java多线程编程实例解析 #### 1. 创建线程的方式 在Java中,创建线程有...

    java多线程编程实例 (源程序)

    Java多线程编程是Java开发中的重要组成部分,它允许程序同时执行多个任务,极大地提高了程序的效率和响应性。在Java中,多线程可以通过实现Runnable接口或继承Thread类来创建。下面我们将深入探讨Java多线程编程的...

    java多线程编程-详细炒作例子

    ### Java多线程编程详解与实战案例 #### 理解多线程概念与Java内存模型 多线程,作为现代编程中的一项关键技术,允许在单一应用程序中并发执行多个指令流,每个这样的指令流被称为一个线程。在Java中,线程被视为...

Global site tag (gtag.js) - Google Analytics