版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
1、<p> 操作系統(tǒng)課程設(shè)計(jì)報(bào)告</p><p><b> 目錄</b></p><p><b> 目錄2</b></p><p><b> 一、 前期工作3</b></p><p><b> 1.1平臺(tái)搭建3</b></p&
2、gt;<p><b> 二、 代碼分工3</b></p><p><b> 三、設(shè)計(jì)及實(shí)現(xiàn)4</b></p><p> 3.1 Task1.1 KThread.join()4</p><p> 3.1.1 要求分析4</p><p> 3.1.2 設(shè)計(jì)方案4<
3、/p><p> 3.1.3 實(shí)現(xiàn)代碼4</p><p> 3.1.4 測(cè)試代碼及結(jié)果5</p><p> 3.2 Task1.2 condition2類7</p><p> 3.2.1 要求分析7</p><p> 3.2.2 設(shè)計(jì)方案8</p><p> 3.2.3 實(shí)現(xiàn)代碼
4、8</p><p> 3.2.4 測(cè)試代碼及結(jié)果8</p><p> 3.3 Task1.3 alram類11</p><p> 3.3.1 要求分析11</p><p> 3.3.2 設(shè)計(jì)方案11</p><p> 3.3.3 實(shí)現(xiàn)代碼12</p><p> 3.3.4
5、 測(cè)試代碼及結(jié)果12</p><p> 3.4 Task1.4 communicator類14</p><p> 3.4.1要求分析14</p><p> 3.4.2 設(shè)計(jì)方案14</p><p> 3.4.3 實(shí)現(xiàn)代碼14</p><p> 3.4.4 測(cè)試代碼及結(jié)果15</p>
6、<p> 3.5 Task1.5 priority scheduler類17</p><p> 3.5.1 要求分析17</p><p> 3.5.2 設(shè)計(jì)方案17</p><p> 3.5.3 實(shí)現(xiàn)代碼18</p><p> 3.5.4 測(cè)試代碼及結(jié)果18</p><p> 3.6 T
7、ask1.6 boat類23</p><p> 3.6.1 要求分析23</p><p> 3.6.2 設(shè)計(jì)方案23</p><p> 3.6.3 實(shí)現(xiàn)代碼23</p><p> 3.6.4 測(cè)試代碼及結(jié)果30</p><p> 3.7 Task2.1 系統(tǒng)調(diào)用creat, open, read,
8、write, close, unlink30</p><p> 3.7.1 要求分析30</p><p> 3.7.2 設(shè)計(jì)方案31</p><p> 3.7.3 實(shí)現(xiàn)代碼32</p><p> 3.7.4 測(cè)試代碼及結(jié)果34</p><p> 3.8 Task2.2 多進(jìn)程內(nèi)存分配及訪問35&l
9、t;/p><p> 3.8.1 要求分析35</p><p> 3.8.2 設(shè)計(jì)方案35</p><p> 3.8.3 實(shí)現(xiàn)代碼36</p><p> 3.8.4 測(cè)試代碼及結(jié)果39</p><p> 3.9 Task2.3 系統(tǒng)調(diào)用exec, join, exit41</p><p
10、> 3.9.1 要求分析41</p><p> 3.9.2 設(shè)計(jì)方案41</p><p> 3.9.3 實(shí)現(xiàn)代碼42</p><p> 3.9.4 測(cè)試代碼及結(jié)果43</p><p> 3.10 Task2.4 Lottery Schedule類44</p><p> 3.10.1 要求分析
11、44</p><p> 3.10.2 設(shè)計(jì)方案45</p><p> 3.10.3 實(shí)現(xiàn)代碼45</p><p> 3.10.4 測(cè)試代碼及結(jié)果45</p><p><b> 總結(jié)48</b></p><p><b> 前期工作</b></p>
12、<p><b> 1.1平臺(tái)搭建</b></p><p> Nachos For Java</p><p> phrase1部分:IDE環(huán)境可采用Eclipse。</p><p> Phase 2及以后階段:需要把C程序編譯成MIPS二進(jìn)制文件COFF,需要MIPS的C編譯器 mips-x86.linux-xgcc(ub
13、untu12.04平臺(tái)下)。</p><p><b> 代碼分工</b></p><p><b> 三、設(shè)計(jì)及實(shí)現(xiàn)</b></p><p> 3.1 Task1.1 KThread.join()</p><p> 3.1.1 要求分析</p><p> Join()
14、方法的含義:當(dāng)前線程a在運(yùn)行,執(zhí)行b.join(),則a阻塞,直到線程b結(jié)束,a繼續(xù)執(zhí)行。</p><p> Join函數(shù)的作用即為等待某線程運(yùn)行完畢。當(dāng)前線程 (唯一一個(gè)正在運(yùn)行的線程) A調(diào)用另一個(gè)線程 (處于就緒狀態(tài)) B的join函數(shù)時(shí) (A 和 B 在Nachos中均為KThread類型對(duì)象),A被掛起,直到B運(yùn)行結(jié)束后, join函數(shù)返回,A才能繼續(xù)運(yùn)行。注意在一個(gè)KThread對(duì)象上只能調(diào)用一次j
15、oin,且當(dāng)前線程不能對(duì)自身調(diào)用join。</p><p> 3.1.2 設(shè)計(jì)方案</p><p> 為每個(gè)線程創(chuàng)建一個(gè)線程隊(duì)列joinqueue,joinqueue由對(duì)本線程調(diào)用join方法的其他線程對(duì)象構(gòu)成,可“捐贈(zèng)”優(yōu)先級(jí) ,以及布爾形變量joined,判斷本線程是否被其他線程調(diào)用過join方法,以及一個(gè)判斷線程狀態(tài)方法IsAlive()。</p><p>
16、; 假設(shè)當(dāng)前線程A調(diào)用就緒線程B的join函數(shù). 在join函數(shù)中線程A被添加到線程B的線程隊(duì)列joinqueue中,將線程A“休眠”, 線程B得到執(zhí)行,在線程B結(jié)束時(shí)(此時(shí)線程B成為當(dāng)前線程, 發(fā)生在KThead.finish函數(shù)中) 選擇B的joinqueue的nextThread()執(zhí)行, A喚醒并繼續(xù). </p><p> 3.1.3 實(shí)現(xiàn)代碼</p><p> 在KThre
17、ad.java 中: </p><p> private ThreadQueue joinQueue = null; </p><p> private boolean Joined = false;</p><p> public boolean IsAlive(){</p><p> if(this.status==sta
18、tusNew||status==statusReady||</p><p> status==statusRunning||status==statusBlocked)</p><p> return true;</p><p> else return false;</p><p><b> }</b><
19、/p><p> public void join() {</p><p><b> //</b></p><p> if (!this.IsAlive()) return;</p><p> while (IsAlive()) {</p><p><b> //</b>
20、;</p><p> this.joinQueue.acquire(this);</p><p><b> //</b></p><p><b> }</b></p><p> this.joinQueue.waitForAccess(KThread.currentThread());<
21、;/p><p> KThread.sleep();</p><p><b> //</b></p><p><b> }</b></p><p><b> }</b></p><p> public static void finish() {&l
22、t;/p><p><b> ///</b></p><p> if (currentThread.Joined) {</p><p> currentThread.joinQueue.nextThread().ready();</p><p><b> }</b></p><
23、;p><b> ///</b></p><p><b> }</b></p><p> 3.1.4 測(cè)試代碼及結(jié)果</p><p> 創(chuàng)建AThread和Bthread兩個(gè)線程,AThread循環(huán)打印“AThread循環(huán)第..次”語句,Bthread開始打印“B_thread 就緒”語句,中間執(zhí)行AThrea
24、d。Join()方法,最后打印“B_thread執(zhí)行結(jié)束”語句。</p><p><b> 測(cè)試代碼:</b></p><p> package nachos.threads;</p><p> import nachos.machine.*;</p><p> public class KThreadTest
25、{</p><p> public KThreadTest() {</p><p><b> }</b></p><p> public static void simpleJoinTest() {</p><p> KThread A_thread = new KThread(new KThreadTest.A
26、_thread(5));</p><p> KThread B_thread = new KThread(new KThreadTest.B_thread(A_thread));</p><p> B_thread.fork();</p><p> B_thread.join();</p><p><b> }</b&
27、gt;</p><p> public static class B_thread implements Runnable {</p><p> B_thread(KThread joinee) {</p><p> this.joinee = joinee;</p><p><b> }</b></p&g
28、t;<p> public void run() {</p><p> System.out.println("B_thread 就緒");</p><p> System.out.println("Forking and joining A_thread...");</p><p> this.join
29、ee.fork();</p><p> this.joinee.join();</p><p> System.out.println("B_thread 執(zhí)行結(jié)束");</p><p><b> }</b></p><p> private KThread joinee;</p>
30、<p><b> }</b></p><p> public static class A_thread implements Runnable {</p><p> A_thread(int num) {</p><p> this.num = num;</p><p><b> }&
31、lt;/b></p><p> public void run() {</p><p> System.out.println("A_thread 就緒");</p><p> System.out.println("A_thread開始執(zhí)行");</p><p> // This sho
32、uld just kill some cycles</p><p> for (int i = 0; i < this.num; ++i) {</p><p> System.out.println("A_thread 循環(huán) 第" + i + " 次");</p><p> KThread.currentThrea
33、d().yield();</p><p><b> }</b></p><p> System.out.println("A_thread 執(zhí)行結(jié)束");</p><p><b> }</b></p><p> private int num;</p>&l
34、t;p><b> }</b></p><p><b> }</b></p><p><b> 測(cè)試結(jié)果:</b></p><p> 3.2 Task1.2 condition2類</p><p> 3.2.1 要求分析</p><p>
35、 threads.Lock類提供了鎖以保證互斥. 在臨界代碼區(qū)的兩端執(zhí)行Lock.acquire() 和Lock.release() 即可保證同時(shí)只有一個(gè)線程訪問臨界代碼區(qū). 條件變量建立在鎖之上, 由threads.Condition實(shí)現(xiàn), 它是用來保證同步的工具. 每一個(gè)條件變量擁有一個(gè)鎖變量 (該鎖變量亦可被執(zhí)行acquire和release操作, 多個(gè)條件變量可共享同一個(gè)鎖變量). 當(dāng)處于臨界區(qū)內(nèi)的擁有某鎖Lock的當(dāng)前線程對(duì)與
36、鎖Lock聯(lián)系的條件變量執(zhí)行sleep操作時(shí), 該線程失去鎖L并被掛起. 下一個(gè)等待鎖L的線程獲得鎖L (這個(gè)過程由調(diào)度程序完成) 并進(jìn)入臨界區(qū). 當(dāng)擁有鎖L的臨界區(qū)內(nèi)的當(dāng)前線程對(duì)與鎖L聯(lián)系的條件變量執(zhí)行wake操作時(shí) (通常調(diào)用wake之后緊接著就是Lock.release), 等待在該條件變量上的至多一個(gè)被掛起的線程 (由調(diào)用sleep引起) 被重新喚醒并設(shè)置為就緒狀態(tài). 若執(zhí)行wakeall操作, 則等待在該條件變量上的所有被掛起
37、的線程都被喚醒. threads.condition已經(jīng)實(shí)現(xiàn)了一個(gè)條件變量 (采用信號(hào)量實(shí)現(xiàn)), 題目要求用屏蔽/禁止中斷</p><p> 3.2.2 設(shè)計(jì)方案</p><p> 為每個(gè)條件變量添加一個(gè)鎖conditionLock和一個(gè)Kthread對(duì)象組成的等待隊(duì)列waitqueue,condition.sleep 將調(diào)用sleep()的線程加入到等待隊(duì)列,釋放鎖,并阻塞,以便其他
38、線程喚醒它,一旦’喚醒’立即要求鎖,并在sleep函數(shù)開始/結(jié)尾處屏蔽/允許中斷以保證原子性;condition.wake中從等待隊(duì)列中取出線程進(jìn)行ready()實(shí)現(xiàn)喚醒, </p><p> (同樣要在wake函數(shù)開始/結(jié)尾處屏蔽/允許中斷), wakeall函數(shù)的實(shí)現(xiàn)依賴于wake. 只需不斷地wake 直到隊(duì)列為空為止. </p><p> 3.2.3 實(shí)現(xiàn)代碼</p&g
39、t;<p> 見threads/condition2.java </p><p> private Lock conditionLock;</p><p> private LinkedList<KThread> waitQueue;</p><p> public void sleep() {</p><p&g
40、t;<b> //</b></p><p> waitQueue.add(KThread.currentThread());</p><p> conditionLock.release();</p><p> KThread.sleep();</p><p> conditionLock.acquire();
41、</p><p><b> //</b></p><p><b> }</b></p><p> public void wake() {</p><p><b> //</b></p><p> waitQueue.remove().read
42、y();</p><p><b> //</b></p><p><b> }</b></p><p> 3.2.4 測(cè)試代碼及結(jié)果</p><p> 創(chuàng)建共有條件變量c2test,一個(gè)臨界資源count和三個(gè)線程,thread1,thread2,thread3,初始化三個(gè)線程后thread
43、1,thread2調(diào)用c2test.sleep()后開始“睡眠”,thread3調(diào)用c2test.wakeAll()喚醒所有線程,thread1,thead2開始交替訪問臨界資源count并更改其“余額”數(shù)值。</p><p><b> 測(cè)試代碼:</b></p><p> package nachos.threads;</p><p>
44、 import nachos.machine.*;</p><p> public class Condition2Test { </p><p> Lock condlock = new Lock();</p><p> Condition2 c2test = new Condition2(condlock);</p><p> p
45、ublic Condition2Test(){ </p><p><b> }</b></p><p> public void simpleCondition2Test(){ </p><p> System.out.println("\n ****Condition2Test is now executing.***
46、*");</p><p> System.out.println("初始化賬戶余額為10000");</p><p> final MyCount myCount = new MyCount("0000001", 10000); //創(chuàng)建并發(fā)訪問的賬戶</p><p> KThread thread1 = n
47、ew KThread(new Runnable() {</p><p> public void run() {</p><p> new SaveThread("張三", myCount, 2000);</p><p> System.out.println("張三 goes to sleep");</p>
48、<p> condlock.acquire();</p><p> c2test.sleep();</p><p> System.out.println("張三 reacquires lock when woken.");</p><p> condlock.release();</p><p>
49、 System.out.println("張三 is awake!!!");</p><p> myCount.save(2000," 張三");</p><p><b> }</b></p><p><b> });</b></p><p> KTh
50、read thread2 = new KThread(new Runnable() {</p><p> public void run() {</p><p> new SaveThread("李四", myCount, 2000);</p><p> System.out.println("李四 goes to sleep&
51、quot;);</p><p> condlock.acquire();</p><p> c2test.sleep();</p><p> System.out.println("李四 reacquires lock when woken.");</p><p> condlock.release();</
52、p><p> System.out.println("李四 is awake!!!");</p><p> myCount.save(2000," 李四");</p><p><b> }</b></p><p><b> });</b></p&g
53、t;<p> KThread thread3 = new KThread(new Runnable() {</p><p> public void run() {</p><p> System.out.println("thread 3 Waking up the thread");</p><p> condlock.
54、acquire();</p><p> c2test.wakeAll();</p><p> condlock.release();</p><p> System.out.println("張三 and 李四 woke up by wakeAll");</p><p><b> }</b>&
55、lt;/p><p><b> });</b></p><p> thread1.fork();</p><p> thread2.fork();</p><p> thread3.fork();</p><p> thread1.join();</p><p> t
56、hread2.join();</p><p> thread3.join();</p><p> System.out.println("****Condition2Test finished.****\n");</p><p> } </p><p><b> } </b&
57、gt;</p><p> class SaveThread extends KThread { </p><p> private String name; //操作人 </p><p> private MyCount myCount; //賬戶 </p><p> private int
58、 x; //存款金額 </p><p> SaveThread(String name, MyCount myCount, int x) { </p><p> this.name = name; </p><p> this.myCount = myCount; </p><p>
59、 this.x = x; </p><p><b> } </b></p><p> public void run() { </p><p> myCount.save(x, name); </p><p><b> } </b></p><p><b>
60、; } </b></p><p> class MyCount { </p><p> private String id; //賬號(hào) </p><p> private int cash; //賬戶余額 </p>
61、;<p> Lock lock = new Lock();//賬戶鎖 </p><p> Condition2 c2test = new Condition2(lock);</p><p> MyCount(String id, int cash) { </p><p> this.id = id; </p><p
62、> this.cash = cash; </p><p><b> } </b></p><p> public void save(int x, String name) { </p><p> lock.acquire(); //獲取鎖 </p><p> if
63、 (x > 0) { </p><p> cash += x; //存款 </p><p> System.out.println(name + "存款" + x + ",當(dāng)前余額為" + cash); </p><p><b> } </b></p
64、><p> lock.release(); //釋放鎖 </p><p><b> } </b></p><p><b> }</b></p><p><b> 測(cè)試結(jié)果:</b></p><p> 3.3 Task1.3 a
65、lram類</p><p> 3.3.1 要求分析</p><p> 一個(gè)線程調(diào)用waitUntil(x) 后, 它即被掛起. 在當(dāng)前時(shí)鐘走過x個(gè)滴答后, 該線程被重新喚醒. 線程被喚醒后并不一定立即進(jìn)入運(yùn)行態(tài), 只要將其放入就緒隊(duì)列中即可. 另外, 多個(gè)線程可以分別調(diào)用waitUntil函數(shù)。</p><p> 3.3.2 設(shè)計(jì)方案</p>&
66、lt;p> 于Alarm類有關(guān)的是machine.Timer類. 它在大約每500個(gè)時(shí)鐘滴答使調(diào)用回調(diào)函數(shù) (由Timer.setInterruptHandler函數(shù)設(shè)置). 因此, Alarm類的構(gòu)造函數(shù)中首先要設(shè)置該回調(diào)函數(shù)Alarm.timerInterrupt(). </p><p> 為了實(shí)現(xiàn)waitUntil, 需要在Alarm類中實(shí)現(xiàn)一個(gè)隊(duì)列 Alarm.alarmQueue. 隊(duì)列中的每
67、個(gè)項(xiàng)目是 (線程, 喚醒時(shí)間, 相應(yīng)條件變量) 的三元組. 在調(diào)用waitUntil(x) 函數(shù)時(shí), 首先得到關(guān)于</p><p> 以上三元組的信息: (線程: 當(dāng)前線程, 喚醒時(shí)間: 當(dāng)前時(shí)間+x, 新分配的條件變量), 然后將該元組放入隊(duì)列中, 并對(duì)條件變量sleep操作使當(dāng)前線程掛起. 在時(shí)鐘回調(diào)函數(shù)中 (大約每500個(gè)時(shí)鐘間隔調(diào)用一次) 則依次檢查隊(duì)列中的每個(gè)三元組. 如果喚醒時(shí)間大于當(dāng)前時(shí)間, 則將
68、元組移出隊(duì)列并對(duì)元組中的條件變量執(zhí)行wake操作將相應(yīng)線程喚醒. </p><p> 3.3.3 實(shí)現(xiàn)代碼</p><p> public void timerInterrupt() {</p><p><b> //</b></p><p> while(!alarmqueue.isEmpty() &&
69、amp; (alarmqueue.peek().wakeTime <= Machine.timer().getTime())){</p><p> AlarmThread thread = alarmqueue.remove();</p><p> thread.waitThread.ready();</p><p><b> }</b&
70、gt;</p><p><b> // </b></p><p><b> }</b></p><p> public void waitUntil(long x) {</p><p><b> //</b></p><p> Alarm
71、Thread thread = new AlarmThread(KThread.currentThread(), wakeTime,conditionlock);</p><p> alarmqueue.add(thread);</p><p> KThread.sleep();</p><p><b> //</b></p>
72、<p><b> }</b></p><p> private KThread waitThread;</p><p> private Condition timecondition;</p><p> private long wakeTime;</p><p> 3.3.4 測(cè)試代碼及結(jié)果&
73、lt;/p><p> 創(chuàng)建5個(gè)線程,依次“睡眠”20000ticks,線程醒來后各自打印出睡眠開始時(shí)間,要求睡眠時(shí)間,醒來時(shí)間,以及實(shí)際睡眠時(shí)間。</p><p><b> 測(cè)試代碼: </b></p><p> public static void selfTest(int numOfTest){</p><p&g
74、t; Runnable a = new Runnable() {</p><p> public void run() {</p><p> AlarmTestThread();</p><p><b> }</b></p><p><b> };</b></p><p
75、> for(int i=0; i<numOfTest; i++){</p><p> int numOfThreads = 5;</p><p> print("Creating " + numOfThreads + " num of threads");</p><p> for(int j=0; j&l
76、t;numOfThreads; j++){</p><p> KThread thread = new KThread(a);</p><p> thread.setName("thread");</p><p> thread.fork(); }</p><p> ThreadedKernel.alarm.wai
77、tUntil(30000000); </p><p><b> } </b></p><p><b> }</b></p><p> static void AlarmTestThread(){</p><p> long sleepTime =20000;</p><p
78、> long timeBeforeSleep = Machine.timer().getTime();</p><p> ThreadedKernel.alarm.waitUntil(sleepTime);</p><p> long timeAfterSleep = Machine.timer().getTime();</p><p> long a
79、ctualSleepTime = timeAfterSleep - timeBeforeSleep;</p><p> print(KThread.currentThread().toString() + " 被要求在: " + timeBeforeSleep + " 睡眠 " + sleepTime + " ticks");</p>&
80、lt;p> print(KThread.currentThread().toString() + " 被喚醒時(shí)間: " + Machine.timer().getTime() + "要求睡眠時(shí)間:: " + sleepTime + " ticks, 實(shí)際睡眠時(shí)間: " + actualSleepTime);</p><p><b>
81、}</b></p><p> public static void print(String Message) {</p><p> System.out.println(Message);</p><p><b> }</b></p><p><b> 測(cè)試結(jié)果:</b><
82、;/p><p> 3.4 Task1.4 communicator類</p><p><b> 3.4.1要求分析</b></p><p> 用條件變量實(shí)現(xiàn)Communicator類中的speaker()和listen()函數(shù)一個(gè)鎖lock</p><p> 兩個(gè)條件變量speaker(lock),listener(
83、lockw)用來用來控制聽說動(dòng)作。</p><p> 3.4.2 設(shè)計(jì)方案</p><p> 每個(gè)Communicator有一個(gè)鎖 lock(保證操作的原子性) 和與該鎖聯(lián)系的兩個(gè)條件變量用于保證speaker和listener間的同步. 在speak函數(shù)中, 首先檢查若已經(jīng)有一個(gè)speaker在等待(message變量) 或無listener等待, 則掛起. 否則設(shè)置message變
84、量, 準(zhǔn)備數(shù)據(jù)并喚醒一個(gè)listener. 在listen函數(shù)中, 增加一個(gè)listener后, 若speaker數(shù)目非零且listener數(shù)目為1(及恰好可以配對(duì))首先喚醒speaker, 然后將自己掛起以等待 speaker準(zhǔn)備好數(shù)據(jù)再將自己?jiǎn)拘? 這個(gè)問題其實(shí)是一個(gè)緩沖區(qū)長(zhǎng)度為0的生產(chǎn)者/消費(fèi)者問題。</p><p> 3.4.3 實(shí)現(xiàn)代碼</p><p> Communicat
85、or.java</p><p> int speakersize,listenersize;</p><p> int Message; </p><p> Condition speaker,listener;</p><p> public void speak(int word) {</p><p>&l
86、t;b> //</b></p><p> while(speakersize!=0)</p><p> {speaker.sleep();}</p><p> this.Message=word;</p><p> speakersize++;</p><p> while(listene
87、rsize==0)</p><p> {speaker.sleep();}</p><p> listener.wake();</p><p> speakersize--;</p><p> speaker.wake();</p><p><b> //</b></p>
88、<p><b> }</b></p><p> public int listen() {</p><p><b> //</b></p><p> listenersize++;</p><p> if(listenersize==1&&speakersize!
89、=0)</p><p> speaker.wake();</p><p> listener.sleep();</p><p> int myMessage = this.Message;</p><p> listenersize--;</p><p><b> //</b></
90、p><p> return myMessage;</p><p><b> }</b></p><p> 3.4.4 測(cè)試代碼及結(jié)果</p><p><b> 測(cè)試代碼:</b></p><p> package nachos.threads;</p>&
91、lt;p> import nachos.machine.*;</p><p> import java.util.ArrayList;</p><p> import java.util.Random;</p><p> public class CommunicatorTest {</p><p> public Commu
92、nicatorTest(){ </p><p> Message = 1;</p><p> numOfSpeakers = 5;</p><p> numOfListeners = 5;</p><p> communicator = new Communicator();</p><p><b>
93、 }</b></p><p> public void commTest(int num){</p><p> System.out.println("\nCommunicatorTest開始"); </p><p> for (int i=0; i<num; i++){ </p><p> c
94、reateSpeakers(numOfSpeakers);</p><p> createListeners(numOfListeners);</p><p> print("\n 演講者: " + numOfSpeakers);</p><p> print(" 收聽者: " + numOfListeners + &q
95、uot;\n");</p><p> sleep(numOfSpeakers+numOfListeners);</p><p> print("\n 演講者與收聽者創(chuàng)建完畢");</p><p><b> }</b></p><p> print(" Communic
96、ator Test結(jié)束 ");</p><p><b> }</b></p><p> public void sleep(int numThreadsCreated){</p><p> ThreadedKernel.alarm.waitUntil(numThreadsCreated*100);</p><
97、p><b> }</b></p><p> public class Listener implements Runnable{</p><p> public void run(){</p><p> int messageToRecieve = communicator.listen();</p><p&g
98、t; System.out.print(Message);</p><p> print(" " + KThread.currentThread().getName() + " 收到信息 " + messageToRecieve);</p><p><b> }</b></p><p><
99、b> }</b></p><p> public class Speaker implements Runnable{</p><p> public void run(){</p><p> communicator.speak(Message++);</p><p> System.out.print(Mess
100、age);</p><p> print(" " + KThread.currentThread().getName() + " 發(fā)出信息 " + Message); </p><p><b> }</b></p><p><b> }</b&g
101、t;</p><p> public void createSpeakers(int speakers){</p><p><b> int j;</b></p><p> for(j=1; j<=speakers; j++){ </p><p> KThread speaker
102、Thread = new KThread(new Speaker());</p><p> speakerThread.setName("Speaker" + j);</p><p> speakerThread.fork();</p><p><b> };</b></p><p><b
103、> }</b></p><p> public void createListeners(int listeners){</p><p><b> int k;</b></p><p> for(k=1; k<=listeners; k++){ </p><p> KThr
104、ead listenerThread = new KThread(new Listener());</p><p> listenerThread.setName("Listener" + k);</p><p> listenerThread.fork(); </p><p><b> }</b&
105、gt;</p><p><b> }</b></p><p> public static void print(String Message) {</p><p> System.out.println(Message);</p><p><b> }</b></p><
106、;p> public static int MAX_THREADS = 245;</p><p> private int Message; </p><p> private Communicator communicator;</p><p> private int numOfSpeakers;</p><p> pri
107、vate int numOfListeners;</p><p><b> }</b></p><p><b> 測(cè)試結(jié)果:</b></p><p> 3.5 Task1.5 priority scheduler類</p><p> 3.5.1 要求分析</p><p&g
108、t; 實(shí)現(xiàn)優(yōu)先級(jí)調(diào)度。優(yōu)先級(jí)調(diào)度的傳統(tǒng)算法如下: 每個(gè)線程擁有一個(gè)優(yōu)先級(jí) (在Nachos中, 優(yōu)先級(jí)是一個(gè)0到7之間的整數(shù), 默認(rèn)為1, 在線程調(diào)度時(shí), 調(diào)度程序選擇一個(gè)擁有最高優(yōu)先級(jí)的處于就緒狀態(tài)的線程運(yùn)行. 這種算法的問題是可能出現(xiàn) “饑餓” 現(xiàn)象: 設(shè)想有一個(gè)低優(yōu)先級(jí)的線程處于臨界區(qū)中運(yùn)行而高優(yōu)先級(jí)的線程在臨界區(qū)外等待. 由于前者優(yōu)先級(jí)較低, 它可能不會(huì)被調(diào)度器選中從而高優(yōu)先級(jí)的線程也不得不浪費(fèi)時(shí)間等待. 需要實(shí)現(xiàn)一種 “讓
109、出” 優(yōu)先級(jí)的機(jī)制 (Priority Donation) : 提高擁有鎖的低優(yōu)先級(jí)線程的優(yōu)先級(jí)以使它迅速完成臨界區(qū), 不使其它較高優(yōu)先級(jí)的線程等待太久. 重新計(jì)算后的優(yōu)先級(jí)稱為有效優(yōu)先級(jí), 它可以不斷變化. 以有效優(yōu)先級(jí)為評(píng)判標(biāo)準(zhǔn)。</p><p> 3.5.2 設(shè)計(jì)方案</p><p> 要完善的的PriorityScheduler繼承自Scheduler。</p>
110、<p> 每個(gè)KThread類的線程都有一個(gè)ThreadState類型的 “線程狀態(tài)” 對(duì)象. 它保存了線程的優(yōu)先級(jí)priority, 有效優(yōu)先級(jí)effectivepriority以及正在等待該線程的線程隊(duì)列IHave,該線程正在等待的線程隊(duì)列Iwant。</p><p> 不同的調(diào)度器有不同類型的線程隊(duì)列 (均繼承自ThreadQueue), 線程隊(duì)列由threadstate對(duì)象組成,是否可提高擁
111、有鎖線程優(yōu)先級(jí)取決于創(chuàng)建隊(duì)列時(shí)transferPriority的值,它有兩個(gè)重要方法: waitForAccess (KThread thread) 和 acquire (KThread thread). 前者將一個(gè)線程加入等待隊(duì)列(實(shí)際上加入的是該線程的線程狀態(tài)); 后者將一個(gè)線程從等待隊(duì)列中取出, 表示該線程當(dāng)前處于運(yùn)行狀態(tài). 這兩個(gè)方法中, 首先找出相應(yīng)線程的線程狀態(tài), 再對(duì)線程狀態(tài)執(zhí)行相應(yīng)的waitForAccess和acqui
112、re操作 (對(duì)隊(duì)列的真正操作是在ThreadState.waitForAccess和ThreadState.acquire中完成的). </p><p> 設(shè)置, 返回, 增加, 減少優(yōu)先級(jí)的函數(shù)在Scheduler中. ThreadQueue.nextThread 返回當(dāng)前隊(duì)列中有效優(yōu)先級(jí)最高的線程以運(yùn)行之. 而對(duì)某線程有效優(yōu)先級(jí)的計(jì)算在ThreadState.getEffectivePriority中. 當(dāng)
113、該線程沒有正在等待該線程的線程隊(duì)列時(shí),其有效優(yōu)先級(jí)等于其原始優(yōu)先級(jí),不然其有效優(yōu)先級(jí)增加到不小于其等待隊(duì)列IHave中各線程的有效優(yōu)先級(jí)的最大值。它也有兩個(gè)重要方法: waitForAccess (PriorityQueue waitQueue) 和 acquire (PriorityQueue waitQueue). 前者將waitqueue從IHave中移出(如果有的話),加入到Iwant中;后者將waitqueue從Iwant中移
114、出(如果有的話),加入到IHave中。</p><p> 3.5.3 實(shí)現(xiàn)代碼</p><p> 實(shí)現(xiàn)代碼見PriorityScheduler.java</p><p> 3.5.4 測(cè)試代碼及結(jié)果</p><p> 測(cè)試方法:創(chuàng)建兩個(gè)可捐贈(zèng)優(yōu)先級(jí)q1,q2及一個(gè)不可捐贈(zèng)優(yōu)先級(jí)隊(duì)列q3,三個(gè)優(yōu)先級(jí)分別高中低線程l,m,h。第一次將l,
115、m加到q1隊(duì)列,l,m加到q2隊(duì)列,測(cè)試q1的nextthread是m。分別測(cè)試優(yōu)先級(jí)是否“捐贈(zèng)”,低優(yōu)先級(jí)線程提告優(yōu)先級(jí)后是否被選中執(zhí)行。</p><p><b> 測(cè)試代碼:</b></p><p> package nachos.threads;</p><p> import nachos.machine.*;</p>
116、<p> import nachos.threads.*;</p><p> import java.util.*;</p><p> public final class PrioritySchedulerTest {</p><p><b> /**</b></p><p> * A very
117、 simple test of priority donation.</p><p><b> */</b></p><p> public static void simplePrioritySchedulerTest() {</p><p> final boolean intStatus = Machine.interrupt().
118、disable();</p><p> final Scheduler s = new PriorityScheduler();</p><p> // Enable priority donation.</p><p> final ThreadQueue q1 = s.newThreadQueue(false);</p><p>
119、 final ThreadQueue q2 = s.newThreadQueue(false);</p><p> final ThreadQueue q3 = s.newThreadQueue(true);</p><p> final Runnable dummyRunnable = new Runnable() {</p><p> public voi
120、d run() {</p><p> // do nothing</p><p><b> }</b></p><p><b> };</b></p><p> // Create dummy Threads</p><p> final KThread lowPr
121、iorityThread = new KThread(dummyRunnable);</p><p> final KThread mediumPriorityThread = new KThread(dummyRunnable);</p><p> final KThread highPriorityThread = new KThread(dummyRunnable);</p
122、><p> // Link dummy Threads with dummy ThreadState</p><p> s.setPriority(lowPriorityThread, PriorityScheduler.priorityMinimum);</p><p> s.setPriority(mediumPriorityThread, PriorityS
123、cheduler.priorityDefault);</p><p> s.setPriority(highPriorityThread, PriorityScheduler.priorityMaximum);</p><p> // Test that effectivePriorities start as the initial priorities</p><
124、;p> Lib.assertTrue(s.getEffectivePriority(lowPriorityThread) == PriorityScheduler.priorityMinimum);</p><p> Lib.assertTrue(s.getEffectivePriority(mediumPriorityThread) == PriorityScheduler.priorityDefau
125、lt);</p><p> Lib.assertTrue(s.getEffectivePriority(highPriorityThread) == PriorityScheduler.priorityMaximum);</p><p> // Low priority and Medium priority thread are on the ready queue</p>
126、;<p> q1.waitForAccess(lowPriorityThread);</p><p> q1.waitForAccess(mediumPriorityThread);</p><p> q2.waitForAccess(lowPriorityThread);</p><p> q2.waitForAccess(mediumPri
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 眾賞文庫(kù)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 操作系統(tǒng)程序設(shè)計(jì)課程設(shè)計(jì)報(bào)告-操作系統(tǒng)模擬實(shí)現(xiàn)
- 操作系統(tǒng)課程設(shè)計(jì)——操作系統(tǒng)課程設(shè)計(jì)模擬操作系統(tǒng)
- 操作系統(tǒng)課程設(shè)計(jì)報(bào)告
- 課程設(shè)計(jì)報(bào)告--操作系統(tǒng)
- 操作系統(tǒng)課程設(shè)計(jì)報(bào)告
- 《操作系統(tǒng)》課程設(shè)計(jì)報(bào)告
- 操作系統(tǒng)課程設(shè)計(jì)報(bào)告
- 操作系統(tǒng)課程設(shè)計(jì)報(bào)告
- 操作系統(tǒng)課程設(shè)計(jì)--模擬操作系統(tǒng)的實(shí)現(xiàn)
- 操作系統(tǒng)課程設(shè)計(jì)-- 操作系統(tǒng)
- 操作系統(tǒng)課程設(shè)計(jì)子用戶界面及托盤實(shí)現(xiàn)
- 操作系統(tǒng)課程設(shè)計(jì)報(bào)告2014217151
- 《操作系統(tǒng)原理》課程設(shè)計(jì)報(bào)告
- 操作系統(tǒng)課程設(shè)計(jì)報(bào)告2
- 操作系統(tǒng)課程設(shè)計(jì)-子用戶界面及托盤實(shí)現(xiàn)
- 操作系統(tǒng)課程設(shè)計(jì)
- 操作系統(tǒng)課程設(shè)計(jì)
- 操作系統(tǒng)課程設(shè)計(jì)
- 操作系統(tǒng)課程設(shè)計(jì)
- 操作系統(tǒng)課程設(shè)計(jì)
評(píng)論
0/150
提交評(píng)論