版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡介
1、整理課件第八章第八章 線程線程 本章導(dǎo)讀本章導(dǎo)讀n Java中的線程n 線程的生命周期n 線程的優(yōu)先級(jí)與調(diào)度管理n Thread的子類創(chuàng)建線程n Runable接口n 線程同步n wait()、notify 和notifyAll()方法n 掛起、恢復(fù)和終止線程n 線程的聯(lián)合n 守護(hù)線程整理課件線程概述線程概述 Java語言的一大特點(diǎn)就是內(nèi)置對(duì)多線程的支持(包中的Thread類)。多線程是指同時(shí)存在幾個(gè)執(zhí)行體,按幾條不同的執(zhí)行線索共同工作的情況,它使得編程人員可以很方便地開發(fā)出具有多線程功能、能同時(shí)處理多個(gè)任務(wù)的功能強(qiáng)大的應(yīng)用程序。雖然執(zhí)行線程給人一種幾個(gè)事件同時(shí)發(fā)生的感覺,但這只是一種錯(cuò)覺,因
2、為我們的計(jì)算機(jī)在任何給定的時(shí)刻只能執(zhí)行這些線程中的一個(gè)。為了建立這些線程正在同步執(zhí)行的感覺,Java快速地把控制從一個(gè)線程切換到另一個(gè)線程。 class Qution public static void main(String args) while(true) System.out.println(123); while(true) System.out.println(abc); 整理課件8.1 Java中的線程中的線程 程序是一段靜態(tài)的代碼,它是應(yīng)用軟件執(zhí)行的藍(lán)本。進(jìn)程是程序的一次動(dòng)態(tài)執(zhí)行過程,它對(duì)應(yīng)了從代碼加載、執(zhí)行至執(zhí)行完畢的一個(gè)完整過程,這個(gè)過程也是進(jìn)程本身從產(chǎn)生、發(fā)展至消亡的過
3、程。線程是比進(jìn)程更小的執(zhí)行單位。一個(gè)進(jìn)程在其執(zhí)行過程中,可以產(chǎn)生多個(gè)線程,形成多條執(zhí)行線索,每條線索,即每個(gè)線程也有它自身的產(chǎn)生、存在和消亡的過程,也是一個(gè)動(dòng)態(tài)的概念。 Java應(yīng)用程序總是從主類的main方法開始執(zhí)行。當(dāng)JVM加載代碼,發(fā)現(xiàn)main方法之后,就會(huì)啟動(dòng)一個(gè)線程,這個(gè)線程稱為“主線程”,該線程負(fù)責(zé)執(zhí)行main方法。那么,在main方法中再創(chuàng)建的線程,就稱為主線程中的線程。如果main方法中沒有創(chuàng)建其他線程,那么當(dāng)main方法執(zhí)行完最后一個(gè)語句,即main方法返回時(shí),JVM就會(huì)結(jié)束該Java應(yīng)用程序。如果main方法中又創(chuàng)建了其他線程,那么JVM就要在主線程和其他線程之間輪流切換,
4、保證每個(gè)線程都有機(jī)會(huì)使用CPU資源,main方法即使執(zhí)行完最后的語句,JVM也不會(huì)結(jié)束該程序,JVM一直要等到主線程中的所有線程都結(jié)束之后,才結(jié)束該Java應(yīng)用程序。 整理課件8.2 線程的生命周期線程的生命周期 在Java語言中,Thread類及其子類創(chuàng)建的對(duì)象稱為線程。新建的線程在它的一個(gè)完整的生命周期中通常要經(jīng)歷4種狀態(tài)。1)新建:線程對(duì)象被聲明并創(chuàng)建時(shí) 2)運(yùn)行:線程對(duì)象調(diào)用start()方法3)中斷:中斷的原因消除時(shí),線程可以從中斷處繼續(xù)運(yùn)行. 有4種原因的中斷: *JVM將CPU資源從當(dāng)前線程切換給其他線程 *線程使用CPU資源期間,執(zhí)行了sleep(int millsecond)
5、方法 *線程使用CPU資源期間,執(zhí)行了wait()方法 *線程使用CPU資源期間,執(zhí)行某個(gè)操作進(jìn)入阻塞狀態(tài)4)死亡: 釋放分配給線程對(duì)象的內(nèi)存。 線程死亡的原因有二: 即執(zhí)行完run()方法中的全部語句或或線程被提前強(qiáng)制性終止,即強(qiáng)制run()方法結(jié)束。 例8-1 Thread的子類WriteWordThread創(chuàng)建了兩個(gè)線程。 注注: :上述程序在不同的計(jì)算機(jī)運(yùn)行或在同一臺(tái)計(jì)算機(jī)反復(fù)運(yùn)行的結(jié)果不盡相同,輸出結(jié)果依賴當(dāng)前CPU資源的使用情況。為了使結(jié)果盡量不依賴于當(dāng)前CPU資源的使用情況,我們應(yīng)當(dāng)讓線程主動(dòng)調(diào)用sleep()方法讓出CPU的使用權(quán)進(jìn)入中斷狀態(tài)。 例8-2 整理課件例子8-1效果
6、圖整理課件8.3 線程的優(yōu)先級(jí)與調(diào)度管理線程的優(yōu)先級(jí)與調(diào)度管理 Java虛擬機(jī)中的線程調(diào)度器負(fù)責(zé)管理線程,調(diào)度器把線程的優(yōu)先級(jí)分為10個(gè)級(jí)別,分別用Thread類中的類常量表示。每個(gè)Java線程的優(yōu)先級(jí)都在常數(shù)1(Thread.MIN PRIORITY)到常數(shù)10(Thread.MAX_PRIORITY)的范圍內(nèi)。如果沒有明確地設(shè)置線程的優(yōu)先級(jí)別,每個(gè)線程的優(yōu)先級(jí)都為常數(shù)5(包括主線)。 線程的優(yōu)先級(jí)可以通過setPriority(int grade)方法調(diào)整,這一方法需要一個(gè)int類型參數(shù)。有些操作系統(tǒng)只能識(shí)別3個(gè)級(jí)別:1,5,10。 在采用時(shí)間片的系統(tǒng)中,每個(gè)線程都有機(jī)會(huì)獲得CPU的使用權(quán)
7、,以便使用CPU資源執(zhí)行線程中的操作。當(dāng)線程使用CPU資源的時(shí)間結(jié)束后,即使線程沒有完成自己的全部操作,Java調(diào)度器也會(huì)中斷當(dāng)前線程的執(zhí)行,把CPU的使用權(quán)切換給下一個(gè)排隊(duì)等待的線程,當(dāng)前線程將等待CPU資源的下一次輪回,然后從中斷處繼續(xù)執(zhí)行。 Java調(diào)度器的任務(wù)是使高優(yōu)先級(jí)的線程能始終運(yùn)行,一旦時(shí)間片有空閑,則使具有同等優(yōu)先級(jí)的線程以輪流的方式順序使用時(shí)間片。 整理課件8.4 Thread 的子類創(chuàng)建線程的子類創(chuàng)建線程 在Java語言中,用Thread類或子類創(chuàng)建線程對(duì)象。 用戶可以擴(kuò)展 Thread類,但需要重寫父類的run()方法,其目的是規(guī)定線程的具體操作,否則線程就什么也不做,因
8、為父類的run()方法中沒有任何操作語句。 例子8-3中除主線程外還有兩個(gè)線程,這兩個(gè)線程分別在命令行窗口的左側(cè)和右側(cè)順序地一行一行地輸出字符串。主線程負(fù)責(zé)判斷輸出的行數(shù),當(dāng)其中任何一個(gè)線程輸出8行后,就結(jié)束進(jìn)程。 本例題中用到了System類中的類方法exit(int n),主線程使用該方法結(jié)束整個(gè)程序。 整理課件8.5 Runnable接口接口 使用Thread子類創(chuàng)建線程的優(yōu)點(diǎn)是:我們可以在子類中增加新的成員變量,使線程具有某種屬性,也可以在子類中新增加方法,使線程具有某種功能。但是,Java不支持多繼承,Thread類的子類不能再擴(kuò)展其他的類。整理課件1Runnable接口與目標(biāo)對(duì)象
9、創(chuàng)建線程的另一個(gè)途徑就是用Thread類直接創(chuàng)建線程對(duì)象。使用Thread創(chuàng)建線程對(duì)象時(shí),通常使用的構(gòu)造方法是: Thread(Runnable target) 該構(gòu)造方法中的參數(shù)是一個(gè)Runnable類型的接口,因此,在創(chuàng)建線程對(duì)象時(shí)必須向構(gòu)造方法的參數(shù)傳遞一個(gè)實(shí)現(xiàn)Runnable接口類的實(shí)例,該實(shí)例對(duì)象稱作所創(chuàng)線程的目標(biāo)對(duì)象,當(dāng)線程調(diào)用start()方法后,一旦輪到它來享用CPU資源,目標(biāo)對(duì)象就會(huì)自動(dòng)調(diào)用接口中的run()方法(接口回調(diào)),這一過程是自動(dòng)實(shí)現(xiàn)的,用戶程序只需要讓線程調(diào)用start()方法即可,也就是說,當(dāng)線程被調(diào)度并轉(zhuǎn)入運(yùn)行狀態(tài)時(shí),所執(zhí)行的就是run()方法中所規(guī)定的操作。
10、 例8-4不使用Thread類的子類創(chuàng)建線程,而是使用Thread類創(chuàng)建left和right線程。 線程間可以共享相同的內(nèi)存單元(包括代碼與數(shù)據(jù)),并利用這些共享單元來實(shí)現(xiàn)數(shù)據(jù)交換、實(shí)時(shí)通信和必要的同步操作。 例8-5中,線程zhang和chengmoney。當(dāng)money的值小于100時(shí),線程zhang結(jié)束自己的run()方法進(jìn)入死亡狀態(tài);當(dāng)money的值小于60時(shí),線程cheng結(jié)束自己的run()方法進(jìn)入死亡狀態(tài),(效果如圖所示)。 例8-6中中共有4個(gè)線程threadA、threadB、threadC和threadD,其中threadA和threadB的目標(biāo)對(duì)象a1,threadC和th
11、readD的目標(biāo)對(duì)象是a2。threadA和threadB共享a1的成員number,而threadC和threadD共享a2的成員number。 整理課件例子8-5效果圖整理課件2目標(biāo)對(duì)象與線程的關(guān) 目標(biāo)對(duì)象和線程的關(guān)系有以下兩種情景。1 1)目標(biāo)對(duì)象和線程完全解藕)目標(biāo)對(duì)象和線程完全解藕 在上述例8-5中,創(chuàng)建目標(biāo)對(duì)象的Bank類并沒有組合zhang和cheng線程對(duì)象,也就是說Bank創(chuàng)建的目標(biāo)對(duì)象bank不包含對(duì)象zhang和cheng線程對(duì)象的引用(完全解藕)。在這種情況下,目標(biāo)對(duì)象經(jīng)常需要通過獲得線程的名字(因?yàn)闊o法獲得線程對(duì)象的引用): String name = Thread.
12、currentThread().getName(); String name = Thread.currentThread().getName();以便確定是哪個(gè)線程正在占用CPU資源,即被JVM正在執(zhí)行的線程。2 2)目標(biāo)對(duì)象組合線程(弱藕合)目標(biāo)對(duì)象組合線程(弱藕合) 目標(biāo)對(duì)象可以組合線程,即將線程作為自己的成員(弱藕合),比如讓線程zhang和cheng在bank中。當(dāng)創(chuàng)建目標(biāo)對(duì)象類組合線程對(duì)象時(shí),目標(biāo)對(duì)象可以通過獲得線程對(duì)象的引用: Thread.currentThread(); Thread.currentThread(); 來確定是哪個(gè)線程正在占用CPU資源,即被JVM正在執(zhí)行的線
13、程,如下面的例8-7中代碼所示。整理課件3 關(guān)于run()方法中的局部變量 對(duì)于具有相同目標(biāo)對(duì)象的線程,當(dāng)其中一個(gè)線程享用CPU資源時(shí),目標(biāo)對(duì)象自動(dòng)調(diào)用接口中的run()方法,這時(shí)run()方法中的局部變量被分配內(nèi)存空間。當(dāng)輪到另一個(gè)線程享用CPU資源時(shí),目標(biāo)對(duì)象會(huì)再次調(diào)用接口中的run()方法,那么run()方法中的局部變量會(huì)再次分配內(nèi)存空間。也就是說,run()方法已經(jīng)啟動(dòng)運(yùn)行了兩次,分別運(yùn)行在不同的線程中,即運(yùn)行在不同的時(shí)間片內(nèi)。 我們稱run()方法中的局部變量為線程的局部變量,不同線程的run()方法中的局部變量互不干擾,一個(gè)線程改變了自己的run()方法中局部變量的值不會(huì)影響其他線
14、程的run()方法中的局部變量。 例8-8演示了兩個(gè)線程的run()方法中的局部變量互不干擾(效果如圖所示)。 整理課件8.6 線程的常用方法線程的常用方法_1 1 1start()start() 線程調(diào)用該方法將啟動(dòng)線程,使之從新建狀態(tài)進(jìn)入就緒隊(duì)列排隊(duì),一旦輪到它來享用CPU資源時(shí),就可以脫離創(chuàng)建它的主線程獨(dú)立開始自己的生命周期了。 2 2run()run() Thread類的run()方法與Runnable接口中的run()方法的功能和作用相同,都用來定義線程對(duì)象被調(diào)度之后所執(zhí)行的操作,都是系統(tǒng)自動(dòng)調(diào)用而用戶程序不得引用的方法。 Thread類中,run()方法沒有具體內(nèi)容,所以用戶程序需
15、要?jiǎng)?chuàng)建自己的Thread類的子類,并重寫run()方法來覆蓋原來的run()方法。當(dāng)run()方法執(zhí)行完畢,線程就變成死亡狀態(tài)。在線程沒有結(jié)束run()方法之前,不建議讓線程再調(diào)用start()方法,否則將發(fā)生IllegalThreadStateException異常。3 3sleep(int millsecond)sleep(int millsecond) 有時(shí),優(yōu)先級(jí)高的線程需要優(yōu)先級(jí)低的線程做一些工作來配合它,或者優(yōu)先級(jí)高的線程需要完成一些費(fèi)時(shí)的操作,此時(shí)優(yōu)先級(jí)高的線程應(yīng)該讓出處理器,使優(yōu)先級(jí)低的線程有機(jī)會(huì)執(zhí)行。為達(dá)到這個(gè)目的,優(yōu)先級(jí)高的線程可以在它的run()方法中調(diào)用sleep()方
16、法來使自己放棄處理器資源,休眠一段時(shí)間。休眠時(shí)間的長短由sleep()方法的參數(shù)決定,millsecond是毫秒為單位的休眠時(shí)間。如果線程在休眠時(shí)被打斷,JVM就拋出InterruptedException異常。因此,必須在try-catch語句塊中調(diào)用sleep()方法。整理課件8.6 線程的常用方法線程的常用方法_2 4 4isAlive()isAlive() 在線程的run方法結(jié)束之前,即沒有進(jìn)入死亡狀態(tài)之前,線程調(diào)用isAlive()方法返回true,當(dāng)線程進(jìn)入死亡狀態(tài)后(實(shí)體內(nèi)存被釋放),線程仍可以調(diào)用方法isAlive(),這時(shí)返回的值是false。 需要注意的是,一個(gè)已經(jīng)運(yùn)行的線
17、程在沒有進(jìn)入死亡狀態(tài)時(shí),不要再給線程分配實(shí)體,由于線程只能引用最后分配的實(shí)體,先前的實(shí)體就會(huì)成為“垃圾”,并且不會(huì)被垃圾收集機(jī)收集掉。內(nèi)存示意如圖所示。 例8-9中,一個(gè)線程每隔1秒鐘在命令行窗口輸出機(jī)器的當(dāng)前時(shí)間,在輸出3秒之后,該線程又被分配了實(shí)體,新實(shí)體又開始運(yùn)行。這時(shí),我們?cè)诿钚忻棵腌娔芸匆妰尚挟?dāng)前時(shí)間,因?yàn)槔鴮?shí)體仍然在工作(效果如圖所示)。 5 5currentThread() currentThread() currentThread()方法是Thread類中的類方法,可以用類名調(diào)用,該方法返回當(dāng)前正在使用CPU資源的線程。 6 6interrupt()interrupt()
18、當(dāng)一些線程調(diào)用sleep()方法處于休眠狀態(tài)時(shí),一個(gè)使用CPU資源的其他線程在執(zhí)行過程中,可以讓休眠的線程分別調(diào)用interrupt()方法“吵醒”自己,即導(dǎo)致休眠的線程發(fā)生InterruptedException異常,從而結(jié)束休眠,重新排隊(duì)等待CPU資源。 例8-10中有3個(gè)線程zhangXiao、zhengMing和teacher,其中2個(gè)線程zhangXiao和zhengMing準(zhǔn)備休眠10秒鐘后,再輸出“早上好!”。teacher線程在輸出3句“上課”后,吵醒休眠的線程:zhangXiao,zhangXiao被吵醒后再吵醒zhengMing(效果)。 整理課件圖整理課件8.7 線程同步
19、線程同步 問題問題: 兩個(gè)或多個(gè)線程會(huì)同時(shí)訪問同一個(gè)變量,并且一個(gè)線程需要修改這個(gè)變量。我們應(yīng)對(duì)這樣的問題做出處理,否則可能發(fā)生混亂。 解決解決: 把修改數(shù)據(jù)的方法用關(guān)鍵字synchronized來修飾。 一個(gè)方法使用關(guān)鍵字synchronized修飾后,如果一個(gè)線程A占有CPU資源期間,使得該方法被調(diào)用執(zhí)行,同步方法調(diào)用執(zhí)行完畢之前,其他占有CPU資源的線程一旦調(diào)用這個(gè)同步方法就會(huì)引起堵塞,堵塞的線程要一直等到堵塞的原因消除(同步方法返回),再排隊(duì)等待CPU資源,以便使用這個(gè)同步方法。 所謂線程同步就是若干個(gè)線程都需要使用一個(gè)synchronized修飾的方法。 例子8-11中有兩個(gè)線程:a
20、ccountant和cashier,它們共同擁有一個(gè)賬本。它們都可以使用saveOrTake(int number)對(duì)賬本進(jìn)行訪問,會(huì)計(jì)使用saveOrTake()方法時(shí),向賬本上寫入存錢記錄;出納使用saveOrTake()方法時(shí),向賬本寫入取錢記錄。因此,當(dāng)會(huì)計(jì)正在使用saveOrTake()方法時(shí),出納被禁止使用,反之也是這樣。比如,會(huì)計(jì)每次使用saveOrTake()方法時(shí),在賬本上存入90萬元,在存入這筆錢時(shí),分3次存完,每存入30萬元,就喝口茶,那么他喝茶休息時(shí)(注意:存錢這件事還沒結(jié)束,即會(huì)計(jì)還沒有使用完saveOrTake()方法),出納仍不能使用saveOrTake()方法。
21、我們要保證其中一人使用saveOrTake()方法時(shí),另一個(gè)人將必須等待()。 整理課件例子8-11效果 如圖整理課件8.8 使用使用wait()、notify()和和notifyall()協(xié)調(diào)同步線程協(xié)調(diào)同步線程 wait()wait()、notify()notify()和和notifyAll()notifyAll()都是Object類中的final方法,被所有的類繼承,且不允許重寫。 wait()wait()方法方法可以中斷線程的執(zhí)行,使本線程等待,暫時(shí)讓出CPU的使用權(quán),并允許其他線程這個(gè)同步方法。 其他線程如果在使用這個(gè)同步方法時(shí)不需要等待,那么它使用完這個(gè)同步方法的同時(shí),應(yīng)當(dāng)用not
22、ifyAll()notifyAll()方法方法通知所有的由于使用這個(gè)同步方法而處于等待的線程結(jié)束等待。曾中斷的線程就會(huì)重新排隊(duì)等待CPU資源,以便從剛才的中斷處繼續(xù)執(zhí)行這個(gè)同步方法 注意:如果使用notify()notify(),那么只是通知處于等待中的線程的某一個(gè)結(jié)束等待。 例子8-12中,為了避免復(fù)雜數(shù)學(xué)算法,我們模擬3個(gè)人排隊(duì)買票,每人買1張票。售票員只有1張五元的錢,電影票五元錢一張。張某拿1張二十元的人民幣排在孫某前面買票,孫某拿1張十元的人民幣排在趙的前面買票,趙某拿1張五元的人民幣排在最后。那么,最終的賣票次序應(yīng)當(dāng)是孫、趙、張(效果如圖8.9所示)。整理課件例子8-12效果 如圖
23、整理課件8.9 掛起、恢復(fù)和終止線程 所謂掛起所謂掛起一個(gè)線程就是讓線程暫時(shí)讓出CPU的使用權(quán)限,暫時(shí)停止執(zhí)行,但停止執(zhí)行的持續(xù)時(shí)間不確定,因此不能使用sleep()方法暫停線程。 如果線程有目標(biāo)對(duì)象,那么當(dāng)前線程在占有CPU期間,只要讓其目標(biāo)對(duì)象在某個(gè)同步方法中調(diào)用wait()就可以掛起當(dāng)前線程。為了恢復(fù)恢復(fù)這個(gè)掛起的線程(從曾中斷處繼續(xù)線程的執(zhí)行),其他線程在占有CPU資源期間,讓掛起的線程的目標(biāo)對(duì)象在同步方法中調(diào)用notifyAll()方法即可。 如果線程沒有目標(biāo)對(duì)象(用Thread子類創(chuàng)建的線程),那么當(dāng)前線程當(dāng)前線程在占有CPU期間,只要在某個(gè)同步方法中調(diào)用wait()就可以掛起自己。為了恢復(fù)恢復(fù)這個(gè)掛起的線程,其他線程在占有CPU資源期間,讓掛起的線
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲(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ì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 2024年綠色環(huán)保食材配送餐飲服務(wù)協(xié)議3篇
- 辦公空間照明系統(tǒng)升級(jí)合同樣本
- 地?zé)豳Y源招投標(biāo)投訴處理措施
- 航空航天計(jì)量變更準(zhǔn)則
- 冷庫安裝合同化妝品研究
- 低碳環(huán)保住宅的二手房買賣合同
- 水利工程保溫施工服務(wù)協(xié)議
- 企業(yè)員工商標(biāo)提案管理辦法
- 玩具制造企業(yè)協(xié)議休假管理辦法
- 預(yù)付賬款審核風(fēng)險(xiǎn)控制的關(guān)鍵
- 塑料污染與環(huán)境保護(hù)
- 2024年鍋爐運(yùn)行值班員(中級(jí))技能鑒定理論考試題庫(含答案)
- 福建省泉州市2023-2024學(xué)年高一上學(xué)期期末質(zhì)檢英語試題(解析版)
- 中華人民共和國民法典(總則)培訓(xùn)課件
- 蘇教版(2024新版)七年級(jí)上冊(cè)生物期末模擬試卷 3套(含答案)
- 《項(xiàng)目管理》完整課件
- IB課程-PYP小學(xué)項(xiàng)目省公開課獲獎(jiǎng)?wù)n件說課比賽一等獎(jiǎng)?wù)n件
- 上市央國企數(shù)智化進(jìn)程中人才就業(yè)趨勢(shì)
- 2024-2030年中國苯胺行業(yè)現(xiàn)狀動(dòng)態(tài)與需求前景展望報(bào)告
- 英雄之旅思維模型
- 釘釘數(shù)字化管理師中級(jí)題庫
評(píng)論
0/150
提交評(píng)論