java 同步鎖(synchronized)

yilip發表於2014-12-24

java 同步鎖(synchronized)

       在java中,Synchronized就是一把鎖,他可以鎖定一個方法,也可以鎖定一個方法,我擦,其實這兩個東西就是一樣的。塊不就是一個沒有名字的方法麼,方法就是一個有名字的塊。本文就用塊來測試。所謂鎖,就是原子操作,把這個鎖定的塊作為一個整體,就像你上廁所,拉了就要擦屁屁,當然你也可以不擦,如果你不在意出現的問題的話。訊號量Semaphore和這個Synchronized 其實實現的功能差不多,不過效率不同,使用的方式也不同。Synchronized使用簡單一點。

     java中cpu分給每個執行緒的時間片是隨機的並且在java中好多都是多個執行緒共用一個資源,比如作為屌絲的我們,寢室四個人被安排在今晚(平安夜)給30個女神發蘋果,一共有30個蘋果,我們四個要將這10個蘋果發出去。這10個蘋果就是我們共享的資源。   

   1.不考慮資源共享:

<span style="font-size:12px;">public class RunnableTest implements Runnable
{
	// 一共有10個蘋果
	private int apple = 10;
	@Override
	public void run()
	{
		try
		{
			for (int i = 0; i < 10; i++)
			{			
//				synchronized (this)
//				{
					
					if (apple > 0)
					{						
						System.out.println(Thread.currentThread().getName()
								+ ":發第 " + apple-- + "個蘋果");
					}
				}
				Thread.sleep(1000);
			//}
		}
		catch (Exception e)
		{
		}
	}

	public static void main(String[] args)
	{
		// 四個人開始送蘋果
		RunnableTest runnableTest1 = new RunnableTest();
		// RunnableTest runnableTest2 = new RunnableTest();
		// RunnableTest runnableTest3 = new RunnableTest();
		(new Thread(runnableTest1, "李一")).start();
		(new Thread(runnableTest1, "沈二")).start();
		(new Thread(runnableTest1, "汪三")).start();
		(new Thread(runnableTest1, "遊四")).start();
	}

}</span>

     看到沒,第6個蘋果應該很大,這幾個傢伙都想搶著送女神。但是這是不符合實際的,只可能有一個可以送的出去6號蘋果。所以,沒有考慮同步,這是錯誤的


  2.考慮同步   

    將上面的註釋取消掉:

    

     考慮同步後,這個大蘋果還是由我一個人送給了女神。

   但是,這個問題由於sleep和Synchronized的位置不同,出現了很多不一樣的結果,其實也是值得研究一下的:

    情形1:

   

<span style="font-size:12px;">	public void run()
	{
		try
		{
			for (int i = 0; i < 10; i++)
			{	
				
				synchronized (this)
				{
					Thread.sleep(1000);
					if (apple > 0)
					{						
						System.out.println(Thread.currentThread().getName()
								+ ":發第 " + apple-- + "個蘋果");
					}
				}			
			}
		}
		catch (Exception e)
		{
		}
	}</span>
結果:

  

    全是我一個人發蘋果,唉,蛋疼啊。仔細考慮一下,這個sleep寫其實和不寫是一樣的,你睡了,其他哥們也在睡覺。

情形2:

  

<span style="font-size:12px;">	public void run()
	{
		try
		{
			synchronized (this)
			{
				for (int i = 0; i < 10; i++)
				{
					Thread.sleep(1000);

					if (apple > 0)
					{
						System.out.println(Thread.currentThread().getName()
								+ ":發第 " + apple-- + "個蘋果");
					}
				}
			}
		}
		catch (Exception e)
		{
		}
	}</span>

  這和情形1差不多,繼續疼。   因為鎖的就是讓我一個人把10個蘋果發完。    

情形3

<span style="font-size:12px;">	public void run()
	{
		try
		{
			for (int i = 0; i < 10; i++)
			{
				Thread.sleep(1000);
				if (apple > 0)
				{
					synchronized (this)
					{
						System.out.println(Thread.currentThread().getName()
								+ ":發第 " + apple-- + "個蘋果");
					}
				}
			}
		}
		catch (Exception e)
		{
		}
	}</span>

  結果:

  

   你 看,都沒蘋果了。還有倆色狼還一直髮蘋果。這是因為當李一把蘋果發完最後一個蘋果時,那倆貨還以為有一個蘋果(已經判斷了apple>0了),所以就繼續去發,結果把自己的蘋果獻給了妹子。

  情形4

   

<span style="font-size:12px;">	public static void main(String[] args)
	{
		// 四個人開始送蘋果
		RunnableTest runnableTest1 = new RunnableTest();
		 RunnableTest runnableTest2 = new RunnableTest();
		 RunnableTest runnableTest3 = new RunnableTest();
		(new Thread(runnableTest1, "李一")).start();
		(new Thread(runnableTest1, "沈二")).start();
		(new Thread(runnableTest1, "汪三")).start();
		(new Thread(runnableTest1, "遊四")).start();
	}</span>
    結果:

    

      四個人 都想一個人發10個蘋果,這果然是一寢室的狼。這是個最明顯的錯誤了,雖然我找了半天才發現問題所在。

    之所以有鎖,要明白鎖的是什麼東西?鎖的是一個物件下的資源,情形4是由三個物件,無法加鎖。

相關文章