java synchronized详解(一)

 第一篇:  

Java语言的synchronized关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。

一、当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。

二、然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。

三、尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。

四、第三个例子同样适用其它同步代码块。也就是说,当一个线程访问object的一个synchronized(this)同步代码块时,它就获得了这个object的对象锁。结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。

五、以上规则对其它对象锁同样适用.

举例说明:
一、当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。

package ths;

public class Thread1 implements Runnable {
public void run() {
synchronized(this) {
for (int i = 0; i < 5; i++) {
System.out.println(Thread.currentThread().getName() + " synchronized loop " + i);
}
}
}
public static void main(String[] args) {
Thread1 t1 = new Thread1();
Thread ta = new Thread(t1, "A");
Thread tb = new Thread(t1, "B");
ta.start();
tb.start();
}
}

结果:
A synchronized loop 0
A synchronized loop 1
A synchronized loop 2
A synchronized loop 3
A synchronized loop 4
B synchronized loop 0
B synchronized loop 1
B synchronized loop 2
B synchronized loop 3
B synchronized loop 4

二、然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。

package ths;

public class Thread2 {
public void m4t1() {
synchronized(this) {
int i = 5;
while( i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}
}
public void m4t2() {
int i = 5;
while( i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}
public static void main(String[] args) {
final Thread2 myt2 = new Thread2();
Thread t1 = new Thread( new Runnable() { public void run() { myt2.m4t1(); } }, "t1" );
Thread t2 = new Thread( new Runnable() { public void run() { myt2.m4t2(); } }, "t2" );
t1.start();
t2.start();
}
}

结果:
t1 : 4
t2 : 4
t1 : 3
t2 : 3
t1 : 2
t2 : 2
t1 : 1
t2 : 1
t1 : 0
t2 : 0

三、尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。

//修改Thread2.m4t2()方法:
public void m4t2() {
synchronized(this) {
int i = 5;
while( i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}

}

结果:

t1 : 4
t1 : 3
t1 : 2
t1 : 1
t1 : 0
t2 : 4
t2 : 3
t2 : 2
t2 : 1
t2 : 0

四、第三个例子同样适用其它同步代码块。也就是说,当一个线程访问object的一个synchronized(this)同步代码块时,它就获得了这个object的对象锁。结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。

//修改Thread2.m4t2()方法如下:

public synchronized void m4t2() {
int i = 5;
while( i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}

结果:
t1 : 4
t1 : 3
t1 : 2
t1 : 1
t1 : 0
t2 : 4
t2 : 3
t2 : 2
t2 : 1
t2 : 0

五、以上规则对其它对象锁同样适用:

package ths;

public class Thread3 {
class Inner {
private void m4t1() {
int i = 5;
while(i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : Inner.m4t1()=" + i);
try {
Thread.sleep(500);
} catch(InterruptedException ie) {
}
}
}
private void m4t2() {
int i = 5;
while(i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : Inner.m4t2()=" + i);
try {
Thread.sleep(500);
} catch(InterruptedException ie) {
}
}
}
}
private void m4t1(Inner inner) {
synchronized(inner) { //使用对象锁
inner.m4t1();
}

private void m4t2(Inner inner) {
inner.m4t2();
}

public static void main(String[] args) {
final Thread3 myt3 = new Thread3();
final Inner inner = myt3.new Inner();
Thread t1 = new Thread( new Runnable() {public void run() { myt3.m4t1(inner);} }, "t1");
Thread t2 = new Thread( new Runnable() {public void run() { myt3.m4t2(inner);} }, "t2");
t1.start();
t2.start();
}
}

 

结果:

尽管线程t1获得了对Inner的对象锁,但由于线程t2访问的是同一个Inner中的非同步部分。所以两个线程互不干扰。

t1 : Inner.m4t1()=4
t2 : Inner.m4t2()=4
t1 : Inner.m4t1()=3
t2 : Inner.m4t2()=3
t1 : Inner.m4t1()=2
t2 : Inner.m4t2()=2
t1 : Inner.m4t1()=1
t2 : Inner.m4t2()=1
t1 : Inner.m4t1()=0
t2 : Inner.m4t2()=0

现在在Inner.m4t2()前面加上synchronized:

private synchronized void m4t2() {
int i = 5;
while(i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : Inner.m4t2()=" + i);
try {
Thread.sleep(500);
} catch(InterruptedException ie) {
}
}
}

结果:

尽管线程t1与t2访问了同一个Inner对象中两个毫不相关的部分,但因为t1先获得了对Inner的对象锁,所以t2对Inner.m4t2()的访问也被阻塞,因为m4t2()是Inner中的一个同步方法。

t1 : Inner.m4t1()=4
t1 : Inner.m4t1()=3
t1 : Inner.m4t1()=2
t1 : Inner.m4t1()=1
t1 : Inner.m4t1()=0
t2 : Inner.m4t2()=4
t2 : Inner.m4t2()=3
t2 : Inner.m4t2()=2
t2 : Inner.m4t2()=1
t2 : Inner.m4t2()=0

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Javasynchronized关键字可以用来保证一段代码在同一时间只能被一个线程执行。这样可以避免多线程并发导致的数据不一致问题。 synchronized可以应用于方法和代码块。 在方法上使用时,表示整个方法都是同步的,在代码块中使用时,表示代码块中的语句是同步的。 另外,synchronized还可以用来保证共享变量的可见性和有序性。 例如: ``` public class Example { private int sharedData; public synchronized void setData(int data) { sharedData = data; } public synchronized int getData() { return sharedData; } } ``` 上面的例子中,setData和getData方法都是同步的,保证了对sharedData的访问是线程安全的。 ### 回答2: Java中的synchronized关键字是多线程编程中的重要工具,用于解决线程之间的同步问题。当多个线程同时对共享资源进行读写时,就有可能发生冲突或数据不一致的情况。synchronized关键字可以保证同一时间只有一个线程能够访问共享资源,从而避免了数据的不一致性以及避免了竞态条件。 synchronized关键字一般分为两种场景:方法同步 和 代码块同步。 - 方法同步 方法同步是最简单的同步方式,使用synchronized修饰方法,表示该方法是同步的。 synchronized修饰的方法,当一个线程进入该方法后,其他线程就不能进入该方法了,只有等待该线程执行完成后,才能继续执行其他线程。方法同步虽然简单,但是对性能会有所影响,因为每次只允许一个线程进入方法,其他线程都会被暂时阻塞。 - 代码块同步 代码块同步,是在同步控制方法的基础上,进一步细化代码的同步范围,使用synchronized(this){ … } 或 synchronized(obj){ … }的方式。 代码块同步和方法同步的不同在于,代码块同步只锁定了代码块中的共享资源,而方法同步是锁定整个方法。因此,如果在一个类中有多个共享资源,且这些共享资源之间没有必然的关系,那么就应该使用代码块同步对每个共享资源进行同步控制,而不是所有的同步都放在一个方法中进行,以免浪费不必要的时间和资源。 此外,synchronized还有一个重要的概念:对象监视器。对象监视器是实现synchronized同步的基础,当代码进入synchronized代码块时,会尝试获取对象监视器的锁,如果获取成功,则当前线程可以顺利执行同步代码块,如果获取失败,则当前线程就会被挂起等待,直到获取锁成功时才能进入同步代码块。 总之,在多线程编程中,synchronized是一种非常重要的同步方式,可以有效地避免线程之间的资源竞争和数据不一致性问题。正确地使用synchronized可以大大提高程序的稳定性和可靠性,减少线程安全问题的出现。 ### 回答3: Java中的synchronized关键字用于实现并发编程,防止多个线程同时访问共享数据时可能会发生的数据不一致、安全性等问题。当一个线程获得某个对象的锁时,其他试图访问该对象的线程将被挂起,直到该线程释放该对象锁。 synchronized可以应用于方法或代码块中,分别称为方法同步和代码块同步。在方法同步中,synchronized关键字修饰整个方法,表示该方法代码块为原子操作,只有获得该方法锁的线程才能访问该方法;在代码块同步中,synchronized关键字修饰代码块,用于锁定某个对象,只有获取该对象锁的线程才能执行该代码块。 synchronized语法格式如下: 1.方法同步: public synchronized void method() { // todo something... } 2.代码块同步: public void method() { synchronized (this) { // 同步锁为当前对象 // todo something... } } synchronized关键字的作用是保证多个线程之间的同步性,但也有一些注意点: 1.synchronized锁住的范围要尽可能的小,这样可以避免一些不必要的线程阻塞和等待。如果锁住的范围过大,会大大影响程序的性能。 2.在代码块同步中,synchronized锁的对象应尽可能不发生改变。 3.synchronized具有可重入性,即如果一个线程已经持有了某个对象的锁,可以多次获取该对象锁而不发生死锁。 总之,synchronizedJava中用于实现并发编程的一种工具,合理使用可以保证程序的同步性和安全性。但在避免死锁、线程安全等方面仍需进一步了解和规避。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值