Java記憶體區域有哪些構成?

来源:https://www.cnblogs.com/xiaoniuhululu/archive/2023/01/09/17036803.html
-Advertisement-
Play Games

作者:小牛呼嚕嚕 | https://xiaoniuhululu.com 電腦內功、JAVA底層、面試相關資料等更多精彩文章在公眾號「小牛呼嚕嚕 」 大家好,我是呼嚕嚕,這次我們一起來看看Java記憶體區域,本文 基於HotSpot 虛擬機,JDK8, 乾貨滿滿 前言 Java 記憶體區域, 也叫運行 ...


目錄

作者:小牛呼嚕嚕 | https://xiaoniuhululu.com
電腦內功、JAVA底層、面試相關資料等更多精彩文章在公眾號「小牛呼嚕嚕

大家好,我是呼嚕嚕,這次我們一起來看看Java記憶體區域,本文 基於HotSpot 虛擬機,JDK8, 乾貨滿滿

前言

Java 記憶體區域, 也叫運行時數據區域、記憶體區域、JVM記憶體模型,和 Java 虛擬機(JVM)的運行時區域相關,是指 JVM運行時將數據分區域存儲,強調對記憶體空間的劃分。
經常與Java記憶體模型(JMM)混淆,其定義了程式中各個變數的訪問規則,即在虛擬機中將變數存儲到記憶體和從記憶體中取出變數這樣的底層細節。
JVM並不是只有唯一版本的,在Java發展歷史中,有許多優秀的Java虛擬機,其中目前大家最熟悉的就是HotSpot虛擬機,什麼你不知道?

我們去Oracle官網,下載JDK,其自帶的虛擬機,就是HotSpot。

HotSpot VM的最大特色:熱點代碼探測,其可以通過執行計數器,找出最具有編譯價值的代碼,然後通知JIT編譯器進行編譯,通過編譯器和解釋器的協同合作,在最優程式響應時間和最佳執行性能中取得平衡。

簡單介紹一下,上圖的主要組成部分:

  • 類載入器系統:主要用於子系統將編譯好的.class文件載入到JVM中,瞭解見:類載入器
  • 執行引擎:包括即時編譯器和垃圾回收器,即時編譯器將Java位元組碼編譯成具體的機器碼,垃圾回收器用於回收在運行過程中不再使用的對象
  • 本地庫介面:用於調用操作系統的本地方法庫,完成具體的指令操作
  • 運行時數據區:用於儲存在JVM運行過程中產生的數據,不同的虛擬機在記憶體分配上也略有差異,但總體來說都遵循《Java虛擬機規範》。在《Java虛擬機規範》中規定了五種虛擬機運行時數據區,他們分別為:程式計數器、Java虛擬機棧、本地方法棧、本地方法區、堆 以及方法區。下文我們以此圖為基準,詳細地分析各個部分,慢慢道來

Java 記憶體區域

程式計數器

程式計數器(Program Counter Register)是用於存放下一條指令所在單元地址的一塊記憶體,在虛擬機的規範里,位元組碼解析器的工作是通過改變這個計數器的值來選取下一條需要執行的位元組碼指令,分支、迴圈、跳轉、異常處理、線程恢復等基礎功能都需要依賴這個計數器來完成。

我們來對Java中class文件反編譯:

在JVM邏輯上規定,程式計數器是一塊較小的記憶體空間,可以看作是當前線程所執行位元組碼的行號指示器,PC寄存器,也叫"程式計數器",其是CPU中寄存器的一種,偏硬體概念

由於程式計數器保存了 下一條指令要執行地址,所以在JVM中,執行指令的一般過程:執行引擎會從 程式計數器中獲得下一條指令的地址,拿到其對應的操作指令,對其進行執行,當該指令結束,位元組碼解釋器根據pc寄存器里的值選取下一條指令並修改pc寄存器裡面的值,達到執行下一條指令的目的,周而複始直至程式結束。

位元組碼解釋器可以拿到所有的位元組碼指令執行順序,而程式計數器只是為了記錄當前執行的位元組碼指令地址,防止線程切換找不到下一條指令地址

我們知道操作系統中線程是由CPU調度來執行指令的,JVM的多線程是通過CPU時間片輪轉來實現的,某個線程在執行的過程中可能會因為時間片耗盡而掛起。當它再次獲取時間片時,需要從掛起的地方繼續執行。在JVM中,通過程式計數器來記錄程式的位元組碼執行位置。

執行程式在單線程情況下還好,但在多線程的情況下:線程在執行的指令時,CPU可能切換線程,去另一個更緊急的指令,執行完再繼續執行先前的指令。特別是單核CPU的情況下,CPU會頻繁的切換線程,"同時"執行多個任務。為了CPU切換線程後,依舊能恢復到先前指令執行的位置,這就需要每個線程有自己獨立的程式計數器,互不影響。我們可以發現程式計數器是線程私有的,每條線程都有一個程式計數器。

程式計數器是java虛擬機規範中唯一一個沒有規定任何OutofMemeryError(記憶體泄漏)的區域,它的生命周期隨著線程的創建而創建,隨著線程的結束而死亡。因為當前線程正在執行Java中的方法,程式計數器記錄的就是正在執行虛擬機位元組碼指令的地址,如果是Native方法,這個計數器就為空(undefined)

PC寄存器(程式計數器)與JVM中的程式計數器還是有所區別的:

  1. PC寄存器永遠指向下一條待執行指令的記憶體地址(永遠不會為undefined),並且在程式開始執行前,將程式指令序列的起始地址,即程式的第一條指令所在的記憶體單元地址送入PC, CPU按照PC的指示從記憶體讀取第一條指令(取指)
  2. 當執行指令時,CPU會自動地修改PC的內容,即每執行一條指令PC增加一個量,這個量等於指令所含的位元組數(指令位元組數),使PC總是指向下一條將要取指的指令地址。
  3. 由於大多數指令都是按順序來執行的,所以修改PC的過程通常只是簡單的對PC 加“指令位元組數”。當程式轉移時,轉移指令執行的最終結果就是要改變PC的值,此PC值就是轉去的目標地址。處理器總是按照PC指向,取指、解碼、執行,以此實現了程式轉移。

虛擬機棧

虛擬機棧(JVM Stacks),和數據結構上的棧類似,先進後出。其與程式計數器一樣,也是線程私有的,其生命周期和線程相同,隨著線程的創建而創建,隨著線程的死亡而死亡。

虛擬機棧描述的是Java方法執行的記憶體模型:每個方法在執行的同時都會創建一個棧幀,用於存儲局部變數表、操作數棧、動態連接、方法出口等信息。棧幀在虛擬機棧中入棧到出棧(順序: 先進後出)的過程,其實就對應Java中方法的調用至執行完成的過程

棧幀是用於支持虛擬機進行方法調用和方法執行的數據結構,它是虛擬機運行時數據區中的虛擬機棧的棧元素,每個棧幀存儲了方法的變數表、操作數棧、動態連接和方法返回等信息。

其中:

  1. 在當前活動線程中,只有位於棧頂的幀才是有效的,稱為當前棧幀。正在執行的方法稱為當前方法,棧幀是方法運行的基本結構。在執行引擎運行時,所有指令都只能針對當前棧幀進行操作。
  2. 方法調用的數據需要通過棧進行傳遞,每一次方法調用都會有一個對應的棧幀被壓入棧中,每一個方法調用結束後,都會有一個棧幀被彈出。
  3. 每個棧幀包含四個區域:局部變數表、操作數棧、動態連接、返回地址
  4. 在《Java虛擬機規範》中,對這個記憶體區域規定了兩類異常狀況:
  • 如果線程請求的棧深度大於虛擬機所允許的深度,將拋出StackOverflowError異常
  • 如果Java虛擬機棧容量可以動態擴展,當棧嘗試擴展時無法申請到足夠的記憶體或為一個新線程初始化JVM棧時沒有足夠的記憶體時會拋出OutOfMemoryError異常。《Java虛擬機規範》明確允許Java虛擬機實現自行選擇是否支持棧的動態擴展HotSpot虛擬機是選擇不支持擴展,所以HotSpot虛擬機線上程運行時是不會因為擴展而導致OutOfMemoryError(記憶體溢出)的異常

我們下麵主要介紹一下棧幀的結構:

  1. 局部變數表

局部變數表:是存放方法參數和局部變數的區域,主要存放了編譯期可知的各種數據類型(boolean、byte、char、short、int、float、long、double)、對象引用(reference 類型,它不同於對象本身,可能是一個指向對象起始地址的引用指針,也可能是指向一個代表對象的句柄或其他與此對象相關的位置)

我們知道局部變數沒有賦初始值是不能使用的,而全局變數是放在堆的,有兩次賦值的階段,一次在類載入的準備階段,賦予系統初始值;另外一次在類載入的初始化階段,賦予代碼定義的初始值。拓展見:類載入器

局部變數表的容量以 Variable Slot(變數槽)為最小單位,每個變數槽都可以存儲 32 位長度的記憶體空間.基本類型數據以及引用和 returnAddress(返回地址)占用一個變數槽,long 和 double 需要兩個

在方法執行時,虛擬機使用局部變數表完成參數值到參數變數列表的傳遞過程的,如果執行的是實例方法,那局部變數表中第 0 位索引的 Slot 預設是用於傳遞方法所屬對象實例的引用(在方法中可以通過關鍵字 this 來訪問到這個隱含的參數)其餘參數則按照參數表順序排列,占用從 1 開始的局部變數 Slot。關鍵字this詳解
我們可以寫個例子驗證一下

public class Test {
    void fun(){
    }
}

javac -g:vars Test.java生成Test.class文件,一定要加參數-g:vars,不然反編譯時,無法顯示局部變數表LocalVariableTable
我們接著反編譯一下:

javap -v Test


Classfile /D:/GiteeProjects/study-java/study/src/com/company/test3/Test.class
  Last modified 2022-11-20; size 261 bytes
  MD5 checksum 72c7d1fcc5d83dd6fc82c43ae55f2b34
public class com.company.test3.Test
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #3.#11         // java/lang/Object."<init>":()V
   #2 = Class              #12            // com/company/test3/Test
   #3 = Class              #13            // java/lang/Object
   #4 = Utf8               <init>
   #5 = Utf8               ()V
   #6 = Utf8               Code
   #7 = Utf8               LocalVariableTable
   #8 = Utf8               this
   #9 = Utf8               Lcom/company/test3/Test;
  #10 = Utf8               fun
  #11 = NameAndType        #4:#5          // "<init>":()V
  #12 = Utf8               com/company/test3/Test
  #13 = Utf8               java/lang/Object
{
  public com.company.test3.Test();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lcom/company/test3/Test;

  void fun();
    descriptor: ()V
    flags:
    Code:
      stack=0, locals=1, args_size=1
         0: return
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       1     0  this   Lcom/company/test3/Test; //!!!可以看出this在Slot的第0位!!!
}
  1. 操作數棧

操作數棧 主要用於存放方法執行過程中產生的中間計算結果或者臨時變數,通過變數的入棧、出棧等操作來執行計算。
在方法執行的過程中,會有各種位元組碼指令往操作數棧中寫入和提取內容,也就是出棧和入棧操作。我們前文說的JVM執行引擎,是基於棧的執行引擎, 其中的棧指的就是操作數棧

  1. 動態鏈接

每個棧幀都保存了 一個 可以指向當前方法所在類的 運行時常量池, 目的是: 當前方法中如果需要調用其他方法的時候, 能夠從運行時常量池中找到對應的符號引用, 然後將符號引用轉換為直接引用,然後就能直接調用對應方法, 這就是動態鏈接。本質就是,在方法運行時將符號引用轉為調用方法的直接引用,這種引用轉換的過程具備動態性
不是所有方法調用都需要動態鏈接的, 有一部分符號引用會在 類載入階段, 將符號引用轉換為直接引用, 這部分操作稱之為: 靜態解析. 就是編譯期間就能確定調用的版本, 包括: 調用靜態方法, 調用實例的私有構造器, 私有方法, 父類方法

  1. 返回地址

Java 方法有兩種返回方式:

  • 正常退出,即正常執行到任何方法的返回位元組碼指令,如 return等;
  • 異常退出

無論何種退出情況,都將返回至方法當前被調用的位置。方法退出的過程相當於彈出當前棧幀
我們可以發現:棧幀隨著方法調用而創建,隨著方法結束而銷毀。無論方法正常完成還是異常完成都算作方法結束.

本地方法棧

本地方法棧(Native Method Stack):是線程私有的,其與虛擬機棧的作用基本是一樣的,有點區別的是:虛擬機棧是服務Java方法的,而本地方法棧是為虛擬機調用Native方法服務的,通過 JNI (Java Native Interface) 直接調用本地 C/C++ 庫,不再受JVM控制。

JNI 類本地方法最著名的應該是 System.currentTimeMillis() ,JNI使 Java 深度使用操作系統的特性功能,復用非 Java 代碼。 當大量本地方法出現時,勢必會削弱 JVM 對系統的控制力

本地方法被執行的時候,在本地方法棧也會創建一個棧幀,用於存放該本地方法的局部變數表、操作數棧、動態鏈接、出口信息。方法執行完畢後相應的棧幀也會出棧並釋放記憶體空間。與虛擬機棧一樣,本地方法棧區域也會拋出StackOverflowErrorOutOfMemoryError

另外在Java虛擬機規範中對於本地方法棧沒有特殊的要求,虛擬機可以自由的實現它,因此在HotSpot虛擬機直接把本地方法棧和虛擬機棧合二為一了。因此對於HotSpot來說,-Xoss參數(設置 本地方法棧大小)雖然存在,但實際上是沒有任何效果的,棧容量只能由-Xss參數來設定。

堆(Heap)是Java虛擬機所管理的最大的一塊記憶體區域,是被所有線程共用的,Java堆唯一的目的就是存放對象實例幾乎所有的對象實例都在堆上分配記憶體,但是隨著JIT編譯器的發展和逃逸分析技術的逐漸成熟,棧上分配、線程本地分配緩存(TLAB)也可以存放對象實例

Java虛擬機規範規定,Java堆可以處在物理上不連續的記憶體空間中,只要邏輯上連續即可,當前主流的虛擬機都是按照可擴展來實現的(通過 -Xmx 和 -Xms 控制)。如果在堆中沒有記憶體完成實例分配,並且堆也無法再擴展時,將會拋出 OutOfMemoryError 異常。

方法區

方法區(Methed Area)用於存儲已被虛擬機載入的類信息、常量、靜態變數、即時編譯後的代碼等數據。其是所有線程共用的記憶體區域。

在Java 虛擬機規範把方法區描述為堆的一個邏輯部分,但是它卻有一個別名叫做 Non-Heap(非堆),與 Java 堆區分開來。

方法區是JVM規範的一個概念定義,並不是一個具體的實現,由於Java虛擬機對於方法區的限制是非常寬鬆的,因此也就導致了不同的虛擬機上方法區有不同的表現,我們還是以HotSpot虛擬機為例:

  • 在JDK8前,HotSpot 虛擬機對Java虛擬機規範中方法區的實現方式是永久代
  • 在JDK8及其以後,HotSpot 虛擬機對Java虛擬機規範中方法區的實現方式變成了元空間

網上許多文章喜歡拿"永久代"或者"元空間" 來代替方法區,但本質上兩者並不等價。方法區是Java虛擬機規範的概念,"永久代"或者"元空間"是方法區的2中實現方式

方法區在JDK7之前是一塊單獨的區域,HotSpot虛擬機的設計團隊把GC分代收集擴展到了方法區。這樣HotSpot的垃圾收集器就可以向管理Java堆一樣管理這部分記憶體。但是對於其它虛擬機(如BEA JRockit、IBM J9等)來說其實是不存在永久代的概念的。

HotSpot的團隊顯然也意識到了,用永久代來實現方法區並不是一個好主意:

  1. 字元串存在永久代中,容易出現性能問題和記憶體溢出
  2. 類及方法的信息等比較難確定其大小,因此對於永久代的大小指定比較困難,太小容易出現永久代溢出,太大則容易導致老年代溢出。
  3. 永久代會為 GC 帶來不必要的複雜度,並且回收效率偏低。

因此,在JDK1.8中完全廢除了“永久代”,使用元空間替代了永久代,其他內容移至元空間,元空間直接在本地記憶體分配。

當方法區無法滿足記憶體分配需求時,將拋出OutOfMemoryError異常。元空間是使用直接記憶體實現的,我們下文再詳細說。

Java記憶體區域大致就這些了,下麵我們再補充幾個比較讓人迷惑的概念

字元串常量池

字元串屬於引用數據類型,但是可以說字元串是Java中使用頻繁的一種數據類型。因此,為了節省程式記憶體,提高性能,Java的設計者開闢了一塊叫字元串常量池的區域,用來存儲這些字元串,避免字元串的重覆創建。字元串常量池是所有類公用的一塊空間,在一個虛擬機中只有一塊常量池區域。

在類載入完成,經過驗證,準備階段之後在堆中生成字元串對象實例,然後將該字元串對象實例的引用值存到字元串常量池中(這裡描述指的是JDK7及以後的HotSpot虛擬機)。 在HotSpot虛擬機中字元串常量池是通過一個StringTable類來實現的。它是一個哈希表,裡面存的是字元串引用

在JDK7以前,字元串常量池在方法區(永久代)中,此時常量池中存放的是字元串對象。而在JDK7及其以後中,字元串常量池從方法區遷移到了堆記憶體,同時將字元串對象存到了堆記憶體,只在字元串常量池中存入了字元串對象的引用。

在JDK7 就已經開始了HotSpot 的永久代的移除工作,主要由於永久代的 GC 回收效率太低。等到JDK 8 的時候,永久代被徹底移除了
Java 程式中通常會有大量的被創建的字元串等待回收,將字元串常量池放到堆中,能夠更高效及時地回收字元串記憶體。

運行時常量池

運行時常量池(Runtime Constant Pool)是方法區的一部分。我們知道Class 文件中除了有類的版本、欄位、方法、介面等常見描述信息外,但還有一項信息是常量池(Constant Pool Table),用於存放編譯期生成的各種字面量,符號引用還有翻譯出來的直接引用,這部分內容將在類載入後進入方法區的運行時常量池中存放。因此,每一個類都會有一個運行時常量池

因為Java語言並不要求常量一定在編譯期間才能生成。也就是並非預置入Class文件常量池中的內容才能進入運行時常量池,運行期間也可以將新的常量放入常量池中,運行時常量池另外一個重要特征是具備動態性

既然運行時常量池是方法區的一部分,自然受到方法區記憶體的限制,當常量池無法再申請到記憶體時會拋出 OutOfMemoryError 異常。

直接記憶體

JDK 8 版本之後 永久代已被元空間取代,元空間使用的就是直接記憶體。直接記憶體(Direct Memory)並不是Java虛擬機運行時數據區的一部分,也不是 Java 虛擬機規範中定義的記憶體區域。

在 JDK 1.4 中新加入了 NIO,引入了一種基於通道(Channel)與緩衝區(Buffer)的 I/O 方式,它可以使用 Native 函數庫直接分配堆外記憶體,然後通過一個存儲在 Java 堆中的 DirectByteBuffer 對象作為這塊記憶體的引用進行操作。這樣能在一些場景中顯著提高性能,因為避免了在 Java 堆和 Native 堆中來回覆制數據。

顯然,本機直接記憶體的分配不會受到 Java 堆大小的限制,但是,既然是記憶體,肯定還是會受到本機總記憶體(包括 RAM 以及 SWAP 區或者分頁文件)大小以及處理器定址空間的限制。伺服器管理員在配置虛擬機參數時,會根據實際記憶體設置 -Xmx 等參數信息,但經常忽略直接記憶體,使得各個記憶體區域總和大於物理記憶體限制(包括物理的和操作系統級的限制),從而導致動態擴展時出現 OutOfMemoryError 異常。

小結

  1. 線程私有區域(包括 程式計數器, 虛擬機棧, 本地方法棧),生命周期跟隨線程的啟動而創建,隨線程的結束而銷毀
  2. 線程共用區域(包括 方法區 和 堆 ),生命周期跟隨虛擬機的啟動而創建,隨虛擬機的關閉而銷毀

參考資料:
《深入理解 Java 虛擬機:JVM 高級特性與最佳實踐》
《On Java 8》
https://www.cnblogs.com/newAndHui/p/11168791.html
https://blog.csdn.net/qq_20394285/article/details/104673913
https://www.cnblogs.com/czwbig/p/11127124.html


本篇文章到這裡就結束啦,很感謝你能看到最後,如果覺得文章對你有幫助,別忘記關註我!更多精彩的文章
電腦內功、JAVA底層、面試相關資料等更多精彩文章在公眾號「小牛呼嚕嚕


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 前段時間一直使用到word文檔轉pdf或者pdf轉word,尋思著用Java應該是可以實現的,於是花了點時間寫了個文件轉換工具 源碼weloe/FileConversion (github.com) 主要功能就是word和pdf的文件轉換,如下 pdf 轉 word pdf 轉 圖片 word ...
  • 2023-01-09 一、Mybatis核心配置文件概述及根標簽 1、核心配置文件的概述(即“mybatis-config.xml”) MyBatis的配置文件包含了會深深影響MyBatis行為的設置和屬性信息。 2、標簽 (1)configuration(配置) (2)properties(屬性) ...
  • Redis 數據結構-簡單動態字元串 無邊落木蕭蕭下,不盡長江滾滾來。 1、簡介 Redis 之所以快主要得益於它的數據結構、操作記憶體資料庫、單線程和多路 I/O 復用模型,進一步窺探下它常見的五種基本數據的底層數據結構。 Redis 常見數據類型對應的的底層數據結構。 String:簡單動態字元串 ...
  • 前言 今天給大家介紹的是Python爬蟲批量下載評書音頻並保存本地,在這裡給需要的小伙伴們代碼,並且給出一點小心得。 首先是爬取之前應該儘可能偽裝成瀏覽器而不被識別出來是爬蟲,基本的是加請求頭,但是這樣的純文本數據爬取的人會很多,所以我們需要考慮更換代理IP和隨機更換請求頭的方式來對評書精選音頻進行 ...
  • 互聯網世界里最流行的開源關係型資料庫之一就是MySQL/MariaDB了,由於高度的相似,故而直接使用mysql統一指稱。 ...
  • 項目地址:https://github.com/pikeduo/TXTReader PyQt5中文手冊:https://maicss.gitbook.io/pyqt-chinese-tutoral/pyqt5/ QtDesigner學習地址:https://youcans.blog.csdn.net ...
  • 14. 最長公共首碼 題目描述 編寫一個函數來查找字元串數組中的最長公共首碼。 如果不存在公共首碼,返回空字元串 ""。 方法 暴力演算法 先判斷字元串數組是否有為空,為空直接返回空 令第一個字元串作為基準進行比較 設置一個長度,作為最後最長公共首碼的長度 迴圈判斷,選取最小長度 代碼 package ...
  • 2023-01-09 一、在IDEA中創建Maven版的web工程 (1)步驟: ①創建一個maven模塊,命名為“maven_web_end”,之後需要創建web工程的目錄。在“maven_web_end.src.main”下創建“webapp”文件夾(命名必須為webapp,否則識別不了);在“ ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...