Java語言的關(guān)鍵字,當(dāng)它用來修飾一個(gè)方法或者一個(gè)代碼塊的時(shí)候,能夠保證在同一時(shí)刻最多只有一個(gè)線程執(zhí)行該段代碼。
創(chuàng)新互聯(lián)專注于贊皇網(wǎng)站建設(shè)服務(wù)及定制,我們擁有豐富的企業(yè)做網(wǎng)站經(jīng)驗(yàn)。 熱誠為您提供贊皇營(yíng)銷型網(wǎng)站建設(shè),贊皇網(wǎng)站制作、贊皇網(wǎng)頁設(shè)計(jì)、贊皇網(wǎng)站官網(wǎng)定制、小程序開發(fā)服務(wù),打造贊皇網(wǎng)絡(luò)公司原創(chuàng)品牌,更為您提供贊皇網(wǎng)站排名全網(wǎng)營(yíng)銷落地服務(wù)。
一、當(dāng)兩個(gè)并發(fā)線程訪問同一個(gè)對(duì)象object中的這個(gè)synchronized(this)同步代碼塊時(shí),一個(gè)時(shí)間內(nèi)只能有一個(gè)線程得到執(zhí)行。另一個(gè)線程必須等待當(dāng)前線程執(zhí)行完這個(gè)代碼塊以后才能執(zhí)行該代碼塊。
二、然而,當(dāng)一個(gè)線程訪問object的一個(gè)synchronized(this)同步代碼塊時(shí),另一個(gè)線程仍然可以訪問該object中的非synchronized(this)同步代碼塊。
三、尤其關(guān)鍵的是,當(dāng)一個(gè)線程訪問object的一個(gè)synchronized(this)同步代碼塊時(shí),其他線程對(duì)object中所有其它synchronized(this)同步代碼塊的訪問將被阻塞。
四、第三個(gè)例子同樣適用其它同步代碼塊。也就是說,當(dāng)一個(gè)線程訪問object的一個(gè)synchronized(this)同步代碼塊時(shí),它就獲得了這個(gè)object的對(duì)象鎖。結(jié)果,其它線程對(duì)該object對(duì)象所有同步代碼部分的訪問都被暫時(shí)阻塞。
五、以上規(guī)則對(duì)其它對(duì)象鎖同樣適用.
1.把 synchronized (i) { 放在 while外面就沒問題的
2.同步代碼塊的原則之一是:盡量只包計(jì)算的部分。原則之二:計(jì)量要盡量避免其中包了sleep或者yield方法。sleep是為while服務(wù)的,如果把同步放在while之內(nèi),同步把sleep給包括了,那么sleep就沒有意義了。
我覺得這樣寫最好:
public void run() {
while (true) {
try {
Thread.sleep(1);
} catch (Exception e) {
e.printStackTrace();
}
synchronized (i) {
if (i 0) {
System.out.println(Thread.currentThread().getName()
+ " == " + i--);
}
}
}
}
在Java語言中,每一個(gè)對(duì)象有一把鎖。線程可以使用synchronized關(guān)鍵字來獲取對(duì)象上的鎖。synchronized關(guān)鍵字可應(yīng)用在方法級(jí)別(粗粒度鎖)或者是代碼塊級(jí)別(細(xì)粒度鎖)。
問題的由來:
看到這樣一個(gè)面試題:
?
1
2
3
4
5
6
//下列兩個(gè)方法有什么區(qū)別
public synchronized void method1(){}
public void method2(){
synchronized (obj){}
}
synchronized用于解決同步問題,當(dāng)有多條線程同時(shí)訪問共享數(shù)據(jù)時(shí),如果進(jìn)行同步,就會(huì)發(fā)生錯(cuò)誤,Java提供的解決方案是:只要將操作共享數(shù)據(jù)的語句在某一時(shí)段讓一個(gè)線程執(zhí)行完,在執(zhí)行過程中,其他線程不能進(jìn)來執(zhí)行可以。解決這個(gè)問題。這里在用synchronized時(shí)會(huì)有兩種方式,一種是上面的同步方法,即用synchronized來修飾方法,另一種是提供的同步代碼塊。
這里總感覺怪怪的,這兩種方法有什么區(qū)別呢,基礎(chǔ)學(xué)得不好,于是就動(dòng)手做了個(gè)簡(jiǎn)單的測(cè)試,代碼如下:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public class SynObj {
public synchronized void methodA() {
System.out.println("methodA.....");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void methodB() {
synchronized(this) {
System.out.pritntln("methodB.....");
}
}
public void methodC() {
String str = "sss";
synchronized (str) {
System.out.println( "methodC.....");
}
}
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public class TestSyn {
public static void main(String[] args) {
final SynObj obj = new SynObj();
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
obj.methodA();
}
});
t1.start();
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
obj.methodB();
}
});
t2.start();
Thread t3 = new Thread(new Runnable() {
@Override
public void run() {
obj.methodC();
}
});
t3.start();
}
}
這段小代碼片段打印結(jié)果如下:
?
1
2
3
4
methodA.....
methodC.....
//methodB會(huì)隔一段時(shí)間才會(huì)打印出來
methodB.....
這段代碼的打印結(jié)果是,methodA…..methodC…..會(huì)很快打印出來,methodB…..會(huì)隔一段時(shí)間才打印出來,那么methodB為什么不能像methodC那樣很快被調(diào)用呢?
在啟動(dòng)線程1調(diào)用方法A后,接著會(huì)讓線程1休眠5秒鐘,這時(shí)會(huì)調(diào)用方法C,注意到方法C這里用synchronized進(jìn)行加鎖,這里鎖的對(duì)象是str這個(gè)字符串對(duì)象。但是方法B則不同,是用當(dāng)前對(duì)象this進(jìn)行加鎖,注意到方法A直接在方法上加synchronized,這個(gè)加鎖的對(duì)象是什么呢?顯然,這兩個(gè)方法用的是一把鎖。
*由這樣的結(jié)果,我們就知道這樣同步方法是用什么加鎖的了,由于線程1在休眠,這時(shí)鎖還沒釋放,導(dǎo)致線程2只有在5秒之后才能調(diào)用方法B,由此,可知兩種加鎖機(jī)制用的是同一個(gè)鎖對(duì)象,即當(dāng)前對(duì)象。
另外,同步方法直接在方法上加synchronized實(shí)現(xiàn)加鎖,同步代碼塊則在方法內(nèi)部加鎖,很明顯,同步方法鎖的范圍比較大,而同步代碼塊范圍要小點(diǎn),一般同步的范圍越大,性能就越差,一般需要加鎖進(jìn)行同步的時(shí)候,肯定是范圍越小越好,這樣性能更好。
synchronized(這里的對(duì)象你看成一道門) {
這里是一個(gè)人進(jìn)來了,把門反鎖了
需要同步操作的代碼
這里是里面的人事情做完了,出去了,門開著其他人可以進(jìn)了
}
至于怎么鎖的,這是java和jvm的規(guī)定和實(shí)現(xiàn)細(xì)節(jié),作為普通程序員沒必要深入那么多。