一、同步方法
public synchronized void methodAAA(){
//….
}
锁定的是调用这个同步方法的对象
测试:
a、不使用这个关键字修饰方法,两个线程调用同一个对象的这个方法。
目标类:
1public class TestThread {
2 public void execute(){ //synchronized,未修饰
3 for(int i=0;i<100;i++){
4 System.out.println(i);
5 }
6 }
7}
线程类:
1public class ThreadA implements Runnable{
2 TestThread test=null;
3 public ThreadA(TestThread pTest){ //对象有外部引入,这样保证是同一个对象
4 test=pTest;
5 }
6 public void run() {
7 test.execute();
8 }
9}
调用:
1TestThread test=new TestThread();
2Runnable runabble=new ThreadA(test);
3Thread a=new Thread(runabble,"A");
4a.start();
5Thread b=new Thread(runabble,"B");
6b.start();
结果:
输出的数字交错在一起。说明不是同步的,两个方法在不同的线程中是异步调用的。
b、修改目标类,增加synchronized修饰
1public class TestThread {
2 public synchronized void execute(){ //synchronized修饰
3 for(int i=0;i<100;i++){
4 System.out.println(i);
5 }
6 }
7}
结果:
输出的数字是有序的,首先输出A的数字,然后是B,说明是同步的,虽然是不同的线程,但两个方法是同步调用的。
注意:上面虽然是两个不同的线程,但是是同一个实例对象。下面使用不同的实例对象进行测试。
c、每个线程都有独立的TestThread对象。
目标类:
1public class TestThread {
2 public synchronized void execute(){ //synchronized修饰
3 for(int i=0;i<100;i++){
4 System.out.println(i);
5 }
6 }
7}
线程类:
1public class ThreadA implements Runnable{
2 public void run() {
3 TestThread test=new TestThread();
4 test.execute();
5 }
6}
7
调用:
1Runnable runabble=new ThreadA();
2Thread a=new Thread(runabble,"A");
3a.start();
4Thread b=new Thread(runabble,"B");
5b.start();
结果:
输出的数字交错在一起。说明虽然增加了synchronized 关键字来修饰方法,但是不同的线程调用各自的对象实例,两个方法仍然是异步的。
引申:
对于这种多个实例,要想实现同步即输出的数字是有序并且按线程先后顺序输出,我们可以增加一个静态变量,对它进行加锁(后面将说明锁定的对象)。
修改目标类:
1public class TestThread {
2 private static Object lock=new Object(); //必须是静态的。
3 public void execute(){
4 synchronized(lock){
5 for(int i=0;i<100;i++){
6 System.out.println(i);
7 }
8 }
9 }
10}
二、同步代码块
1public void method(SomeObject so){
2 synchronized(so)
3 //…..
4 }
5}
锁定一个对象,其实锁定的是该对象的引用(object reference)
谁拿到这个锁谁就可以运行它所控制的那段代码。当有一个明确的对象作为锁时,就可以按上面的代码写程序,但当没有明确的对象作为锁,只是想让一段代码同步时,可以创建一个特殊的instance变量(它必须是一个对象)来充当锁(上面的解决方法就是增加了一个状态锁)。
a、锁定一个对象,它不是静态的
private byte[] lock = new byte[0]; // 特殊的instance变量
目标类:
1public class TestThread {
2 private Object lock=new Object();
3 public void execute(){
4 synchronized(lock){ //增加了个锁,锁定了对象lock,在同一个类实例中,是线程安全的,但不同的实例还是不安全的。
5
6因为不同的实例有不同对象锁lock
7 for(int i=0;i<100;i++){
8 System.out.println(i);
9 }
10 }
11 }
12}
其实上面锁定一个方法,等同于下面的:
1public void execute(){
2 synchronized(this){ //同步的是当然对象
3 for(int i=0;i<100;i++){
4 System.out.println(i);
5 }
6 }
7}
b、锁定一个对象或方法,它是静态的
这样锁定,它锁定的是对象所属的类
public synchronized static void execute(){
//...
}
等同于
1public class TestThread {
2 public static void execute(){
3 synchronized(TestThread.class){
4 //
5 }
6 }
7}
测试:
目标类:
1public class TestThread {
2 private static Object lock=new Object();
3 public synchronized static void execute(){ //同步静态方法
4 for(int i=0;i<100;i++){
5 System.out.println(i);
6 }
7 }
8 public static void execute1(){
9 for(int i=0;i<100;i++){
10 System.out.println(i);
11 }
12 }
13 public void test(){
14 execute(); //输出是有序的,说明是同步的
15 //execute1(); //输出是无须的,说明是异步的
16 }
17}
线程类:调用不同的方法,于是建立了两个线程类
1public class ThreadA implements Runnable{
2 public void run() {
3 TestThread.execute();//调用同步静态方法
4 }
5}
6public class ThreadB implements Runnable{
7 public void run() {
8 TestThread test=new TestThread();
9 test.test();//调用非同步非静态方法
10 }
11}
调用:
1Runnable runabbleA=new ThreadA();
2Thread a=new Thread(runabbleA,"A");
3a.start();
4Runnable runabbleB=new ThreadB();
5Thread b=new Thread(runabbleB,"B");
6b.start();
注意:
1、用synchronized 来锁定一个对象的时候,如果这个对象在锁定代码段中被修改了,则这个锁也就消失了。看下面的实例:
目标类:
1public class TestThread {
2 private static final class TestThreadHolder {
3 private static TestThread theSingleton = new TestThread();
4 public static TestThread getSingleton() {
5 return theSingleton;
6 }
7 private TestThreadHolder() {
8 }
9 }
10
11 private Vector ve =null;
12 private Object lock=new Object();
13 private TestThread(){
14 ve=new Vector();
15 initialize();
16 }
17 public static TestThread getInstance(){
18 return TestThreadHolder.getSingleton();
19 }
20 private void initialize(){
21 for(int i=0;i<100;i++){
22 ve.add(String.valueOf(i));
23 }
24 }
25 public void reload(){
26 synchronized(lock){
27 ve=null;
28 ve=new Vector();
29 //lock="abc";
30 for(int i=0;i<100;i++){
31 ve.add(String.valueOf(i));
32 }
33 }
34 System.out.println("reload end");
35 }
36
37 public boolean checkValid(String str){
38 synchronized(lock){
39 System.out.println(ve.size());
40 return ve.contains(str);
41 }
42 }
43}
说明:在reload和checkValid方法中都增加了synchronized关键字,对lock对象进行加锁。在不同线程中对同一个对象实例分别调用reload和checkValid方法。
在reload方法中,不修改lock对象即注释lock="abc"; ,结果在控制台输出reload end后才输出100。说明是同步调用的。
如果在reload方法中修改lock对象即去掉注释,结果首先输出了一个数字(当前ve的大小),然后输出reload end。说明是异步调用的。
2、单例模式中对多线程的考虑
1public class TestThread {
2 private static final class TestThreadHolder {
3 private static TestThread theSingleton = new TestThread();
4 public static TestThread getSingleton() {
5 return theSingleton;
6 }
7 private TestThreadHolder() {
8 }
9 }
10 private Vector ve =null;
11 private Object lock=new Object();
12 private TestThread(){
13 ve=new Vector();
14 initialize();
15 }
16 public static TestThread getInstance(){
17 return TestThreadHolder.getSingleton();
18 }
19 '''
20}
说明:增加了一个内部类,在内部类中申明一个静态的对象,实例化该单例类,初始化的数据都在单例类的构造函数中进行。这样保证了多个实例同时访问的时候,初始化的数据都已经成功初始化了。
总结:
A. 无论synchronized关键字加在方法上还是对象上,它取得的锁都是对象,而不是把一段代码或函数当作锁,所以首先应知道需要加锁的对象
B.每个对象只有一个锁(lock)与之相关联。
C.实现同步是要很大的系统开销作为代价的,甚至可能造成死锁,所以尽量避免无谓的同步控制。
分享到:
相关推荐
在 Java 中,synchronized 关键字可以作用于 instance 变量、object reference(对象引用)、static 函数和 class literals(类名称字面常量)身上。 Synchronized 关键字的作用是取得对象的锁,而不是把一段代码或...
Java关键字volatile和synchronized作用和区别 Java关键字volatile和synchronized是Java语言中两个常用的关键字,它们都是用来解决多线程编程中的同步问题的,但它们的作用和用法却有所不同。 volatile关键字是变量...
一、synchronized 关键字的作用域 synchronized 关键字可以被用于两种范围内:对象实例和类范围。 1. 对象实例范围:synchronized 关键字可以被用于对象实例的方法上,以防止多个线程同时访问同一个对象的 ...
`synchronized`关键字可以修饰方法或用作代码块,其主要作用是控制对共享资源的并发访问,防止数据不一致和竞态条件。当一个线程进入`synchronized`代码块或方法时,其他试图进入同一块的线程将被阻塞,直到持有锁的...
### volatile与synchronized的区别 #### 一、锁的特性:互斥与可见性 在并发编程中,锁作为实现线程安全的一种手段,其核心作用在于提供两种特性:互斥和可见性。 - **互斥**:互斥是指在任何时刻,只允许一个...
二、`synchronized`的作用 1. **线程互斥**:当一个线程在执行`synchronized`代码时,其他线程必须等待该线程释放锁后才能进入。 2. **内存可见性**:确保线程在读取或修改共享变量时,能看到其他线程对变量的最新...
### synchronized关键字的深入解析 #### 一、synchronized关键字的重要性 `synchronized`关键字在Java语言中扮演着极其重要的角色,它是实现线程安全的核心手段之一。通过`synchronized`关键字,开发人员可以在多...
`synchronized`关键字在Java多线程编程中起到关键作用,它可以防止数据竞争,保证共享资源的线程安全。通过同步方法和同步语句块,我们可以控制对共享数据的访问,并通过类锁和对象锁区分不同级别的锁定粒度。结合...
#### 二、synchronized的作用机制 `synchronized`主要通过对象锁(也称为监视器锁)来实现其功能。当一个线程进入某个对象的一个`synchronized`代码块时,它会自动获得该对象的锁;离开该代码块时,则自动释放锁。...
4. **synchronized的作用** - **防止数据不一致**:通过同步,可以避免多个线程同时修改共享数据,从而保证数据的一致性。 - **避免死锁**:虽然`synchronized`不能完全避免死锁,但它可以帮助降低发生死锁的概率...
当synchronized作用于函数时,实际上锁的也是对象,锁定的对象就是该函数所在类的对象。而synchronized作用于class时则是锁的这个Class类,并非具体对象。 public class SynchronizedClass { public synchronized ...
当`synchronized`作用于实例方法时,它锁定的是调用该方法的对象实例,这意味着在同一时间,只有一个线程可以执行该对象上的同步方法。如果存在多个对象实例,即使它们的同步方法相同,不同实例的同步方法也可以同时...
在深入探讨`synchronized`的关键知识点之前,我们先来明确`synchronized`在Java中的核心作用:它是一种用于实现线程同步的机制,确保了共享资源在多线程环境下的正确访问和修改,避免了数据不一致性和竞态条件等问题...
Synchronized 同步锁的作用是解决多线程访问实例变量时可能造成的数据脏读现象。Synchronized 的使用方法有四种: 1. synchronized 修饰普通方法 2. synchronized 修饰普通代码块 3. synchronized 修饰静态方法 4. ...
本示例"java synchronized demo"旨在探讨`synchronized`关键字的用法及其作用机制。下面将详细阐述`synchronized`的相关知识点。 1. **synchronized的作用**: - `synchronized`关键字主要用于解决多线程环境中的...
当synchronized作用于方法时,它锁定的是调用这个方法的对象实例;当作用于静态方法时,它锁定的是类对象。因此,静态方法锁可以看作是类级别的全局锁。synchronized关键字确保了线程的互斥访问,同时也提供了线程间...
synchronized作用于方法或代码块,确保在进入和退出同步区域时,对共享变量的修改对其他线程可见。 总之,Java内存模型通过规定内存访问规则,解决了并发环境下的数据一致性问题。理解和掌握JMM,使用volatile和...
### Synchronized与ThreadLocal #### 一、Synchronized机制详解 **Synchronized** 是 Java 中一个非常重要的关键字,主要用于实现线程同步。它通过在对象上加锁来确保多个线程能够安全地访问共享资源。 - **作用...