在Java应用中,使用多线程进行工作的需要是越来越多,使用多线程进行工作,大大的提高了系统的工作效率,然而因此而产生的问题也是层出不穷,而且因为多线程而产生的问题跟踪是一个难题。
同步的概念:
同步分为 方法同步 和 同步块 两种方式。
使用同步的原因
1. 在系统中对访类要使用多线程进行访问;
2. 在该类中有 类变量, 或者是 在类的方法中有访问 公共资源(如一个外部文件的读写)。
同步所锁定的内容是什么?
无论你将Synchronized加在方法前还是加在一个变量前,其锁定的都是一个 类对象。 每一个对象都只有一个锁与之相关联。
下例中分情况的列举各种情况下的同步效果
1. Synchronized 加在方法上, (方法同步)
public synchronized void function1(){ ……}
public void function2(){
synchronized (this){……}
……}
这两种写法的效果是一样的,锁定的都是类实例对象。
如果有一个 类实例对象: inst = new ClassSynInst(),
另外有两个线程: threada,threadb,都调用了inst 对象,那么,在同一时间,如果 threada调用了inst.function1(),则threadb在该时间内不能访问inst.function1() 和 inst.function2(); 因为threada把inst这个对象的锁使用了,所以无法分给其它线程使用
但是,如果threada调用 inst1.function1(), threadb调用 inst2.function1(), 则可以同时进行,因为它们调用的是不同的ClassSynInst类对象实例。
2. Synchronized 加在变量上, (同步块)
Object a = new Object();
Object b = new Object();
public void function1(){
synchronized (a){……}
……}
public void function2(){
synchronized (b){……}
……}
这种情况下,是实现代码块锁定,锁定的对象是 变量 a 或 b; (注意,a 、b 都是非static 的)
如果有一个 类实例对象: inst = new ClassSynInst(),
另外有两个线程: threada,threadb,都调用了inst 对象,那么,在同一时间,如果 threada调用了inst.function1(),则threadb在该时间内可以访问inst.function2(); 但不能访问 inst.function1() 的同步块, 因为a被 threada锁定了。
3. Synchronized 锁定的是 类变量 ,即static 变量
class Test{
static Object o= new Object();
public static synchronized void f1(){ ……}
public static void f2(){
synchronized(Test.class){ ……}
}
public static void f3(){
try {
synchronized (Class.forName("Test")) { ……}
}
catch (ClassNotFoundException ex) {
}
}
public static void f4(){
synchronized(o){ ……}
}
}
以上4个方法中实现的效果都是一样的,其锁定的对象都是类Test,而不是类实例对象,即在多线程中,其共享的资源是属于类的,而不是属于类对象的。
在这种情况下,如果threada 访问了这4个方法中的任何一个, 在同一时间内其它的线程都不能访问 这4个方法。
4. 类的方法中访问了多线程共同的资源, 且该资源是可变的,这种情况下也是需要进行同步的
例:
class test {
static String path = “ file path”;
public void readConfiFile(){
synchronized (path){
// 读取该path指定的文件。
}
}
public void writeConfiFile(){
synchronized (path){
// 写信息到该path指定的文件。
}
}
}
这种情况下,必须锁定为 类变量,而不能进行锁定类实例对象,因为这是变象的一种类资源共享,而不是类实例对象资源共享。
线程,成也其,败也其,用好了可以提升性能,用不好则会使系统后患无穷。
PS: 进行线程同步需要很大的系统开销, 所以,在使用时,如果不是必须的,则尽量不使用同步功能。
<!--EndFragment-->
分享到:
相关推荐
在VC++编程环境中,线程同步是一个至关重要的概念,特别是在多线程程序设计中,以确保并发执行的线程能够安全地访问共享资源,避免数据竞争和其他潜在的问题。本篇文章将详细探讨线程锁在VC++中的应用,以及如何通过...
总结来说,本实验报告深入浅出地介绍了线程同步的概念,并通过实际操作展示了临界区问题及其解决方案。通过Mutex和软件方案的比较,帮助读者理解不同同步机制的工作原理和适用场景,为多线程编程提供了实用的指导。
MFC 多线程及线程同步 MFC 多线程及线程同步 MFC 多线程及线程同步
总结,多线程批量线程同步解决方案涵盖了多种技术,从基本的互斥量到复杂的线程池,都是为了在多线程环境中保证数据一致性、提高程序效率。理解并掌握这些概念和技术对于开发高效、可靠的并发应用至关重要。
总结起来,线程同步是保证多线程程序正确性的基础,而Windows互斥量和Peterson算法都是实现线程同步的有效方法。未使用同步机制可能会导致数据竞争,而使用互斥量则能提供一种较为可靠的保护手段。Peterson算法作为...
在编程领域,线程同步是多线程编程中的一个核心概念,它涉及到如何有效地管理和协调多个并发执行的线程,确保它们能正确地共享资源,避免数据竞争和死锁等问题。这个“线程同步小例子”是基于孙鑫先生著作中的示例...
线程同步是多线程编程中的重要概念,用于协调多个并发执行的线程,确保它们在访问共享资源时不会产生竞态条件或数据不一致性。在Windows编程中,提供了多种线程同步机制,包括互斥量、临界区、原子操作、事件以及...
"Java多线程同步.pdf" Java多线程同步是指在Java语言中,如何使用synchronized关键字和其他同步机制来确保多线程程序的正确执行。在Java语言中,synchronized关键字用于对方法或者代码块进行同步,但是仅仅使用...
总结来说,临界区、互斥内核对象、事件内核对象和信号量内核对象是实现线程同步的关键技术,它们有助于防止数据不一致性和资源竞争,保证多线程环境下的正确性和效率。理解并熟练运用这些工具对于开发高效、稳定的多...
总结,线程同步在多线程编程中扮演着至关重要的角色,而C#提供了多种工具,如Semaphore,来帮助开发者有效地管理和同步线程。通过理解并熟练运用这些机制,我们可以编写出高效且可靠的多线程应用程序。在实际项目中...
在编程领域,线程同步是多线程编程中的一个重要概念,它确保了多个线程在访问共享资源时的正确性和一致性。在这个“VC++线程同步实例”中,我们将探讨如何利用VC++(Visual C++)来实现线程间的同步,以避免数据竞争...
在多线程编程中,线程同步是一种控制多个线程并发执行时访问共享资源的方式,以避免数据不一致和死锁等问题。以下是对线程同步的四种主要方式的详细解释: 1. **事件(Event)** 事件是Windows API提供的一种线程...
1.使用三种VC的多线程同步方法编写一个多线程的程序(要求在屏幕上先显示Hello,再显示World)。 1)基于全局变量的多线程同步程序; 2)基于事件的多线程同步程序; 3)基于临界区的多线程同步程序。
关于线程同步,需要牢牢记住的第一点是:线程同步就是线程排队。同步就是排队。线程同步的真实意思和字面意思恰好相反。线程同步的目的就是避免线程“同步”执行。 关于线程同步,需要牢牢记住的第二点是 “共享”...
本篇文章将深入探讨三种在C++中实现多线程同步的方法:事件对象、关键代码段和互斥对象。 首先,我们来看**事件对象**。事件对象是一种信号机制,用于线程间通信和同步。在Windows API中,CreateEvent函数创建一个...
C#线程同步是多线程编程中的一个重要概念,它涉及到如何控制多个线程对共享资源的访问,以避免数据不一致性和竞态条件。在C#中,线程同步通常用于确保在某一时刻只有一个线程可以访问特定的代码块或资源,从而保证...
在IT领域,线程同步和事件机制是多线程编程中的关键概念,特别是在Windows系统下,Visual C++提供了丰富的工具来实现这些功能。本项目"Visual C++事件机制线程同步工程"着重于如何利用事件对象(CEvent)进行线程间...