`
insertyou
  • 浏览: 904906 次
  • 性别: Icon_minigender_1
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

Java编程那些事儿98——多线程问题及处理1

阅读更多

Java编程那些事儿98——多线程问题及处理1

陈跃峰

出自:http://blog.csdn.net/mailbomb

12.4 多线程问题及处理

多线程编程为程序开发带来了很多的方便,但是也带来了一些问题,这些问题是在程序开发过程中必须进行处理的问题。

这些问题的核心是,如果多个线程同时访问一个资源,例如变量、文件等,时如何保证访问安全的问题。在多线程编程中,这种会被多个线程同时访问的资源叫做临界资源。

下面通过一个简单的示例,演示多个线程访问临界资源时产生的问题。在该示例中,启动了两个线程类DataThread的对象,该线程每隔200毫秒输出一次变量n的值,并将n的值减少1。变量n的值存储在模拟临界资源的Data类中,该示例的核心是两个线程类都使用同一个Data类的对象,这样Data类的这个对象就是一个临界资源了。示例代码如下:

package syn1;

/**

* 模拟临界资源的类

*/

public class Data {

public int n;

public Data(){

n = 60;

}

}

package syn1;

/**

* 测试多线程访问时的问题

*/

public class TestMulThread1 {

public static void main(String[] args) {

Data data = new Data();

DataThread d1 = new DataThread(data,"线程1");

DataThread d2 = new DataThread(data,"线程2");

}

}

package syn1;

/**

* 访问数据的线程

*/

public class DataThread extends Thread {

Data data;

String name;

public DataThread(Data data,String name){

this.data = data;

this.name = name;

start();

}

public void run(){

try{

for(int i = 0;i < 10;i++){

System.out.println(name + ":" + data.n);

data.n--;

Thread.sleep(200);

}

}catch(Exception e){}

}

}

在运行时,因为不同情况下该程序的运行结果会出现不同,该程序的一种执行结果为:

线程1:60

线程2:60

线程2:58

线程1:58

线程2:56

线程1:56

线程2:54

线程1:54

线程2:52

线程1:52

线程2:50

线程1:50

线程2:48

线程1:48

线程2:47

线程1:46

线程2:44

线程1:44

线程2:42

线程1:42

从执行结果来看,第一次都输出60是可以理解的,因为线程在执行时首先输出变量的值,这个时候变量n的值还是初始值60,而后续的输出就比较麻烦了,在开始的时候两个变量保持一致的输出,而不是依次输出n的每个值的内容,而到将要结束时,线程2输出47这个中间数值。

出现这种结果的原因很简单:线程1改变了变量n的值以后,还没有来得及输出,这个变量n的值就被线程2给改变了,所以在输出时看的输出都是跳跃的,偶尔出现了连续。

出现这个问题也比较容易接受,因为最基本的多线程程序,系统只保证线程同时执行,至于哪个先执行,哪个后执行,或者执行中会出现一个线程执行到一半,就把CPU的执行权交给了另外一个线程,这样线程的执行顺序是随机的,不受控制的。所以会出现上面的结果。

这种结果在很多实际应用中是不能被接受的,例如银行的应用,两个人同时取一个账户的存款,一个使用存折、一个使用卡,这样访问账户的金额就会出现问题。或者是售票系统中,如果也这样就出现有人买到相同座位的票,而有些座位的票却未售出。

在多线程编程中,这个是一个典型的临界资源问题,解决这个问题最基本,最简单的思路就是使用同步关键字synchronized

synchronized关键字是一个修饰符,可以修饰方法或代码块,其的作用就是,对于同一个对象(不是一个类的不同对象),当多个线程都同时调用该方法或代码块时,必须依次执行,也就是说,如果两个或两个以上的线程同时执行该段代码时,如果一个线程已经开始执行该段代码,则另外一个线程必须等待这个线程执行完这段代码才能开始执行。就和在银行的柜台办理业务一样,营业员就是这个对象,每个顾客就好比线程,当一个顾客开始办理时,其它顾客都必须等待,及时这个正在办理的顾客在办理过程中接了一个电话 (类比于这个线程释放了占用CPU的时间,而处于阻塞状态),其它线程也只能等待。

使用synchronized关键字修改以后的上面的代码为:

package syn2;

/**

* 模拟临界资源的类

*/

public class Data2 {

public int n;

public Data2(){

n = 60;

}

public synchronized void action(String name){

System.out.println(name + ":" + n);

n--;

}

}

package syn2;

/**

* 测试多线程访问时的问题

*/

public class TestMulThread2 {

public static void main(String[] args) {

Data2 data = new Data2();

Data2Thread d1 = new Data2Thread(data,"线程1");

Data2Thread d2 = new Data2Thread(data,"线程2");

}

}

package syn2;

/**

* 访问数据的线程

*/

public class Data2Thread extends Thread {

Data2 data;

String name;

public Data2Thread(Data2 data,String name){

this.data = data;

this.name = name;

start();

}

public void run(){

try{

for(int i = 0;i < 10;i++){

data.action(name);

Thread.sleep(200);

}

}catch(Exception e){}

}

}

该示例代码的执行结果会出现不同,一种执行结果为:

线程1:60

线程2:59

线程2:58

线程1:57

线程2:56

线程1:55

线程2:54

线程1:53

线程2:52

线程1:51

线程2:50

线程1:49

线程1:48

线程2:47

线程2:46

线程1:45

线程2:44

线程1:43

线程2:42

线程1:41

在该示例中,将打印变量n的代码和变量n变化的代码组成一个专门的方法action,并且使用修饰符synchronized修改该方法,也就是说对于一个Data2的对象,无论多少个线程同时调用action方法时,只有一个线程完全执行完该方法以后,别的线程才能够执行该方法。这就相当于一个线程执行到该对象的synchronized方法时,就为这个对象加上了一把锁,锁住了这个对象,别的线程在调用该方法时,发现了这把锁以后就继续等待下去了。

分享到:
评论

相关推荐

    Java练手小项目——多线程聊天室.zip

    【Java练手小项目——多线程...通过这个“Java练手小项目——多线程聊天室”,你可以深入了解Java多线程的原理和实践,同时学习到网络编程和并发集合的运用,这对于提升你的Java编程技能和解决实际问题的能力大有裨益。

    java高级编程——多线程,输入输出流,网络编程

    java高级编程——对多线程,输入输出流,网络编程这三大块的深入讲解。

    Java编程那些事儿

    "Java编程那些事儿"无疑是对这个强大语言的深入探讨,旨在帮助开发人员提升技能,拓宽视野。这份资料可能是由一系列章节或主题组成的文档,比如基础语法、面向对象编程、异常处理、集合框架、多线程、IO流、网络编程...

    java编程——Java线程

    Java线程是Java编程语言中用于实现并发处理的关键机制之一。线程是程序执行流的最小单位,它允许一个程序在同一时间执行多个任务,从而提高了程序的效率和资源利用率。在Java中,线程的创建和管理被内置到语言本身,...

    JAVA项目——多线程下载代码

    本项目以"JAVA项目——多线程下载代码"为主题,使用Eclipse集成开发环境进行实现,适合于Java初学者或毕业设计实践。下面我们将深入探讨相关的Java多线程下载知识点。 1. **线程基础**:在Java中,线程是程序执行的...

    Java——多线程编程技术.pdf

    "Java多线程编程技术.pdf" Java多线程编程技术是Java平台的一个重要技术优势,是第一个直接把跨平台线程模型和标准的内存模型集成到语言中的主流语言。多线程编程技术是Java平台的一个重要技术优势,它可以使程序的...

    java上机报告5——Java的多线程编程1

    Java 多线程编程应用场景 —— 电影院售票系统设计 本资源摘要信息将对 Java 多线程编程在电影院售票系统设计中的应用进行详细介绍。该系统模拟了电影院三个售票窗口同时出售电影票的过程,通过 Java 多线程编程...

    Visual C++高级编程技术——MFC与多线程篇.rar

    4. 多线程编程:深入讲解如何使用CWinThread创建多线程应用程序,如何在不同线程间通信,以及如何处理线程安全问题。 5. 线程同步机制:介绍互斥量(CMutex)、事件(CEvent)和临界区(CCriticalSection)的使用,...

    JAVA多线程编程技术PDF

    在Java编程领域,多线程是一项至关重要的技术,它允许程序同时执行多个任务,从而提高系统资源的利用率和程序的响应速度。这份“JAVA多线程编程技术PDF”是学习和掌握这一领域的经典资料,涵盖了多线程的全部知识点...

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

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

    java编程那些事儿-陈跃峰71-102

    《Java编程那些事儿》是陈跃峰先生的一本关于Java编程技术的著作,该书深入浅出地介绍了Java编程中的重要概念和实践技巧。这里我们主要聚焦于压缩包中的部分内容,涉及Java集合框架、时间日期处理、文件操作、多线程...

    android——多线程

    标题"android——多线程"和描述"android——Handler与多线程应用范例"暗示我们将深入探讨如何在Android中使用Handler来管理多线程。 Android系统默认运行在一个单线程环境中,即主线程,也被称为UI线程。主线程主要...

    并发编程——Java线程的6种状态及切换(csdn)————程序.pdf

    并发编程——Java线程的6种状态及切换(csdn)————程序

    JAVA面试题——多线程.docx

    多线程是Java并发编程的基础,通过理解和掌握多线程的概念、实现方式及启动方法,可以帮助开发者更好地利用Java语言的优势,构建出高效稳定的并发程序。在实际开发过程中,应根据具体需求选择合适的实现方式,同时...

    文件复制——多线程

    1. **线程创建**:在Java或其他支持多线程的编程语言中,创建线程的方式有多种,如Java中的`Thread`类或者`Runnable`接口。创建线程后,需要定义线程的执行逻辑,比如文件读取和写入的代码。 2. **文件读取**:在多...

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

    Java多线程编程实战指南...本书以基本概念、原理与方法为主线,辅以丰富的实战案例和生活化实例,并从Java虚拟机、操作系统和硬件多个层次与角度出发,循序渐进、系统地介绍Java平台下的多线程编程核心技术及相关工具。

    学习JAVA编程,完成作业——编一个控制台梭哈游戏.zip

    在本项目中,"学习JAVA编程,完成作业——编一个控制台梭哈游戏.zip",开发者采用Java语言实现了一个基于控制台的梭哈(Poker)游戏。这是一门编程课程的作业,目的是让学生掌握Java的基本语法、面向对象编程概念...

    易语言实例模块——多线程模块.e

    易语言实例模块——多线程模块.e

    Java多线程-Socket编程

    Java 多线程-Socket 编程 Java 多线程-Socket 编程是指在 Java 语言中使用多线程技术来实现网络编程,特别是使用 Socket 编程来实现客户端和服务器端的通信。在 Java 中,多线程可以使用 Thread 类和 Runnable 接口...

Global site tag (gtag.js) - Google Analytics