歡迎您光臨本站 註冊首頁

Java Synchronized鎖失敗案例及解決方案

←手機掃碼閱讀     kyec555 @ 2020-05-01 , reply:0

synchronized關鍵字,一般稱之為”同步鎖“,用它來修飾需要同步的方法和需要同步代碼塊,默認是當前對象作為鎖的對象。

同步鎖鎖的是同一個對象,如果對象發生改變,則鎖會不生效。

鎖失敗的代碼:

public class IntegerSynTest { //線程實現Runnable接口 private static class Worker implements Runnable{ private Integer num; public Worker(Integer num){ this.num=num; } @Override public void run() { synchronized (num){ Thread thread = Thread.currentThread(); //System.identityHashCode:返回原生的hashCode值,不管Object對象是被重寫;空引用的哈希代碼為零 System.out.println(thread.getName()+"--@:---"+System.identityHashCode(num)); num++; System.out.println(thread.getName()+"------num:"+num+"---"+System.identityHashCode(num)); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(thread.getName()+"------num:"+num+"---"+System.identityHashCode(num)); } } public static void main(String[] args) { Worker worker = new Worker(1); for (int i = 0; i < 5; i++) { new Thread(worker).start(); } } } }

鎖失敗的運行結果:

鎖失敗的原因:

1.num++ 的 .class 實現是這樣的 Integer integer1 = this.num, integer2 = this.num = Integer.valueOf(this.num.intValue() + 1);

2.查看 Integer.valueOf()的源代碼

這時發現,它是重新 new出一個新的Integer,這樣的話,每 ++一次,那麼就會產生一個新的對象,而Synchronize鎖是鎖同一個對象,當鎖不同對象時,則會鎖失敗。

解決方法:

Synchronized同步鎖只要鎖的對象不發生改變即可,那麼由此只需要聲明一個對象,不修改它,鎖這一個對象即可(還有其他方法暫不一一列舉,以後也不會列舉了)。

鎖成功的代碼

public class IntegerSynTest { //線程實現Runnable接口 private static class Worker implements Runnable{ private Integer num; /** * ---重點看這裡--- * 聲明要鎖的對象 * ---重點看這裡--- */ private Object object = new Object(); public Worker(Integer num){ this.num=num; } @Override public void run() { //修改鎖對象 synchronized (num){ Thread thread = Thread.currentThread(); //System.identityHashCode:返回原生的hashCode值,不管Object對象是被重寫;空引用的哈希代碼為零 System.out.println(thread.getName()+"--@:---"+System.identityHashCode(num)); num++; System.out.println(thread.getName()+"------num:"+num+"---"+System.identityHashCode(num)); try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(thread.getName()+"------num:"+num+"---"+System.identityHashCode(num)); } } public static void main(String[] args) { Worker worker = new Worker(1); for (int i = 0; i < 5; i++) { new Thread(worker).start(); } } } }


[kyec555 ] Java Synchronized鎖失敗案例及解決方案已經有223次圍觀

http://coctec.com/docs/java/show-post-232405.html