你好我是辰兮,很高興你能來閱讀,本篇文章爲大家講解Java多線程之synchronized關鍵詞,下面有案例的截圖和相關代碼可以自行實踐,相關的更多面試知識已經提前整理好文章可以閱讀學習,分享獲取新知,希望對Java初學者有幫助。
1.Java線程的實現方式:
Java線程的實現方式詳解2.線程進程基礎知識參考:
進程線程的面試問題小結3.線程面試常見問題參考:
Java常見多線程基礎面試問題
一、序言
多線程訪問臨界資源時的數據安全問題
產生原因:有多個線程在同時訪問一個資源,如果一個線程在取值的過程中,時間片又被其他線程搶走了,臨界資源問題就產生了
how to解決臨界資源問題
解決方案:一個線程在訪問臨界資源的時候,如果給這個資源“上一把鎖”,這個時候如果其他線程也要訪問這個資源, 就得在“鎖”外面等待
鎖
對象鎖:任意的對象都可以被當做鎖來使用
類鎖:把一個類當做鎖,語法爲:類名.class
同步代碼塊
語法:
synchronized(鎖) {
//需要訪問臨界資源的代碼段
}
說明:
- a.程序走到代碼段中,就用鎖來鎖住了臨界資源,這個時候,其他線程不能執行代碼段中的代碼,只能在鎖外邊等待
- b.執行完代碼段中的這段代碼,會自動解鎖。然後剩下的其他線程開始爭搶cpu時間片
- c.一定要保證不同的線程看到的是同一把鎖,否則同步代碼塊沒有意義.
synchronized是Java中的關鍵字,是一種同步鎖。它修飾的對象有以下幾種:
- 修飾一個代碼塊,被修飾的代碼塊稱爲同步語句塊,其作用的範圍是大括號{}括起來的代碼,作用的對象是調用這個代碼塊的對象;
- 修飾一個方法,被修飾的方法稱爲同步方法,其作用的範圍是整個方法,作用的對象是調用這個方法的對象;
- 修改一個靜態的方法,其作用的範圍是整個靜態方法,作用的對象是這個類的所有對象;
- 修改一個類,其作用的範圍是synchronized後面括號括起來的部分,作用主的對象是這個類的所有對象。
二、同步代碼塊和對象鎖的使用
在Java中,synchronized關鍵字是用來控制線程同步的,就是在多線程的環境下,控制synchronized代碼段不被多個線程同時執行。
同步代碼塊結合對象鎖
參考代碼如下,大家自行運行實踐(ps:可以去除 synchronized關鍵字)
package demo1;
public class SellTickets {
static int count = 10;
//任何對象都可以充當一個對象鎖
static Object obj = new Object();
static Runnable r = new Runnable() {
@Override
public void run() {
while(count > 0) {
System.out.println(Thread.currentThread().getName()+"--");
synchronized(obj){
count--;
if(count <= 0) {
return;
}
System.out.println(Thread.currentThread().getName() + "售出了 一張票,剩餘" + count);
}
}
}
};
public static void main(String[] args) {
Thread t1 = new Thread(r);
Thread t2 = new Thread(r);
Thread t3 = new Thread(r);
// t1.setPriority(8);
t1.start();
t2.start();
t3.start();
}
}
三、同步代碼塊和類鎖的使用
使用類鎖的案例如下,參考格式 即 類.class
四、 同步方法
首先給大家展示沒有加鎖的方法
然後再給大家展示加鎖的方法
參考代碼如下,大家自行運行實踐(ps:可以去除 synchronized關鍵字)
package demo1;
public class SellTickets {
static int count = 100;
//任何對象都可以充當一個對象鎖
//static Object obj = new Object();
static Runnable r = new Runnable() {
@Override
public void run() {
while(count > 0) {
sellTickets();
}
}
//同步方法,作用和同步代碼塊一樣
public synchronized void sellTickets() {
if (count <= 0) {
return;
}
count--;
System.out.println("售票員" + Thread.currentThread().getName() + "售出一張票,餘額 爲" + count);
}
};
public static void main(String[] args) {
Thread t1 = new Thread(r);
Thread t2 = new Thread(r);
Thread t3 = new Thread(r);
// t1.setPriority(8);
t1.start();
t2.start();
t3.start();
}
}
The best investment is to invest in yourself
2020.05.31 記錄辰兮的第72篇博客