线程基础(第二部分)Java线程的缺陷和副作用几解决办法

<--在线程基础的第二部分中,我们将了解一下使用Java线程的缺陷和副作用,以及在SUN JDK 1.2中是如何修改线程的运行机制的-->

在上篇文章《Java 101之线程基础》中,我们介绍了线程的概念以及如何使用线程。这个月,我们将转到更高级的话题,包括线程的缺陷及副作用,以及在SUN JDK 1.2中,是如何改进线程的运行机制的。

synchronize(同步)

让我们回忆一下上篇文章中讲的:线程允许两个或者更多个进程同时执行。实际上,这些线程也可以共享对象和数据,在这种情形下,你要知道不同的线程在同一时间内不能存取同一数据,因为一开始设计Java的时候,就采用了线程的概念,Java语言定义了一个特殊的关键字synchronize(同步),该关键字可以应用到代码块上,代码块也包括入口方法,该关键字的目的是防止多个线程在同一时间执行同一代码块内的代码。

定义一个同步的方法,格式如下:

[public|private] synchronized {type}

methodname(...)

一个把同步这个关键字应用到方法中的简单的例子:

public class someClass {
public void aMethod() {
// Some code
synchronized(this) {
// Synchronized code block
}
// more code.
}
}

同步化的关键字可以保证在同一时间内只有一个线程可以执行该代码段,而任何其他要用到该段代码的线程将被阻塞,直到第一个线程执行完该段代码。

死锁和饥饿

对于饥饿的定义-由于别的并发的激活的过程持久占有所需资源,是莫个异步过程载客预测的时间内不能被激活。

最常遇到的线程的两个缺陷是死锁和饥饿。当一个或者多个进程,在一个给定的任务中,协同作用,互相干涉,而导致一个或者更多进程永远等待下去,死锁就发生了。与此类似,它当一个进程永久性地占有资源,使得其他进程得不到该资源,就发生了饥饿。

首先我们看一下死锁问题。考虑一个简单的例子,假如你到ATM机上取钱,但是你却看到如下的信息“现在有没有现金,请等会儿再试。”你需要钱,所以你就等了一会儿再试,但是你又看到同样的信息。与此同时,在你后面,一辆运款装甲车正等待着把钱放进ATM中,但是运款装甲车到不了ATM取款机,因为你的汽车挡着道。而你又要取到钱,才会离开原地。这种情况下,就发生了死锁。

在饥饿的情形下,系统不处于死锁状态中,因为有一个进程仍在处理之中,只是其他进程永远得不到执行的机会。在什么样的环境下,会导致饥饿的发生,没有预先确定好的规则。而一旦发生下面四种情况之一,就会导致死锁的发生。

相互排斥: 一个线程或者进程永远占有一共享资源,例如,独占该资源。

循环等待: 进程A等待进程B,而后者又在等待进程C,而进程C又在等待进程A。

部分分配: 资源被部分分配。例如,进程A和B都需要用访问一个文件,并且都要用到打印机,进程A获得了文件资源,进程B获得了打印机资源,但是两个进程不能获得全部的资源。

缺少优先权: 一个进程访问了某个资源,但是一直不释放该资源,即使该进程处于阻塞状态。

如果上面四种情形都不出现,系统就不会发生死锁。请再看一下刚才的文件/打印机的例子,当其中一个进程判断出它得不到它所需要的第二个资源,就释放已经得到的第一个资源,那么第二个教程可以获得两个资源,并能够运行下去。

时间: 2024-11-02 20:48:32

线程基础(第二部分)Java线程的缺陷和副作用几解决办法的相关文章

Java enum关键字不识别的快速解决办法_java

从别人那儿拷贝过来的myeclipse java工程,打开一看标红了一大片,仔细一看,原来是不识别enum关键字,这就有点尴尬了. 我自己重新建了一个java工程,测试了下,假如我在新建工程的时候选择JavaSE-1.6: 是识别enum关键字的,但是假如我选择第二个或者默认的jdk1.8,那就不行了,后来发现java compiler compliance level中的最高上限也才1.7,所以,两种解决方法(原理是一样的),一是将compiler compliance level设置到1.5

线程中止-请教一个JAVA线程的奇怪问题

问题描述 请教一个JAVA线程的奇怪问题 在学习JAVA线程时候的遇到一个很奇怪的现象.让我们先来看代码 public class TestThread { public static void main(String[] args) { IRun ir = new IRun(); Thread it = new Thread(ir); it.start(); try { Thread.sleep(1000); } catch (InterruptedException ex) { Logger

java中提示unmappable character for encoding解决办法

•删除该字符 •将源代码文件存为UTF-8编码 unmappable character for encoding错误, 不过上网查询后得知,java的class文件使用的是UTF-8编码.test2.java被编译到test2.class 的过程,包含到 UTF-8编码转换的过程. 解决办法  代码如下 复制代码 编译 javac -encoding gbk test2.java 运行 java -Dfile.encoding="GBK" test2

输出java进程的jstack信息示例分享 通过线程堆栈信息分析java线程_java

复制代码 代码如下: #!/bin/shpro_name=java #process namekeys=`ps -ef |grep "$pro_name" |grep -v "grep" | awk '{print $2}'`nowdate=`date +%Y%m%d%H%M%S` jstackpath="/usr/java/jdk1.6.0_07/bin/jstack"cpulogpath="/home/" for key

java在mac下乱码问题各种解决办法

  java和javac在简体中文的Mac OSX的终端(Terminal.app)环境下,默认是以GBK编码的中文输出各种诸如语法错误,数组访问越界之类的信息. 但是,Mac的终端的默认编码是UTF-8,所以javac和java在未配置的情况下会输出乱码. 解决方法有3个: 1)指定输出的编码为UTF-8 javac -J-Dfile.encoding=UTF-8 2)输出英文 export LC_ALL=en 3)通过iconv转码 javac something|iconv -f GBK

java线程技术——线程的创建运行终止

http://blog.chinaunix.net/uid-122937-id-192835.html 1.创建和运行线程在Java中,多线程的实现有两种方式:      扩展java.lang.Thread类      实现java.lang.Runnable接口 (1)扩展Thread类      Thread Test = new Thread();      Test.start(); (2)实现Runnable接口将实现Runnable接口的类实例化            Test i

Java 线程同步 synchronized

先来看一个不带线程同步的例子,这个例子很简单,只是让两个线程输出同样的内容,并不做其他的事, 所以,线程同步在这里体现的并不明显. import java.util.Date; public class ThreadTest extends Thread{ int pauseTime; String name; /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method s

Java线程系列

Java线程:线程私有变量 Java线程:深入ThreadLocal Java线程:线程的调度-休眠 Java线程:线程的调度-优先级 Java线程:线程的调度-让步 Java线程:线程的调度-合并 Java线程:线程的调度-守护线程 Java线程:线程的同步-同步方法 Java线程:线程的同步-同步块 Java线程:并发协作-生产者消费者模型 Java线程:并发协作-死锁 Java线程:volatile关键字 Java线程:新特征-线程池 Java线程:新特征-有返回值的线程 Java线程:新

Java线程:线程的调度-优先级

线程的优先级用1-10之间的整数表示,数值越大优先级越高,默认的优先级为5. 在一个线程中开启另外一个新线程,则新开线程称为该线程的子线程,子线程初始优先级与父线程相同. /** * Java线程:线程的调度-优先级 * * @author leizhimin */ public class Test { public static void main(String[] args) { Thread t1 = new MyThread1(); Thread t2 = new Thread(new