主頁 > 後端開發 > 代表Java未來的ZGC深度剖析,牛逼!

代表Java未來的ZGC深度剖析,牛逼!

2020-09-29 01:43:06 後端開發

JAVA程式最爽的地方是它的GC機制,開發人員不需要關注記憶體申請和回收問題,同時,JAVA程式最頭疼的地方也是它的GC機制,因為掌握JVM和GC調優是一件非常困難的事情,在ParallelOldGC、CMS、G1之后,JDK11帶來的全新的「ZGC」為我們解決了什么問題?Oracle官方介紹它是一個Scalable、Low Latency的垃圾回收器,所以它的目的是「降低停頓時間」,由此會導致吞吐量會有所降低,吞吐量降低問題不大,橫向擴展幾臺服務器就能解決問題了啦,

ZGC目標

如下圖所示,ZGC的目標主要有4個:

  • 支持TB量級的堆,這你受得了嗎?我們生產環境的硬碟還沒有上TB呢,這應該可以滿足未來十年內,所有JAVA應用的需求了吧,
  • 最大GC停頓時間不超10ms,這你受得了嗎?目前一般線上環境運行良好的JAVA應用Minor GC停頓時間在10ms左右,Major GC一般都需要100ms以上(G1可以調節停頓時間,但是如果調的過低的話,反而會適得其反),之所以能做到這一點是因為它的停頓時間主要跟Root掃描有關,而Root數量和堆大小是沒有任何關系的,
  • 奠定未來GC特性的基礎,牛逼,牛逼!
  • 最糟糕的情況下吞吐量會降低15%,這都不是事,停頓時間足夠優秀,至于吞吐量,通過擴容分分鐘解決,

 

image.png

 

另外,Oracle官方提到了它最大的優點是:它的停頓時間不會隨著堆的增大而增長!也就是說,幾十G堆的停頓時間是10ms以下,幾百G甚至上T堆的停頓時間也是10ms以下,

ZGC概述

接下來從幾個維度概述一下ZGC,

  1. New GC
  2. Single Generation
  3. Region Based
  4. Partial Compaction
  5. NUMA-aware
  6. Colored Pointers
  7. Load Barriers
  8. ZGC tuning
  9. Change Log

New GC

ZGC是一個全新的垃圾回收器,它完全不同以往HotSpot的任何垃圾回收器,比如:PS、CMS、G1等,如果真要說它最像誰的話,那應該是Azul公司的商業化垃圾回收器:「C4」,ZGC所采用的演算法就是Azul Systems很多年前提出的Pauseless GC,而實作上它介于早期Azul VM的Pauseless GC與后來Zing VM的C4之間,不過需要說明的是,JDK11中ZGC只能運行在Linux64作業系統之上,JDK14新增支持了MacOS和Window平臺:

 

image.png

 

如下圖所示,是ZGC和Parallel以及G1的壓測對比結果(CMS在JDK9中已經被標記deprecated,更高版本中已經被徹底移除,所以不在對比范圍內),我們可以明顯的看到,停頓時間方面,ZGC是100%不超過10ms的,簡直是秒天秒地般的存在:

 

image.png

 

接下來,再看一下ZGC的垃圾回收程序,如下圖所示,由圖我們可知,ZGC依然沒有做到整個GC程序完全并發執行,依然有3個STW階段,其他3個階段都是并發執行階段:

 

image.png

 

  • Pause Mark Start

這一步就是初始化標記,和CMS以及G1一樣,主要做Root集合掃描,「GC Root是一組必須活躍的參考,而不是物件」,例如:活躍的堆疊幀里指向GC堆中的物件參考、Bootstrap/System類加載器加載的類、JNI Handles、參考型別的靜態變數、String常量池里面的參考、執行緒堆疊/本地(native)堆疊里面的物件指標等,但不包括GC堆里的物件指標,所以這一步驟的STW時間非常短暫,并且和堆大小沒有任何關系,不過會根據執行緒的多少、執行緒堆疊的大小之類的而變化,

 

image.png

 

  • Concurrent Mark/Remap

第二步就是并發標記階段,這個階段在第一步的基礎上,繼續往下標記存活的物件,并發標記后,還會有一個短暫的暫停(Pause Mark End),確保所有物件都被標記,

 

image.png

 

  • Concurrent Prepare for Relocate

即為Relocation階段做準備,選取接下來需要標記整理的Region集合,這個階段也是并發執行的,接下來又會有一個Pause Relocate Start步驟,它的作用是只移動Root集合物件參考,所以這個STW階段也不會停頓太長時間,

 

image.png

 

  • Concurrent Relocate
    最后,就是并發回收階段了,這個階段會把上一階段選中的需要整理的Region集合中存活的物件移到一個新的Region中(這個行為就叫做「Relocate」,即重新安置物件),如上圖所示,Relocate動作完成后,原來占用的Region就能馬上回收并被用于接下來的物件分配,細心的同學可能有疑問了,這就完了?Relocate后物件地址都發生變化了,應用程式還怎么正常操作這些物件呢?這就靠接下來會詳細說明的Load Barrier了,

Single Generation

單代,即ZGC「沒有分代」,我們知道以前的垃圾回收器之所以分代,是因為源于“「大部分物件朝生夕死」”的假設,事實上大部分系統的物件分配行為也確實符合這個假設,

那么為什么ZGC就不分代呢?因為分代實作起來麻煩,作者就先實作出一個比較簡單可用的單代版本,用符合我們國情的話來解釋,大概就是說:作業量太大了,人力又不夠,老板,先上個1.0版本吧!!!

Region Based

這一點和G1一樣,都是基于Region設計的垃圾回收器,ZGC中的Region也被稱為「ZPages」,ZPages被動態創建,動態銷毀,不過,和G1稍微有點不同的是,G1的每個Region大小是完全一樣的,而ZGC的Region大小分為3類:2MB,32MB,N×2MB,如此一來,靈活性就更好了:

 

image.png

 

Partial Compaction

部分壓縮,這一點也很G1類似,以前的ParallelOldGC,以及CMS GC在壓縮Old區的時候,無論Old區有多大,必須整體進行壓縮(CMS GC默認情況下只是標記清除,只會發生FGC時才會采用Mark-Sweep-Compact對Old區進行壓縮),如此一來,Old區越大,壓縮需要的時間肯定就越長,從而導致停頓時間就越長,

而G1和ZGC都是基于Region設計的,在回收的時候,它們只會選擇一部分Region進行回收,這個回收程序采用的是Mark-Compact演算法,即將待回收的Region中存活的物件拷貝到一個全新的Region中,這個新的Region物件分配就會非常緊湊,幾乎沒有碎片,垃圾回收演算法這一點上,和G1是一樣的,

NUMA-aware

NUMA對應的有UMA,UMA即Uniform Memory Access Architecture,NUMA就是Non Uniform Memory Access Architecture,UMA表示記憶體只有一塊,所有CPU都去訪問這一塊記憶體,那么就會存在競爭問題(爭奪記憶體總線訪問權),有競爭就會有鎖,有鎖效率就會受到影響,而且CPU核心數越多,競爭就越激烈,NUMA的話每個CPU對應有一塊記憶體,且這塊記憶體在主板上離這個CPU是最近的,每個CPU優先訪問這塊記憶體,那效率自然就提高了:

 

image.png

 

服務器的NUMA架構在中大型系統上一直非常盛行,也是高性能的解決方案,尤其在系統延遲方面表現都很優秀,ZGC是能自動感知NUMA架構并充分利用NUMA架構特性的,

Colored Pointers

Colored Pointers,即顏色指標是什么呢?如下圖所示,ZGC的核心設計之一,以前的垃圾回收器的GC資訊都保存在物件頭中,而ZGC的GC資訊保存在指標中,每個物件有一個64位指標,這64位被分為:

  • 18位:預留給以后使用;
  • 1位:Finalizable標識,次位與并發參考處理有關,它表示這個物件只能通過finalizer才能訪問;
  • 1位:Remapped標識,設定此位的值后,物件未指向relocation set中(relocation set表示需要GC的Region集合);
  • 1位:Marked1標識;
  • 1位:Marked0標識,和上面的Marked1都是標記物件用于輔助GC;
  • 42位:物件的地址(所以它可以支持2^42=4T記憶體):

 

image.png

 

通過對配置ZGC后物件指標分析我們可知,物件指標必須是64位,那么ZGC就無法支持32位作業系統,同樣的也就無法支持壓縮指標了(CompressedOops,壓縮指標也是32位),

Load Barriers

這個應該翻譯成讀屏障(與之對應的有寫屏障即Write Barrier,之前的GC都是采用Write Barrier,這次ZGC采用了完全不同的方案),這個是ZGC一個非常重要的特性,在標記和移動物件的階段,每次「從堆里物件的參考型別中讀取一個指標」的時候,都需要加上一個Load Barriers,那么我們該如何理解它呢?看下面的代碼,第一行代碼我們嘗試讀取堆中的一個物件參考obj.fieldA并賦給參考o(fieldA也是一個物件時才會加上讀屏障),如果這時候物件在GC時被移動了,接下來JVM就會加上一個讀屏障,這個屏障會把讀出的指標更新到物件的新地址上,并且把堆里的這個指標“修正”到原本的欄位里,這樣就算GC把物件移動了,讀屏障也會發現并修正指標,于是應用代碼就永遠都會持有更新后的有效指標,而且不需要STW,那么,JVM是如何判斷物件被移動過呢?就是利用上面提到的顏色指標,如果指標是Bad Color,那么程式還不能往下執行,需要「slow path」,修正指標;如果指標是Good Color,那么正常往下執行即可:

 

image.png

 

這個動作是不是非常像JDK并發中用到的CAS自旋?讀取的值發現已經失效了,需要重新讀取,而ZGC這里是之前持有的指標由于GC后失效了,需要通過讀屏障修正指標,

后面3行代碼都不需要加讀屏障:Object p = o這行代碼并沒有從堆中讀取資料;o.doSomething()也沒有從堆中讀取資料;obj.fieldB不是物件參考,而是原子型別,

正是因為Load Barriers的存在,所以會導致配置ZGC的應用的吞吐量會變低,官方的測驗資料是需要多出額外4%的開銷:

 

image.png

 

那么,判斷物件是Bad Color還是Good Color的依據是什么呢?就是根據上一段提到的Colored Pointers的4個顏色位,當加上讀屏障時,根據物件指標中這4位的資訊,就能知道當前物件是Bad/Good Color了,

「擴展閱讀」:既然低42位指標可以支持4T記憶體,那么能否通過預約更多位給物件地址來達到支持更大記憶體的目的呢?答案肯定是不可以,因為目前主板地址總線最寬只有48bit,4位是顏色位,就只剩44位了,所以受限于目前的硬體,ZGC最大只能支持16T的記憶體,JDK13就把最大支持堆記憶體從4T擴大到了16T,

ZGC tuning

啟用ZGC比較簡單,設定JVM引數即可:-XX:+UnlockExperimentalVMOptions 「-XX:+UseZGC」,調優也并不難,因為ZGC調優引數并不多,遠不像CMS那么復雜,它和G1一樣,可以調優的引數都比較少,大部分作業JVM能很好的自動完成,下圖所示是ZGC可以調優的引數:

 

image.png

 

下面對部分引數進行更加詳細的說明,

UseNUMA
ZGC默認是開啟支持NUMA的,不過,如果JVM探測到系統系結的是CPU子集,就會自動禁用NUMA,我們可以通過引數-XX:+UseNUMA顯示啟動,或者通過引數-XX:-UseNUMA顯示禁用,如果運行在NUMA服務器上,并且設定-XX:+UseNUMA,那對性能提升是顯而易見的,

UseLargePages
配置ZGC使用large page通常就會得到更好的性能,比如在吞吐量、延遲、啟動時間等方面,而且沒有明顯的缺點,除了配置程序復雜一點,因為它需要root權限,這也是默認并沒有開啟使用large page的原因,

ConcGCThreads
ZGC是一個并發垃圾收集器,那么并發GC執行緒數就非常重要了,如果設定并發GC執行緒數越多,意味著應用執行緒數就會越少,這肯定是非常不利于應用系統穩定運行的,這個引數ZGC能自動設定,如果沒有十足的把握,最好不要設定這個引數,

ParallelGCThreads
這是個并行執行緒數,與上一個引數ConcGCThreads有所不同,ConcGCThreads表示GC執行緒和應用執行緒「并發」執行時GC執行緒數量,而ParallelGCThreads表示GC時STW階段的「并行」GC執行緒數量(例如第一階段的Root掃描),這時候只有GC執行緒,沒有應用執行緒,筆者這里解釋了JVM中「并發和并行的區別」,也是JVM中比較容易理解錯誤的地方,

ZUncommit
掌握這個引數之前,我們先說一下JVM申請以及回收記憶體的行為,以前的垃圾回收器比如ParallelOldGC和CMS,只要JVM申請過的記憶體,即使發生了GC回收了很多記憶體空間,JVM也不會把這些記憶體歸還給作業系統,這就會導致top命令中看到的RSS只會越來越高,而且一般都會超過Xmx的值(參考文章:),

不過,默認情況下,ZGC是會把不再使用的記憶體歸還給作業系統的,這對于那些比較注意記憶體占用情況的應用和服務器來說,是很有用的,這種行為可以通過JVM引數-XX:-ZUncommit關閉,不過,無論怎么歸還,JVM至少會保留Xms引數指定的記憶體大小,這就是說,當Xmx和Xms一樣大的時候,這個引數就不起作用了,

和這個引數一起起作用的還有另一個引數:-「XX:ZUncommitDelay=sec」,默認300秒,這個引數表示不再使用的記憶體最多延遲多長時間才會被歸還給作業系統,因為不再使用的記憶體不應該立即歸還給作業系統,這樣會造成頻繁的歸還和申請行為,所以通過這個引數來控制不再使用的記憶體需要經過多久的時間才歸還給作業系統,

Change Log

接下來,我們看一下從JDK11到JDK15這5個版本,ZGC都迭代了哪些特性:

JDK 15 (under development)

  • Improved NUMA awareness
  • Support for Class Data Sharing (CDS)
  • Support for placing the heap on NVRAM

JDK 14

  • macOS support (JEP 364)
  • Windows support (JEP 365)
  • Support for tiny/small heaps (down to 8M)
  • Support for JFR leak profiler
  • Support for limited and discontiguous address space
  • Parallel pre-touch (when using -XX:+AlwaysPreTouch)
  • Performance improvements (clone intrinsic, etc)
  • Stability improvements

JDK 13

  • Increased max heap size from 4TB to 16TB
  • Support for uncommitting unused memory (JEP 351)
  • Support for -XX:SoftMaxHeapSIze
  • Support for the Linux/AArch64 platform
  • Reduced Time-To-Safepoint

JDK 12

  • Support for concurrent class unloading
  • Further pause time reductions

JDK 11

  • Initial version of ZGC
  • Does not support class unloading (using -XX:+ClassUnloading has no effect)

 

看完三件事??

如果你覺得這篇內容對你還蠻有幫助,我想邀請你幫我三個小忙:

  1. 點贊,轉發,有你們的 『點贊和評論』,才是我創造的動力,

  2. 關注公眾號 『 java爛豬皮 』,不定期分享原創知識,

  3. 同時可以期待后續文章ing??

 



轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/136231.html

標籤:Java

上一篇:Java8 Stream原始碼分析

下一篇:重大事故!IO問題引發線上20臺機器同時崩潰

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more