Java中synchronized關鍵字修飾方法同步的用法詳解
Java的最基本的同步方式,即使用synchronized關鍵字來控制一個方法的并發(fā)訪問。 每一個用synchronized關鍵字聲明的方法都是臨界區(qū)。在Java中,同一個對象的臨界區(qū),在同一時間只有一個允許被訪問。
靜態(tài)方法則有不同的行為。用synchronized關鍵字聲明的靜態(tài)方法,同時只能夠被一個執(zhí)行線程訪問,但是其他線程可以訪問這個對象的非靜態(tài)的synchronized方法。必須非常謹慎這一點,因為兩個線程可以同時訪問一個對象的兩個不同的synchronized方法,即其中一個是靜態(tài)synchronized方法,另一個是非靜態(tài)synchronized方法。如果兩個方法都改變了相同的數據,將會出現數據不一致的錯誤。
synchronized塊的語法如下:
public void method()
{
synchronized(表達式)
{
}
}
synchronized關鍵字有兩種用法,一種是只用于方法的定義中,另外一種是synchronized塊,我們不僅可以使用synchronized來同步一個對象變量,你也可以通synchronizedl來同步類中的靜態(tài)方法和非靜態(tài)方法。
第一種:非靜態(tài)方法的同步
從java相關語法可以知道使用synchronized關鍵字來定義方法就會鎖定類中所用使用synchroniezd關鍵字定義的靜態(tài)方法和非靜態(tài)方法,但是這有點不好理解,如果要synchronized塊,來達到這樣的效果,就不難理解為什么會產生這種效果了,如果使用synchronized來鎖定類中所有的同步非靜態(tài)方法,只需要使用this作為synchronized塊的參數傳入synchronized塊中,代碼如下:
public class Test
{
public void method1()
{
synchronized(this)
{
}
}
public synchronized void method2()
{
}
}
public class Test
{
public void method1()
{
synchronized(this)
{
}
}
public synchronized void method2()
{
}
}
在內類中使用synchronized塊中,this只表示內類,和外類(OuterClass)沒有關系。但是內類中的非靜態(tài)方法和外類的非靜態(tài)方法也可以同步。如果在內類中加個方法method3也可以使和Test里面的2個方法同步,代碼如下:
public class Test
{
class InnerClass
{
public void method3()
{
synchronized(Test.this){
}
}
}
}
public class Test
{
class InnerClass
{
public void method3()
{
synchronized(Test.this){
}
}
}
}
上面InnerClass的method3方法與Test的method1和method2方法在同一時間內只能有一個方法執(zhí)行。
synchronized塊不管是正確執(zhí)行完,還是因為程序出錯因異常退出synchronized塊,當前的synchronized塊所持有的同步鎖都會自動釋放,因此在使用synchronized塊不必擔心同步鎖的問題。
二、靜態(tài)方法的同步
由于在調用靜態(tài)方法時,對象實例不一定被創(chuàng)建,因此,就不能使用this來同步靜態(tài)方法,而必須使用Class對象來同步靜態(tài)方法。代碼如下:
public class Test{
pubic static void method1(){
synchronized(Test.class){
}
}
public static synchronized void method2(){
}
}
public class Test{
pubic static void method1(){
synchronized(Test.class){
}
}
public static synchronized void method2(){
}
}
在同步靜態(tài)方法時可以使用類的靜態(tài)字段class來得到class對象,在上例中method1和method2方法只有一個方法執(zhí)行,除了使用class字段可以得到class對象,還可以通過實例的getClass()方法獲取class對象,代碼如下:
public class Test{
public static Test test;
public Test(){
test=this;
}
public static void method1(){
synchronized(test.getClass()){
}
}
}
public class Test{
public static Test test;
public Test(){
test=this;
}
public static void method1(){
synchronized(test.getClass()){
}
}
}
在上面的代碼中,我們通過一個public的靜態(tài)對象得到Test的一個實例,并通過這個實例的getClass方法獲取一個class對象(注意一個類的所有實例通過getClass方法得到的都是同一個Class對象)。我們也可以通過class使不同類的靜態(tài)方法同步,代碼如下:
public class Test1{
public static void method1(){
synchronized(Test.class){
}
}
}
public class Test1{
public static void method1(){
synchronized(Test.class){
}
}
}
注意:在使用synchronized塊來同步方法時,非靜態(tài)方法可以通過this來同步,而靜態(tài)方法必須使用class對象來同步,但是非靜態(tài)方法也可以通過使用class來同步靜態(tài)方法。但是靜態(tài)方法中不能使用this來同步非靜態(tài)方法。這點在使用synchronized塊需要注意。
Note
synchronized關鍵字會降低應用程序的性能,因此只能在并發(fā)情景中需要修改共享數據的方法上使用它。如果多個線程訪問同一個synchronized方法,則只有一個線程可以訪問,其他線程將等待。如果方法聲明沒有使用synchronized關鍵字,所有的線程都能在同一時間執(zhí)行這個方法,因而總運行時間降低。如果已知一個方法不會被一個以上線程調用,則無需使用synchronized關鍵字聲明之。
可以遞歸調用被synchronized聲明的方法。當線程訪問一個對象的同步方法時,它還可以調用這個對象的其他的同步方法,也包含正在執(zhí)行的方法,而不必再次去獲取這個方法的訪問權。
我們可以通過synchronized關鍵字來保護代碼塊(而不是整個方法)的訪問。應該這樣利用synchronized關鍵字:方法的其余部分保持在synchronized代碼塊之外,以獲取更好的性能。臨界區(qū)(即同一時間只能被一個線程訪問的代碼塊)的訪問應該盡可能的短。例如在獲取一幢樓人數的操作中,我們只使用synchronized關鍵字來保護對人數更新的指令,并讓其他操作不使用共享數據。當這樣使用synchronized關鍵字時,必須把對象引用作為傳入參數。同一時間只有一個線程被允許訪問這個synchronized代碼。通常來說,我們使用this關鍵字來引用正在執(zhí)行的方法所屬的對象:
synchronized(this){
//Java code
}
- 深入理解java中的synchronized關鍵字
- 詳解Java中synchronized關鍵字的死鎖和內存占用問題
- java多線程編程之使用Synchronized關鍵字同步類方法
- Java關鍵字volatile和synchronized作用和區(qū)別
- 舉例講解Java中synchronized關鍵字的用法
- 實例解析Java中的synchronized關鍵字與線程安全問題
- 深入了解Java中Synchronized關鍵字的實現原理
- java基本教程之synchronized關鍵字 java多線程教程
- java synchronized關鍵字的用法
- Java中synchronized關鍵字鎖的力度與位置示例詳解
相關文章
深入淺析Mybatis與Hibernate的區(qū)別與用途
這篇文章主要介紹了Mybatis與Hibernate的區(qū)別與用途的相關資料,需要的朋友可以參考下2017-10-10
Java?報錯?java.util.ConcurrentModificationException:?null?
這篇文章主要介紹了Java?報錯?java.util.ConcurrentModificationException:?null?的原因和解決方案,這個異常通常在多線程環(huán)境下出現,意味著在迭代過程中,集合或者映射的結構發(fā)生了變化,本文分享完美解決方案,需要的朋友可以參考下2023-07-07

