您好,登錄后才能下訂單哦!
這篇文章主要介紹怎么實(shí)現(xiàn)Java線程安全問題,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!
這篇文章主要介紹了如何實(shí)現(xiàn)Java線程安全問題,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
解決線程安全問題的第一種方案:使用同步代碼塊
格式:
synchronized(鎖對象) {
可能會(huì)出現(xiàn)線程安全問題的代碼(訪問了共享數(shù)據(jù)的代碼)
}
注意:代碼塊中的鎖對象,可以是任意對象,但必須保證多個(gè)線程之間使用的是同一個(gè)
鎖對象的作用是把同步代碼塊鎖住,同一時(shí)間只能讓一個(gè)線程在同步代碼塊中執(zhí)行
package com.fgy.demo02;/** * 實(shí)現(xiàn)賣票案例 */public class RunnableImpl implements Runnable { private int ticket = 100; Object obj = new Object(); @Override public void run() { while (true) { synchronized (obj) { if (ticket > 0) { try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + "正在買第" + ticket + "張票"); ticket--; } } } }}
package com.fgy.demo02;public class Demo01Ticket { public static void main(String[] args) { RunnableImpl run = new RunnableImpl(); new Thread(run).start(); new Thread(run).start(); new Thread(run).start(); }}
解決線程安全問題的第二種方案:使用同步方法
使用步驟:
1.把訪問了共享數(shù)據(jù)的代碼抽取出來,放到一個(gè)方法中
2.在方法上添加synchronized修飾符
格式:
修飾符 synchronized 返回值類型 方法名(...) {
可能會(huì)出現(xiàn)線程安全問題的代碼(訪問了共享數(shù)據(jù)的代碼)
}
同步方法的鎖對象是:this
靜態(tài)同步方法的鎖對象不能是this,因?yàn)閠his是創(chuàng)建對象后產(chǎn)生的,靜態(tài)方法優(yōu)先于對象
靜態(tài)方法的鎖對象是本類的class文件對象
package com.fgy.demo03;/** * 實(shí)現(xiàn)賣票案例 */public class RunnableImpl implements Runnable { private int ticket = 100; @Override public void run() { while (true) { payTicket(); } } public synchronized void payTicket() { if (ticket > 0) { try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + "正在買第" + ticket + "張票"); ticket--; } }}
解決線程安全問題的第三種方案:使用lock鎖
使用步驟:
1.在成員位置創(chuàng)建ReenterantLock對象
2.在可能出現(xiàn)安全問題的代碼前調(diào)用Lock接口中的方法lock()獲取鎖
3.在可能出現(xiàn)安全問題的代碼后調(diào)用Lock接口中的方法unlock()釋放鎖
package com.fgy.demo04;import java.util.concurrent.locks.Lock;import java.util.concurrent.locks.ReentrantLock;/** * 實(shí)現(xiàn)賣票案例 */public class RunnableImpl implements Runnable { private int ticket = 100; Lock l = new ReentrantLock(); /*@Override public void run() { while (true) { l.lock(); if (ticket > 0) { try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + "正在買第" + ticket + "張票"); ticket--; } l.unlock(); } }*/ @Override public void run() { while (true) { l.lock(); if (ticket > 0) { try { Thread.sleep(100); System.out.println(Thread.currentThread().getName() + "正在買第" + ticket + "張票"); ticket--; } catch (InterruptedException e) { e.printStackTrace(); } finally { // 無論程序是否發(fā)生異常都會(huì)釋放鎖 l.unlock(); } } } }}
以上是“怎么實(shí)現(xiàn)Java線程安全問題”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。