Java多線(xiàn)程同步Synchronized使用分析
由于servlet 是多線(xiàn)程單例的。
struts1 的前端控制器是單例的,線(xiàn)程不安全的;每次請(qǐng)求都會(huì)用同一actionServlet;
struts2 的action不是單例的,線(xiàn)程安全的;每次請(qǐng)求都會(huì)新new一個(gè)action來(lái)接收;
因此如果使用 servlet 或者strtus1 線(xiàn)程安全問(wèn)題,不能不考慮,下面內(nèi)容。
同步的概念:
同步分為 同步方法 和 同步塊 兩種方式。
鎖定的內(nèi)容分為 鎖定類(lèi)的某個(gè)特定實(shí)例 和 鎖定類(lèi)對(duì)象(類(lèi)的所有實(shí)例)
變量分為 實(shí)例變量(不帶static的變量) 和 類(lèi)變量(帶static的變量)
使用同步的原因
1. 在系統(tǒng)中對(duì)訪(fǎng)類(lèi)要使用多線(xiàn)程進(jìn)行訪(fǎng)問(wèn);
2. 在該類(lèi)中有 類(lèi)變量, 或者是 在類(lèi)的方法中有訪(fǎng)問(wèn) 公共資源(如一個(gè)外部文件的讀寫(xiě))。
同步鎖鎖定的內(nèi)容是什么?
無(wú)論你將Synchronized加在方法前還是加在一個(gè)變量前,其鎖定的都是一個(gè) 類(lèi)對(duì)象。 每一個(gè)對(duì)象都只有一個(gè)鎖與之相關(guān)聯(lián)。
下例中分情況的列舉各種情況下的同步效果
1. Synchronized 加在方法上, (同步方法,鎖定類(lèi)實(shí)例)
Java代碼
- public class Demo1 {
- public synchronized void m1(){
- //...............
- }
- public void m2(){
- //............
- synchronized(this){
- //.........
- }
- //........
- }
- }
這兩種寫(xiě)法的效果是一樣的,鎖定的都是類(lèi)實(shí)例對(duì)象。如果有一個(gè) 類(lèi)實(shí)例對(duì)象: demo = new Demo1(),另外有兩個(gè)線(xiàn)程: thread1,thread2,都調(diào)用了demo 對(duì)象,那么,在同一時(shí)間,如果 thread1調(diào)用了demo.m1(),則thread2在該時(shí)間內(nèi)不能訪(fǎng)問(wèn)demo.m1() 和 demo.m2(); 因?yàn)閠hread1把demo這個(gè)對(duì)象的鎖使用了,所以無(wú)法分給其它線(xiàn)程使用
但是,如果thread1調(diào)用 demo1.m1(), thread2調(diào)用 demo2.m1(), 則可以同時(shí)進(jìn)行,因?yàn)樗鼈冋{(diào)用的是不同的Demo1類(lèi)對(duì)象實(shí)例。
2. Synchronized 加在變量上, (同步塊,鎖定類(lèi)實(shí)例)
Java代碼
- public class Demo2 {
- Object a = new Object();
- Object b = new Object();
- public void m1(){
- //............
- synchronized(a){
- //.........
- }
- //........
- }
- public void m2(){
- //............
- synchronized(b){
- //.........
- }
- //........
- }
- }
這種情況下,是實(shí)現(xiàn)代碼塊鎖定,鎖定的對(duì)象是 變量 a 或 b; (注意,a 、b 都是非static 的)如果有一個(gè) 類(lèi)實(shí)例對(duì)象: demo = new Demo2(),另外有兩個(gè)線(xiàn)程: thread1,thread2,都調(diào)用了demo 對(duì)象,那么,在同一時(shí)間,如果 thread1調(diào)用了demo.m1(),則thread2在該時(shí)間內(nèi)可以訪(fǎng)問(wèn)demo.m2();但不能訪(fǎng)問(wèn) demo.m1() 的同步塊, 因?yàn)閍被 thread1鎖定了。
3. Synchronized 鎖定的是 類(lèi)變量 ,即static 變量(可能是屬性,可能是方法)(鎖定類(lèi)對(duì)象)
Java代碼
- public class Demo3 {
- static Object o = new Object();
- public static synchronized void m1() {
- //....
- }
- public static void m2() {
- //...
- synchronized (Demo3.class) {
- //.....
- }
- //.....
- }
- public static void m3() {
- //..........
- try {
- synchronized (Class.forName("Demo3")) {
- //............
- }
- } catch (ClassNotFoundException ex) {
- }
- //.............
- }
- public static void m4() {
- //............
- synchronized(o){
- //........
- }
- //..........
- }
- }
以上4個(gè)方法中實(shí)現(xiàn)的效果都是一樣的,其鎖定的對(duì)象都是類(lèi)Demo3,而不是類(lèi)實(shí)例對(duì)象 ,即在多線(xiàn)程中,其共享的資源是屬于類(lèi)的,而不是屬于類(lèi)對(duì)象的。在這種情況下,如果thread1 訪(fǎng)問(wèn)了這4個(gè)方法中的任何一個(gè), 在同一時(shí)間內(nèi)其它的線(xiàn)程都不能訪(fǎng)問(wèn) 這4個(gè)方法。
4. 類(lèi)的方法中訪(fǎng)問(wèn)了多線(xiàn)程共同的資源, 且該資源是可變的,這種情況下也是需要進(jìn)行同步的
Java代碼
- public class Demo4 {
- static String path = "file path";
- public void readConfiFile() {
- synchronized (path) {
- // 讀取該path指定的文件。
- }
- }
- public void writeConfiFile() {
- synchronized (path) {
- //寫(xiě)信息到該path指定的文件。
- }
- }
- }
這種情況下,必須鎖定為 類(lèi)變量,而不能進(jìn)行鎖定類(lèi)實(shí)例對(duì)象,因?yàn)檫@是變象的一種類(lèi)資源共享,而不是類(lèi)實(shí)例對(duì)象資源共享。
線(xiàn)程,成也其,敗也其,用好了可以提升性能,用不好則會(huì)使系統(tǒng)后患無(wú)窮。
PS: 進(jìn)行線(xiàn)程同步需要很大的系統(tǒng)開(kāi)銷(xiāo), 所以,在使用時(shí),如果不是必須的,則盡量不使用同步功能。
【編輯推薦】
Java多線(xiàn)程編程實(shí)戰(zhàn)精要