JAVA基础Word文件下载.docx

上传人:b****1 文档编号:15395905 上传时间:2022-10-30 格式:DOCX 页数:14 大小:40.78KB
下载 相关 举报
JAVA基础Word文件下载.docx_第1页
第1页 / 共14页
JAVA基础Word文件下载.docx_第2页
第2页 / 共14页
JAVA基础Word文件下载.docx_第3页
第3页 / 共14页
JAVA基础Word文件下载.docx_第4页
第4页 / 共14页
JAVA基础Word文件下载.docx_第5页
第5页 / 共14页
点击查看更多>>
下载资源
资源描述

JAVA基础Word文件下载.docx

《JAVA基础Word文件下载.docx》由会员分享,可在线阅读,更多相关《JAVA基础Word文件下载.docx(14页珍藏版)》请在冰豆网上搜索。

JAVA基础Word文件下载.docx

每条线程都有自己的工作内存(Working 

Memory),工作内存中保存的是主存中某些变量的拷贝,线程对所有变量的操作都是在工作内存中进行,线程之间无法相互直接访问,变量传递均需要通过主存完成。

多个线程的执行是并发的,也就是在逻辑上“同时”,而不管是否是物理上的“同时”。

如果系统只有一个CPU,那么真正的“同时”是不可能的。

多线程和传统的单线程在程序设计上最大的区别在于,由于各个线程的控制流彼此独立,使得各个线程之间的代码是乱序执行的,将会带来线程调度,同步等问题。

二:

在Java中实现多线程

  我们不妨设想,为了创建一个新的线程,我们需要做些什么?

很显然,我们必须指明这个线程所要执行的代码,而这就是在Java中实现多线程我们所需要做的一切!

  作为一个完全面向对象的语言,Java提供了类java.lang.Thread来方便多线程编程,这个类提供了大量的方法来方便我们控制自己的各个线程。

  那么如何提供给Java我们要线程执行的代码呢?

让我们来看一看Thread类。

Thread类最重要的方法是run(),它为Thread类的方法start()所调用,提供我们的线程所要执行的代码。

为了指定我们自己的代码,只需要覆盖它!

方法一:

继承Thread类,重写方法run(),我们在创建的Thread类的子类中重写run(),加入线程所要执行的代码即可。

下面是一个例子:

publicclassTwoThreadextendsThread{

publicvoidrun(){

for(inti=0;

i<

10;

i++){

System.out.println("

Newthread"

);

}

publicstaticvoidmain(String[]args){

TwoThreadtt=newTwoThread();

tt.start();

Mainthread"

}

  这种方法简单明了,符合大家的习惯,但是,它也有一个很大的缺点,那就是如果我们的类已经从一个类继承,则无法再继承Thread类。

方法二:

实现Runnable接口

Runnable接口只有一个方法run(),我们声明自己的类实现Runnable接口并提供这一方法,将我们的线程代码写入其中,就完成了这一部分的任务。

但是Runnable接口并没有任何对线程的支持,我们还必须创建Thread类的实例,这一点通过Thread类的构造函数publicThread(Runnabletarget);

来实现。

publicclassMyThreadimplementsRunnable{

intcount=1,number;

publicMyThread(intnum){

number=num;

System.out.println("

创建线程"

+number);

publicvoidrun(){

while(true){

线程"

+number+"

:

计数"

+count);

if(++count==6)return;

publicstaticvoidmain(Stringargs[]){

for(inti=0;

5;

i++)

newThread(newMyThread(i+1)).start();

使用Runnable接口来实现多线程使得我们能够在一个类中包容所有的代码,有利于封装下面让我们一起来研究一下多线程使用中的一些问题。

三:

线程的四种状态

1. 

新状态:

线程已被创建但尚未执行(start()尚未被调用)。

2. 

可执行状态:

线程可以执行,虽然不一定正在执行。

CPU时间随时可能被分配给该线程,从而使得它执行。

3. 

阻塞状态:

线程不会被分配CPU时间,无法执行;

可能阻塞于I/O,或者阻塞于同步锁。

4. 

死亡状态:

正常情况下run()返回使得线程死亡。

调用stop()或destroy()亦有同样效果,但是不被推荐,前者会产生异常,后者是强制终止,不会释放锁。

四:

线程的优先级

  线程的优先级代表该线程的重要程度,当有多个线程同时处于可执行状态并等待获得CPU时间时,线程调度系统根据各个线程的优先级来决定给谁分配CPU时间,优先级高的线程有更大的机会获得CPU时间,优先级低的线程也不是没有机会,只是机会要小一些罢了。

你可以调用Thread类的方法getPriority()和setPriority()来存取线程的优先级,线程的优先级界于1(MIN_PRIORITY)和10(MAX_PRIORITY)之间,缺省是5(NORM_PRIORITY)。

状态图1

状态图2

下一页>

>

发表评论 

字体[ 

大中小 

打印 

进入博客 

进入论坛 

推荐给朋友 

]

【JAVA基础】JAVA多线程编程详解

(2)

五:

线程的同步

  由于同一进程的多个线程共享同一片存储空间,在带来方便的同时,也带来了访问冲突这个严重的问题。

Java语言提供了专门机制以解决这种冲突,有效避免了同一个数据对象被多个线程同时访问。

我们只需针对方法提出一套机制,这套机制就是 

synchronized关键字,它包括两种用法:

synchronized方法和synchronized块。

1.synchronized方法:

通过在方法声明中加入synchronized关键字来声明synchronized方法。

synchronized方法控制对类成员变量的访问:

每个类实例对应一把锁,每个synchronized方法都必须获得调用该方法的类实例的锁方能执行,否则所属线程阻塞,方法一旦执行,就独占该锁,直到从该方法返回时才将锁释放,此后被阻塞的线程方能获得该锁,重新进入可执行状态。

这种机制确保了同一时刻对于每一个类实例,其所有声明为synchronized的成员函数中至多只有一个处于可执行状态(因为至多只有一个能够获得该类实例对应的锁),从而有效避免了类成员变量的访问冲突(只要所有可能访问类成员变量的方法均被声明为synchronized)。

在 

Java中,不光是类实例,每一个类也对应一把锁,这样我们也可将类的静态成员函数声明为synchronized,以控制其对类的静态成员变量的访问。

synchronized方法的缺陷:

若将一个大的方法声明为synchronized将会大大影响效率,典型地,若将线程类的方法run()声明为synchronized,由于在线程的整个生命期内它一直在运行,因此将导致它对本类任何synchronized方法的调用都永远不会成功。

2.synchronized块:

通过synchronized关键字来声明synchronized块。

语法如下:

synchronized(syncObject){

//允许访问控制的代码

synchronized块是这样一个代码块,其中的代码必须获得对象syncObject的锁方能执行,具体机制同前所述。

由于可以针对任意代码块,且可任意指定上锁的对象,故灵活性较高。

六:

线程的阻塞

  为了解决对共享存储区的访问冲突,Java引入了同步机制,现在让我们来考察多个线程对共享资源的访问,显然同步机制已经不够了,因为在任意时刻所要求的资源不一定已经准备好了被访问,反过来,同一时刻准备好了的资源也可能不止一个。

为了解决这种情况下的访问控制问题,Java引入了对阻塞机制的支持。

  阻塞指的是暂停一个线程的执行以等待某个条件发生(如某资源就绪)。

Java提供了大量方法来支持阻塞,下面让对它们逐一分析。

1.sleep()方法:

sleep()允许指定以毫秒为单位的一段时间作为参数,它使得线程在指定的时间内进入阻塞状态,不能得到CPU时间,指定的时间一过,线程重新进入可执行状态。

典型地,sleep()被用在等待某个资源就绪的情形:

测试发现条件不满足后,让线程阻塞一段时间后重新测试,直到条件满足为止。

2.suspend()和resume()方法:

两个方法配套使用,suspend()使得线程进入阻塞状态,并且不会自动恢复,必须其对应的resume()被调用,才能使得线程重新进入可执行状态。

典型地,suspend()和resume()被用在等待另一个线程产生的结果的情形:

测试发现结果还没有产生后,让线程阻塞,另一个线程产生了结果后,调用resume()使其恢复。

3.yield()方法:

yield()使得线程放弃当前分得的CPU时间,但是不使线程阻塞,即线程仍处于可执行状态,随时可能再次分得CPU时间。

调用yield()的效果等价于调度程序认为该线程已执行了足够的时间从而转到另一个线程。

4.wait()和notify()方法:

两个方法配套使用,wait()使得线程进入阻塞状态,它有两种形式,一种允许指定以毫秒为单位的一段时间作为参数,另一种没有参数,前者当对应的notify()被调用或者超出指定时间时线程重新进入可执行状态,后者则必须对应的notify()被调用。

  2和4区别的核心在于,前面叙述的所有方法,阻塞时都不会释放占用的锁(如果占用了的话),而这一对方法则相反。

上述的核心区别导致了一系列的细节上的区别。

  首先,前面叙述的所有方法都隶属于Thread类,但是这一对却直接隶属于Object类,也就是说,所有对象都拥有这一对方法。

因为这一对方法阻塞时要释放占用的锁,而锁是任何对象都具有的,调用任意对象的wait()方法导致线程阻塞,并且该对象上的锁被释放。

而调用任意对象的notify()方法则导致因调用该对象的wait()方法而阻塞的线程中随机选择的一个解除阻塞(但要等到获得锁后才真正可执行)。

  其次,前面叙述的所有方法都可在任何位置调用,但是这一对方法却必须在 

synchronized方法或块中调用,理由也很简单,只有在synchronized方法或块中当前线程才占有锁,才有锁可以

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 教学研究 > 教学反思汇报

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1