摘要:前言面試,必然要被問(wèn)內(nèi)存模型和并發(fā)開(kāi)發(fā)。但是線程池啟動(dòng)多線程,是并發(fā)執(zhí)行的。線程線程同時(shí)對(duì)值為的變量進(jìn)行操作,結(jié)果返回,而不是。要不簡(jiǎn)單點(diǎn),記住多線程對(duì)全局變量的寫操作會(huì)發(fā)生沖突。
前言
面試Java,必然要被問(wèn)Java內(nèi)存模型和Java并發(fā)開(kāi)發(fā)。我被問(wèn)到的時(shí)候,心里慌得一批,“額,是在《Thinking in Java》里面寫的嗎?果然每天增刪改太low了”
要了解這些圖嗎? 我希望能解釋的再簡(jiǎn)單一些,以上都不用public class Example1 { public static int count = 0; public static int clientTotal = 5000; public static void main(String[] args) throws Exception { ExecutorService executorService = Executors.newCachedThreadPool(); for (int i = 0; i < clientTotal ; i++) { executorService.execute(() -> { try { add(); } catch (Exception e) { log.error("exception", e); } }); } } private static void add() { count++; } }
如果上面代碼執(zhí)行,count的值是多少?(為了說(shuō)明重點(diǎn)問(wèn)題,沒(méi)有寫最后打印的代碼)
5000?多次運(yùn)行的結(jié)果,count的值是小于5000的。
解釋一下上面的程序,首先定義了一個(gè)線程池,啟動(dòng)5000個(gè)線程執(zhí)行add()操作,add函數(shù)處理靜態(tài)成員變量count。
如果程序順序調(diào)用,count的值應(yīng)該是5000。
for(int i=0;i<5000;i++){ add(); }
但是線程池啟動(dòng)多線程,是并發(fā)執(zhí)行的。每個(gè)線程啟動(dòng)之后,不管是否運(yùn)行結(jié)束,下一個(gè)線程會(huì)馬上啟動(dòng)。
當(dāng)多個(gè)線程對(duì)同一個(gè)變量add進(jìn)行操作的時(shí)候,就會(huì)發(fā)生寫寫沖突。
線程1、線程2 同時(shí)對(duì)值為0的變量進(jìn)行操作,結(jié)果返回1,而不是2。如果這個(gè)地方想不明白,就請(qǐng)留言,或者看看文章頂部那些原理圖。
public class CountExample2 { // 請(qǐng)求總數(shù) public static int clientTotal = 5000; // 同時(shí)并發(fā)執(zhí)行的線程數(shù) public static int threadTotal = 200; public static AtomicInteger count = new AtomicInteger(0); public static void main(String[] args) throws Exception { ExecutorService executorService = Executors.newCachedThreadPool(); final Semaphore semaphore = new Semaphore(threadTotal); final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); for (int i = 0; i < clientTotal ; i++) { executorService.execute(() -> { try { semaphore.acquire(); add(); semaphore.release(); } catch (Exception e) { log.error("exception", e); } countDownLatch.countDown(); }); } countDownLatch.await(); executorService.shutdown(); log.info("count:{}", count.get()); } private static void add() { count.incrementAndGet(); // count.getAndIncrement(); } }
注,上面的代碼用了生成者消費(fèi)者模式,5000個(gè)生產(chǎn)者,200個(gè)消費(fèi)者,對(duì)程序并發(fā)做一定限制,防止5000個(gè)線程卡死計(jì)算機(jī)。
內(nèi)存模型,也說(shuō)點(diǎn)簡(jiǎn)單的棧(heap),函數(shù)加載的時(shí)候,為函數(shù)內(nèi)部變量分配的空間。和父函數(shù)的內(nèi)部變量和運(yùn)行指針共享同一塊區(qū)域。
函數(shù)運(yùn)行時(shí),new的空間,都是放在堆中的。
這個(gè)就是C的內(nèi)存模型,做shellcode的基礎(chǔ)知識(shí)。
作者:白頭雁
鏈接:https://www.jianshu.com/p/8cb...
終于,我還是碼造一個(gè)了中國(guó)地圖
SVG前戲—讓你的View多姿多彩
分享幾個(gè)Android很強(qiáng)勢(shì)的的開(kāi)源框架
(Android)面試題級(jí)答案(精選版)
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/108131.html
摘要:前言面試,必然要被問(wèn)內(nèi)存模型和并發(fā)開(kāi)發(fā)。但是線程池啟動(dòng)多線程,是并發(fā)執(zhí)行的。線程線程同時(shí)對(duì)值為的變量進(jìn)行操作,結(jié)果返回,而不是。要不簡(jiǎn)單點(diǎn),記住多線程對(duì)全局變量的寫操作會(huì)發(fā)生沖突。 前言 面試Java,必然要被問(wèn)Java內(nèi)存模型和Java并發(fā)開(kāi)發(fā)。我被問(wèn)到的時(shí)候,心里慌得一批,額,是在《Thinking in Java》里面寫的嗎?果然每天增刪改太low了 要了解這些圖嗎? showI...
摘要:今天是年月號(hào),一年一度的情人節(jié)又來(lái)了,今天對(duì)于很多年輕人來(lái)說(shuō)是個(gè)特別的日子。該程序員在聯(lián)誼會(huì)上認(rèn)識(shí)了某藝術(shù)系女生,同鄉(xiāng)的雙方互有好感。 今天是2018年8月17號(hào),一年一度的情人節(jié)又來(lái)了,今天對(duì)于很多年輕人來(lái)說(shuō)是個(gè)特別的日子。但是作為屌絲界的一枚程序員,往往給人的印象是宅,不愛(ài)說(shuō)話,智商不好,也不討好哄女孩子歡心。但是,為什么我身邊的程序員各個(gè)都是高智商,高顏值的小碼農(nóng)呢?或許,他們用...
摘要:使用場(chǎng)景自定義權(quán)限一般用于暴露出去的組件,提高安全性。擁有相同自定義權(quán)限的軟件必須使用同樣的簽名,否則后一個(gè)程序無(wú)法安裝。 最近在研究關(guān)于android自定義權(quán)限的問(wèn)題,關(guān)于自定義權(quán)限一般是保證APP的安全性,那么什么事自定義權(quán)限?今天我們來(lái)補(bǔ)充一下自己的知識(shí) 作者:BrightVan地址:https://www.jianshu.com/p/b60... 1、如何聲明自定義權(quán)限 在M...
閱讀 2113·2021-11-18 10:02
閱讀 2863·2021-09-04 16:41
閱讀 1156·2019-08-30 15:55
閱讀 1420·2019-08-29 17:27
閱讀 1106·2019-08-29 17:12
閱讀 2539·2019-08-29 15:38
閱讀 2864·2019-08-29 13:02
閱讀 2841·2019-08-29 12:29