Java提供了强制原子性的内置锁机制:synchronized块。
一个synchronized块有两部分:
锁对象的引用,以及这个锁保护的代码块。
每个Java对象都可以隐式地扮演一个用于同步的锁的角色;这些内置的锁被称为内部锁或监听器锁。执行线程进入synchronized块之前会自动获得锁;而无论通过正常控制路径退出,还是从块中抛出异常,线程都在放弃对synchronized块的控制时自动释放锁。获得内部锁的唯一途径是:进入这个内部锁保护的同步块或方法。
内部锁在Java中扮演了互斥锁的角色。
重进入:
当一个线程请求其他线程已经占有的锁时,请求线程将被阻塞。然而内部锁是可重进入的,因为线程在试图获得它自己占有的锁时,请求会成功。
重进入意味着所有的请求是基于“每个线程”,而不是基于“每个调用”的。
重进入方便了锁行为的封装,因此简化了面向对象并发代码。否则子类复写父类的synchronized类型的方法,并调用父类中的方法,那么就会产生死锁了。
你不可以将一个原子操作分解到多个synchronized块中。不过你应该尽量从synchronized块中分离耗时的且不影响共享状态的操作。
public void service(ServletRequest req, ServletResponse resp) {
BigInteger i = extractFromRequest(req);
BigInteger[] factors = null;
synchronized (this) {
++hits;
if (i.equals(lastNumber)) {
++cacheHits;
factors = lastFactors.clone();
}
}
if (factors == null) {
factors = factor(i);
synchronized (this) {
lastNumber = i;
lastFactors = factors.clone();
}
}
encodeIntoResponse(resp, factors);
}
请求与释放锁的操作需要开锁,所以将synchronized块分解得过于琐碎是不合理的,即使这样做是为了获得更好的原子性。当访问状态变量或者执行复合操作期间,CachedFactorizer会占有锁,但是执行潜在耗时的因数分解之前,它会释放锁。这样即保护了线程安全性,也不会过多地影响并发性。
public class NoVisibility {
private static boolean ready;
private static int number;
private static class ReaderThread extends Thread {
public void run() {
while (!ready)
Thread.yield();
System.out.println(number);
}
}
public static void main(String[] args) {
new ReaderThread().start();
number = 42;
ready = true;
}
}
Novisibility可能会一直保持循环,因为对于读线程来说,ready的值可能永远不可见。甚至可能会打印0,因为早在number赋值之前,主线程就已经写入ready并使之对读取线程可见,这是一种重拍序现象。读线程看到的顺序可能与发生写入的顺序正好相反,或者完全不同。
在没有同步的情况下,编译器,处理器,运行时安排操作的执行顺序可能完全出人意料。在没有进行适当同步的多线程程序中,尝试推断那些“必然”发生在内存中的动作时,你总是会判断错误。
有一个简单的方法来避免这些复杂的问题:只要数据需要被跨线程共享,就进行恰当的同步。
编写正确的并发程序的关键在于对共享的、可变的状态进行访问管理。
非原子的64位操作
JVM允许将64位的读或写划分为两个32位的操作。如果读和写发生在不同的线程,这种情况读取一个非volatile类型long就可能会出现得到一个值的高32位和另一个值的低32位。伪共享。
Volatile变量
当一个域声明为volatile类型后,编译器与运行时会监视这个变量,它是共享的,而且对它的操作不会与其他的内存操作一起被重排序。Volatice变量不会缓存在寄存器或者缓存在对其他处理器隐藏的地方。所以,读一个volatile类型的变量时,总会返回由某一线程所写入的最新值。
然而访问volatile变量的操作不会加锁,也就不会引起执行线程的阻塞,这使得volatile变量相对于synchronized而言,只是轻量级的同步机制。
正确使用volatile变量的方式包括:用于确保它们所引用的对象的状态的可见性,或者用于标识重要的生命周期事件的发生。
下面的例子示范了一种volatile变量的典型应用;检查状态标记,以确定是否退出一个循环。Asleep标记就必须是volatile的,否则执行检查的线程不会注意到asleep已被其他线程修改。
volatile boolean asleep;
...
while (!asleep)
countSomeSheep();
注意volatile的语义不足以使自增操作原子化,原子变量提供了“读改写”原子操作的支持,而且常被用作更优的volatile变量。
加锁可以保证可见性与原子性,volatile变量只能保证可见性。
只有满足了下面所有的标准后,你才能使用volatile变量:
1. 写入变量时并不以来变量的当前值;或者能够确保只有单一的线程修改变量的值。
2. 变量不需要与其他的状态变量共同参与不变约束。
3. 而且,访问变量时,没有其他的原因需要加锁。
发布和逸出
一个对象在尚未准备好时就将它发布,这种情况叫做逸出。
class UnsafeStates {
private String[] states = new String[] {
"AK", "AL" ...
};
public String[] getStates() { return states; }
}
以上面这种方式发布states会出现问题。任何一个调用者都能够修改它的内容。这个例子中states已经逸出它所属的范围。
线程封闭
线程封闭技术是实现线程安全的最简单方式之一。
ThreadLocal
它允许你将每个线程与持有数据的对象关联在一起。通过利用ThreadLocal存储JDBC连接。
这项技术还用于下面的情况:一个频繁执行的操作既需要像buffer这样的临时对象,同时还需要避免每次都重分配该临时对象。
与线程相关的值存储在线程对象自身中,线程终止后,这些值会被垃圾回收。
不可变性:
不可变对象永远是线程安全的。
只有满足如下状态,一个对象才是不可变的。
1. 它的状态不能在创建后再被修改。
2. 所有域都是final类型
3. 它被正确创建(创建期没有发生this引用的逸出)
使用volatile发布不可变对象
不论何时,对一组相关数值都应该执行原子操作,并且可以考虑为它们创建不可变的容器类,比如下面的OneValueCache。通过使用不可变对象来持有所有的变量,可以消除在访问和更新这些变量时的竞争条件。若使用可变的容器对象,你就必须使用锁以确保原子性;使用不可变对象,一旦一个线程获得了它的引用,永远不必担心其他线程会修改它的状态。
@Immutable
class OneValueCache {
private final BigInteger lastNumber;
private final BigInteger[] lastFactors;
public OneValueCache(BigInteger i,
BigInteger[] factors) {
lastNumber = i;
lastFactors = Arrays.copyOf(factors,
factors.length);
}
public BigInteger[] getFactors(BigInteger i) {
if (lastNumber == null || !lastNumber.equals(i))
return null;
else
return Arrays.copyOf(lastFactors, lastFactors.length);
} }
重点:当一个线程设置volatile类型的cache域引用到一个新的OneValueCache后,新数据会立即对其他线程可见。
与cache域相关的操作不会相互干扰,因为OneValueCache是不可变的,而且每次只有一条相应的代码路径访问它。不可变的容器对象持有与不变约束相关的多个状态变量,并利用volatile引用确保及时的可见性。
目前为止我们都关注确保对象不会被发布。比如,让对象限制在线程中或者另一个对象的内部。
@ThreadSafe
public class VolatileCachedFactorizer implements Servlet {
private volatile OneValueCache cache =
new OneValueCache(null, null);
public void service(ServletRequest req, ServletResponse resp) {
BigInteger i = extractFromRequest(req);
BigInteger[] factors = cache.getFactors(i);
if (factors == null) {
factors = factor(i);
cache = new OneValueCache(i, factors);
}
encodeIntoResponse(resp, factors);
}
}
安全发布的模式
如果一个对象不是不可变的,它就必须被安全地发布,通常发布线程与消费线程都必须同步化。
为了安全地发布对象,对象的引用以及对象的状态必须同时对其他线程可见。一个正确创建的对象可以通过下列条件安全地发布。
1. 通过静态初始化器初始化对象的引用;
2. 将它的引用存储到volatile域或AtomicReference;
3. 将它的引用存储到正确创建的对象的final域中。
4. 或者将它的引用存储到由锁正确保护的域中。
4条中的:置入Hashtable、synchronizedMap、ConcurrentMap中的主键以及键值,会安全地发布到可以从Map获得它们的任意线程中,无论是直接获得还是通过迭代器或者。
通常,以最简单和最安全的方式发布一个被静态创建的对象,就是使用静态初始化器:
public static Holder holder = new Holder(42)
静态初始化由JVM在类的初始阶段执行,由JVM内在的同步,该机制确保了以这种方式初始化的对象可以被安全地发布。
安全地共享对象
在并发程序中,使用和共享对象的一些最有效的策略如下:
1. 线程限制:一个线程限制的对象,通过限制在线程中,而被线程独占,且只能被占有它的线程修改。
2. 共享只读:一个共享只读对象,在没有额外同步的情况下,可以被多个线程并发地访问,但是任何线程都不能修改它。共享只读对象包括可变对象与搞笑不可变对象。
3. 共享线程安全:一个线程安全的对象在内部进行同步,所以其他线程无须额外同步,久可以通过公共接口随意地访问它。
4. 被守护的:一个被守护的对象只能通过特定的锁来访问。被守护的对象包括那些被线程安全对象封装的对象,和已知被特定的锁保护起来的已发布对象。
分享到:
相关推荐
一个进程中可以同时运行多个线程,每个线程可以执行不同的任务,这就是所谓的多线程。同一进程中的多个线程将共享该进程中的全部系统资源,如虚拟地址空间、文件描述符和信号处理等,但是同一个进程中的多个线程都有...
多线程是实现复杂任务并发执行的关键技术,能够提高资源利用率,优化系统响应时间。在STM32上实现多线程,通常会借助实时操作系统(RTOS)如RT-Thread。 RT-Thread是一个轻量级、开源的实时操作系统,它为STM32等微...
在计算机科学领域,多线程是一种程序设计技术,它允许应用程序同时执行多个任务或子任务。这极大地提高了软件的效率和响应性,特别是在现代多核处理器的环境下。本主题将深入探讨多线程的实现方式及其与单线程的对比...
"大漠多线程模板"是一个专门针对C#开发的多线程处理框架,它为开发者提供了便捷的方式来管理和优化多线程应用。这个框架由知名开发者"大漠"创建,旨在简化复杂的并发编程,提高代码的可读性和可维护性。 多线程允许...
安卓多线程,安卓高级开发中,必备技能,项目开发中都要使用
QTcpServer多线程 每个客户端连接的tcpSocket分别分配一个专门的线程来处理。 核心思想:继承并重写QTcpServer的incomingConnection函数去自己实现tcpsocket连接的建立和分配。 incomingConnection函数说明: 当...
多线程(英语:multithreading),是指从软件或者硬件上实现多个线程并发执行的技术。具有多线程能力的计算机因有硬件支持而能够在同一时间执行多于一个线程,进而提升整体处理性能。具有这种能力的系统包括对称多...
在IT行业中,多线程是程序设计中的一个重要概念,尤其在Java编程中,它被广泛应用于提高应用程序的并发性能和响应速度。本压缩包“多线程基础与基于多线程的简单聊天室”提供了对多线程技术的实践理解和二次开发的...
在IT领域,多线程是程序设计中的一个重要概念,尤其在现代计算机系统中,它能够提升应用程序的效率和响应性。MFC(Microsoft Foundation Classes)是微软提供的一个C++类库,用于简化Windows应用程序的开发,包括...
qt 使用QWebSocket 创建websocket客户端来读取数据,异步链接,并且放入到线程中去执行,线程池的基础,代码使用两个用户,放入到一个线程中执行,同理,可以多个用户放入到多个线程中执行,为线程池执行websocket ...
Java多线程是并发编程中的一个重要概念,它允许程序在同一时刻执行多个任务。以下是对Java多线程的深入理解: 线程概述 基本概念:线程是操作系统能够进行运算调度的最小单位,一个进程可以包含多个线程。 特性:...
在Windows Presentation Foundation(WPF)开发中,多线程是一个重要的技术,特别是在处理大量数据或进行耗时操作时,为了保持用户界面(UI)的响应性,通常会使用多线程来实现非UI任务。本实例是关于如何在WPF应用...
在Java编程中,多线程导入Excel数据是一项常见的任务,特别是在大数据处理和高并发场景下。这个场景通常涉及到性能优化和资源管理,以确保系统稳定性和数据一致性。下面将详细阐述多线程导入Excel数据的核心知识点。...
使用一个简单的数字递减案例,来模拟多线程下的工作逻辑
易语言关闭多线程句柄方法 易语言是一种功能强大且灵活的编程语言,多线程编程是其重要特性之一。然而,在多线程编程中,如何正确地关闭线程句柄是非常重要的。今天,我们将分享易语言关闭多线程句柄方法的相关知识...
Java多线程详解 在Java编程中,多线程是一种重要的技术,它使得程序能够同时执行多个任务,提高系统的效率和响应性。本教程将详细讲解Java中的多线程概念,包括线程的创建、状态、同步以及高级主题,旨在帮助初学者...
Java多线程是Java编程语言中一个非常重要的概念,它允许开发者在一个程序中创建多个执行线程并行运行,以提高程序的执行效率和响应速度。在Java中,线程的生命周期包含五个基本状态,分别是新建状态(New)、就绪...
易语言多线程启动详解 易语言多线程启动是指在易语言中使用多线程相关的API、支持库或模块来实现多线程编程的技术。多线程编程可以极大地提高程序的执行效率和响应速度,特别是在需要执行大量计算或I/O操作的场景下...
该文档是笔者在学习李刚老师《Java疯狂讲义》中有关多线程的用法而总结出来的笔记,其中主要的内容包括线程创建和启动、线程的生命周期、控制线程、线程同步、线程通信线程池等基本内容。对Java多线程有详细的介绍。
在C#编程中,多线程是一个至关重要的概念,尤其对于开发高性能、高并发的应用程序而言。本资源“C#多线程开发之并发编程经典实例”提供了丰富的实例,旨在帮助C#开发者深入理解并掌握多线程技术。以下是关于C#多线程...