主頁 > 後端開發 > 類加載器詳解

類加載器詳解

2020-10-08 20:12:53 後端開發

2 章 類加載器詳解

微信搜 : 全堆疊小劉 ,獲取 文章pdf版本

1、記憶體結構概述

如果自己想手寫一個Java虛擬機的話,主要考慮哪些結構呢?

  1. 類加載器
  2. 執行引擎

完整框圖:

2、類加載子系統

類加載器子系統作用

  1. 類加載器子系統負責從檔案系統或者網路中加載Class檔案,class檔案在檔案開頭有特定的檔案標識,
  2. ClassLoader只負責class檔案的加載,至于它是否可以運行,則由Execution Engine決定,
  3. 加載的類資訊存放于一塊稱為方法區的記憶體空間,除了類的資訊外,方法區中還會存放運行時常量池資訊,可能還包括字串字面量和數字常量(這部分常量資訊是Class檔案中常量池部分的記憶體映射)

class --> Java.lang.Class

  1. class file存在于本地硬碟上,可以理解為設計師畫在紙上的模板,而最終這個模板在執行的時候是要加載到JVM當中來根據這個檔案實體化出n個一模一樣的實體,
  2. class file加載到JVM中,被稱為DNA元資料模板,放在方法區
  3. 在.class檔案–>JVM–>最終成為元資料模板,此程序就要一個運輸工具(類裝載器Class Loader),扮演一個快遞員的角色,

3、類加載程序

3.1、類加載程序概述

  • 看代碼
public class HelloLoader {
    public static void main(String[] args) {
        System.out.println("謝謝ClassLoader加載我....");
        System.out.println("你的大恩大德,我下輩子再報!");
    }
}
  • 它的加載程序是怎么樣的呢?

    • 執行 main() 方法(靜態方法)就需要先加載承載類 HelloLoader
    • 加載成功,則進行鏈接、初始化等操作,完成后呼叫 HelloLoader 類中的靜態方法 main
    • 加載失敗則拋出例外

  • 完整的流程圖如下所示: *加載 --> 鏈接(驗證 --> 準備 --> 決議) --> 初始化

3.2、加載階段

加載流程

  1. 通過一個類的全限定名獲取定義此類的二進制位元組流
  2. 將這個位元組流所代表的靜態存盤結構轉化為 方法區的運行時資料結構
  3. 在記憶體中生成一個代表這個類的java.lang.Class物件,作為方法區這個類的各種資料的訪問入口

加載class檔案的方式

  1. 從本地系統中直接加載
  2. 通過網路獲取,典型場景:Web Applet
  3. 從zip壓縮包中讀取,成為日后jar、war格式的基礎
  4. 運行時計算生成,使用最多的是:動態代理技術
  5. 由其他檔案生成,典型場景:JSP應用從專有資料庫中提取.class檔案,比較少見
  6. 從加密檔案中獲取,典型的防Class檔案被反編譯的保護措施

3.3、鏈接階段

  • *鏈接分為三個子階段:驗證 --> 準備 --> 決議

3.3.1、驗證(Verify)

驗證

  1. 目的在于確保Class檔案的位元組流中包含資訊符合當前虛擬機要求,保證被加載類的正確性,不會危害虛擬機自身安全
  2. 主要包括四種驗證,檔案格式驗證,元資料驗證,位元組碼驗證,符號參考驗證,

舉例

  • 使用 BinaryViewer 查看位元組碼檔案,其開頭均為 CAFE BABE ,如果出現不合法的位元組碼檔案,那么將會驗證不通過

3.3.2、準備(Prepare)

準備

  1. 為類變數分配記憶體并且設定該類變數的默認初始值,即零值
  2. 這里不包含用final修飾的static,因為final在編譯的時候就會分配好了默認值,準備階段會顯式初始化
  3. 注意:這里不會為實體變數分配初始化,類變數會分配在方法區中,而實體變數是會隨著物件一起分配到Java堆中

舉例

  • 代碼:變數a在準備階段會賦初始值,但不是1,而是0,在初始化階段會被賦值為 1
public class HelloApp {
    private static int a = 1;

    public static void main(String[] args) {
        System.out.println(a);
    }
}

3.3.3、決議(Resolve)

決議

  1. 將常量池內的符號參考轉換為直接參考的程序
  2. 事實上,決議操作往往會伴隨著JVM在執行完初始化之后再執行
  3. 符號參考就是一組符號來描述所參考的目標,符號參考的字面量形式明確定義在《java虛擬機規范》的class檔案格式中,直接參考就是直接指向目標的指標、相對偏移量或一個間接定位到目標的句柄
  4. 決議動作主要針對類或介面、欄位、類方法、介面方法、方法型別等,對應常量池中的CONSTANT Class info、CONSTANT Fieldref info、CONSTANT Methodref info等

符號參考

  • 反編譯 class 檔案后可以查看符號參考

3.4、初始化階段

初始化階段

  1. 初始化階段就是執行類構造器方法 <clinit>()</clinit> 的程序
  2. 此方法不需定義,是javac編譯器自動收集類中的所有類變數的賦值動作和靜態代碼塊中的陳述句合并而來,也就是說, 當我們代碼中包含static變數的時候,就會有clinit方法
  3. ** <clinit>()</clinit> 方法中的指令按陳述句在源檔案中出現的順序執行**
  4. <clinit>()</clinit>不同于類的構造器,(關聯:構造器是虛擬機視角下的 <init>()</init>
  5. 若該類具有父類,JVM會保證子類的 <clinit>()</clinit> 執行前,父類的 <clinit>()</clinit> 已經執行完畢
  6. 虛擬機必須保證一個類的 <clinit>()</clinit> 方法在多執行緒下被同步加鎖

IDEA 中安裝 JClassLib 插件

在 IDEA 中安裝 JClassLib 插件后,重啟 IDEA 生效

  • 選中對應的 Java 類檔案,注意:不是位元組碼檔案~!
  • 點擊【View --> Show Bytecode With jclasslib】即可查看反編譯后的代碼

當我們代碼中包含static變數的時候,就會有clinit方法

示例 1:無 static 變數

  • 代碼
public class ClinitTest {
    private int a = 1;

    public static void main(String[] args) {
        int b = 2;
    }
}
  • 并沒有生成 clinit 方法

示例 2:有 static 變數

  • 代碼
public class ClinitTest {

    private int a = 1;
    private static int c = 3;

    public static void main(String[] args) {
        int b = 2;
    }

}
  • 在 clinit 方法中初始化靜態變數的值為 3

構造器方法中指令按陳述句在源檔案中出現的順序執行

示例 1

  • 代碼:
public class ClassInitTest {
    private static int num = 1;
    private static int number = 10;

    static {
        num = 2;
        number = 20;
        System.out.println(num);

    }

    public static void main(String[] args) {
        System.out.println(ClassInitTest.num);
        System.out.println(ClassInitTest.number);
    }
}
  • 靜態變數 number 的值變化程序如下
    • 準備階段時:0
    • 執行靜態變數初始化:10
    • 執行靜態代碼塊:20

示例 1

  • 代碼
public class ClassInitTest {
   private static int num = 1;

   static{
       num = 2;
       number = 20;
       System.out.println(num);

   }

   private static int number = 10;

    public static void main(String[] args) {
        System.out.println(ClassInitTest.num);
        System.out.println(ClassInitTest.number);
    }
}
  • 靜態變數 number 的值變化程序如下
    • 準備階段時:0
    • 執行靜態代碼塊:20
    • 執行靜態變數初始化:10

構造器是虛擬機視角下的 <init>()</init>

  • 代碼
public class ClinitTest {

    private int a = 1;
    private static int c = 3;

    public static void main(String[] args) {
        int b = 2;
    }

    public ClinitTest(){
        a = 10;
        int d = 20;
    }

}
  • 在構造器中:
    • 先將類變數 a 賦值為 10
    • 再將區域變數賦值為 20

若該類具有父類,JVM會保證子類的 <clinit>()</clinit> 執行前,父類的 <clinit>()</clinit> 已經執行完畢

  • 代碼
public class ClinitTest1 {
    static class Father{
        public static int A = 1;
        static{
            A = 2;
        }
    }

    static class Son extends Father{
        public static int B = A;
    }

    public static void main(String[] args) {

        System.out.println(Son.B);
    }
}
  • 如上代碼,加載流程如下:
    • 首先,執行 main() 方法需要加載 ClinitTest1 類
    • 獲取 Son.B 靜態變數,需要加載 Son 類
    • Son 類的父類是 Father 類,所以需要先執行 Father 類的加載,再執行 Son 類的加載

虛擬機必須保證一個類的 <clinit>()</clinit> 方法在多執行緒下被同步加鎖

  • 代碼
public class DeadThreadTest {
    public static void main(String[] args) {
        Runnable r = () -> {
            System.out.println(Thread.currentThread().getName() + "開始");
            DeadThread dead = new DeadThread();
            System.out.println(Thread.currentThread().getName() + "結束");
        };

        Thread t1 = new Thread(r, "執行緒1");
        Thread t2 = new Thread(r, "執行緒2");

        t1.start();
        t2.start();
    }
}

class DeadThread {
    static {
        if (true) {
            System.out.println(Thread.currentThread().getName() + "初始化當前類");
            while (true) {

            }
        }
    }
}
  • 程式卡死,分析原因:
    • 兩個執行緒同時去加載 DeadThread 類,而 DeadThread 類中靜態代碼塊中有一處死回圈
    • 先加載 DeadThread 類的執行緒搶到了同步鎖,然后在類的靜態代碼塊中執行死回圈,而另一個執行緒在等待同步鎖的釋放
    • 所以無論哪個執行緒先執行 DeadThread 類的加載,另外一個類也不會繼續執行

4、類加載器的分類

4.1、類加載器概述

類加載器的分類

  1. JVM支持兩種型別的類加載器 ,分別為引導類加載器(Bootstrap ClassLoader)和自定義類加載器(User-Defined ClassLoader)
  2. 從概念上來講,自定義類加載器一般指的是程式中由開發人員自定義的一類類加載器,但是Java虛擬機規范卻沒有這么定義,而是 將所有派生于抽象類ClassLoader的類加載器都劃分為自定義類加載器
  3. 無論類加載器的型別如何劃分,在程式中我們最常見的類加載器始終只有3個,如下所示
  4. 這里的四者之間是包含關系,不是上層和下層,也不是子父類的繼承關系,

為什么 ExtClassLoader 和 AppClassLoader 都屬于自定義加載器

  • 規范定義:所有派生于抽象類ClassLoader的類加載器都劃分為自定義類加載器
  • ExtClassLoader 繼承樹

  • AppClassLoader 繼承樹

  • 代碼
    • 我們嘗試獲取引導類加載器,獲取到的值為 null ,這并不代表引導類加載器不存在, 因為引導類加載器右 C/C++ 語言,我們獲取不到
    • 兩次獲取系統類加載器的值都相同:sun.misc.Launcher$AppClassLoader@18b4aac2 ,這說明 *系統類加載器是全域唯一的
public class ClassLoaderTest {
    public static void main(String[] args) {

        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
        System.out.println(systemClassLoader);

        ClassLoader extClassLoader = systemClassLoader.getParent();
        System.out.println(extClassLoader);

        ClassLoader bootstrapClassLoader = extClassLoader.getParent();
        System.out.println(bootstrapClassLoader);

        ClassLoader classLoader = ClassLoaderTest.class.getClassLoader();
        System.out.println(classLoader);

        ClassLoader classLoader1 = String.class.getClassLoader();
        System.out.println(classLoader1);

    }
}

4.2、虛擬機自帶的加載器

4.2.1、啟動類加載器

啟動類加載器(引導類加載器,Bootstrap ClassLoader)

  1. 這個類加載使用C/C++語言實作的,嵌套在JVM內部
  2. 它用來加載Java的核心庫(JAVA_HOME/jre/lib/rt.jar、resources.jar或sun.boot.class.path路徑下的內容),用于提供JVM自身需要的類
  3. 并不繼承自java.lang.ClassLoader,沒有父加載器
  4. 加載擴展類和應用程式類加載器,并作為他們的父類加載器(當他倆的爹)
  5. 出于安全考慮,Bootstrap啟動類加載器只加載包名為java、javax、sun等開頭的類

4.2.2、擴展類加載器

擴展類加載器(Extension ClassLoader)

  1. Java語言撰寫,由sun.misc.Launcher$ExtClassLoader實作
  2. 派生于ClassLoader類
  3. 父類加載器為啟動類加載器
  4. 從java.ext.dirs系統屬性所指定的目錄中加載類別庫,或從JDK的安裝目錄的jre/lib/ext子目錄(擴展目錄)下加載類別庫,如果用戶創建的JAR放在此目錄下,也會自動由擴展類加載器加載

4.2.3、系統類加載器

應用程式類加載器(系統類加載器,AppClassLoader)

  1. Java語言撰寫,由sun.misc.LaunchersAppClassLoader實作
  2. 派生于ClassLoader類
  3. 父類加載器為擴展類加載器
  4. 它負責加載環境變數classpath或系統屬性java.class.path指定路徑下的類別庫
  5. 該類加載是程式中默認的類加載器,一般來說,Java應用的類都是由它來完成加載
  6. 通過classLoader.getSystemclassLoader()方法可以獲取到該類加載器

代碼舉例說明

  • 代碼
public class ClassLoaderTest1 {
    public static void main(String[] args) {

        System.out.println("**********啟動類加載器**************");

        URL[] urLs = sun.misc.Launcher.getBootstrapClassPath().getURLs();
        for (URL element : urLs) {
            System.out.println(element.toExternalForm());
        }

        ClassLoader classLoader = Provider.class.getClassLoader();
        System.out.println(classLoader);

        System.out.println("***********擴展類加載器*************");
        String extDirs = System.getProperty("java.ext.dirs");
        for (String path : extDirs.split(";")) {
            System.out.println(path);
        }

        ClassLoader classLoader1 = CurveDB.class.getClassLoader();
        System.out.println(classLoader1);

    }
}
  • System.out.println(classLoader); 輸出 null ,再次證明我們無法獲取到啟動類加載器
**********&#x542F;&#x52A8;&#x7C7B;&#x52A0;&#x8F7D;&#x5668;**************
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/lib/resources.jar
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/lib/rt.jar
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/lib/sunrsasign.jar
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/lib/jsse.jar
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/lib/jce.jar
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/lib/charsets.jar
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/lib/jfr.jar
file:/C:/Program%20Files/Java/jdk1.8.0_144/jre/classes
null
***********&#x6269;&#x5C55;&#x7C7B;&#x52A0;&#x8F7D;&#x5668;*************
C:\Program Files\Java\jdk1.8.0_144\jre\lib\ext
C:\WINDOWS\Sun\Java\lib\ext
sun.misc.Launcher$ExtClassLoader@7ea987ac

4.3、用戶自定義類加載器

為什么需要自定義類加載器?

在Java的日常應用程式開發中,類的加載幾乎是由上述3種類加載器相互配合執行的,在必要時,我們還可以自定義類加載器,來定制類的加載方式,那為什么還需要自定義類加載器?

  1. 隔離加載類
  2. 修改類加載的方式
  3. 擴展加載源
  4. 防止原始碼泄漏

如何自定義類加載器?

  1. 開發人員可以通過繼承抽象類java.lang.ClassLoader類的方式,實作自己的類加載器,以滿足一些特殊的需求
  2. 在JDK1.2之前,在自定義類加載器時,總會去繼承ClassLoader類并重寫loadClass()方法,從而實作自定義的類加載類,但是在JDK1.2之后已不再建議用戶去覆寫loadClass()方法,而是建議把自定義的類加載邏輯寫在findclass()方法中
  3. 在撰寫自定義類加載器時,如果沒有太過于復雜的需求,可以直接繼承URIClassLoader類,這樣就可以避免自己去撰寫findclass()方法及其獲取位元組碼流的方式,使自定義類加載器撰寫更加簡潔,

代碼示例

public class CustomClassLoader extends ClassLoader {
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {

        try {
            byte[] result = getClassFromCustomPath(name);
            if (result == null) {
                throw new FileNotFoundException();
            } else {
                return defineClass(name, result, 0, result.length);
            }
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }

        throw new ClassNotFoundException(name);
    }

    private byte[] getClassFromCustomPath(String name) {

        return null;
    }

    public static void main(String[] args) {
        CustomClassLoader customClassLoader = new CustomClassLoader();
        try {
            Class<?> clazz = Class.forName("One", true, customClassLoader);
            Object obj = clazz.newInstance();
            System.out.println(obj.getClass().getClassLoader());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

4.4、關于 ClassLoader

ClassLoader 類介紹

  • ClassLoader類,它是一個抽象類,其后所有的類加載器都繼承自ClassLoader(不包括啟動類加載器)

  • sun.misc.Launcher 它是一個java虛擬機的入口應用

獲取 ClassLoader 途徑

  • 獲取途徑:

  • 代碼示例:
public class ClassLoaderTest2 {
    public static void main(String[] args) {
        try {

            ClassLoader classLoader = Class.forName("java.lang.String").getClassLoader();
            System.out.println(classLoader);

            ClassLoader classLoader1 = Thread.currentThread().getContextClassLoader();
            System.out.println(classLoader1);

            ClassLoader classLoader2 = ClassLoader.getSystemClassLoader();
            System.out.println(classLoader2);

        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

null
sun.misc.Launcher$AppClassLoader@18b4aac2
sun.misc.Launcher$AppClassLoader@18b4aac2

5、雙親委派機制

5.1、雙親委派機制原理

雙親委派機制的原理

Java虛擬機對class檔案采用的是按需加載的方式,也就是說當需要使用該類時才會將它的class檔案加載到記憶體生成class物件,而且 加載某個類的class檔案時,Java虛擬機采用的是雙親委派模式,即把請求交由父類處理,它是一種任務委派模式

  1. 如果一個類加載器收到了類加載請求,它并不會自己先去加載,而是把這個請求委托給父類的加載器去執行;
  2. 如果父類加載器還存在其父類加載器,則進一步向上委托,依次遞回,請求最終將到達頂層的啟動類加載器;
  3. 如果父類加載器可以完成類加載任務,就成功回傳,倘若父類加載器無法完成此加載任務,子加載器才會嘗試自己去加載,這就是雙親委派模式,
  4. 父類加載器一層一層往下分配任務,如果子類加載器能加載,則加載此類,如果將加載任務分配至系統類加載器也無法加載此類,則拋出例外

5.2、雙親委派機制代碼示例

代碼示例

舉例 1 :

  • 代碼:我們自己建立一個 java.lang.String 類,寫上 static 代碼塊
package java.lang;

public class String {
    static{
        System.out.println("我是自定義的String類的靜態代碼塊");
    }
}
  • 在另外的程式中加載 String 類,看看加載的 String 類是 JDK 自帶的 String 類,還是我們自己撰寫的 String 類
public class StringTest {

    public static void main(String[] args) {
        java.lang.String str = new java.lang.String();
        System.out.println("hello,atguigu.com");

        StringTest test = new StringTest();
        System.out.println(test.getClass().getClassLoader());
    }
}
  • 程式并沒有輸出我們靜態代碼塊中的內容,可見仍然加載的是 JDK 自帶的 String 類

舉例 2 :

  • 代碼:在我們自己的 String 類中整個 main() 方法
package java.lang;

public class String {
    static{
        System.out.println("我是自定義的String類的靜態代碼塊");
    }

    public static void main(String[] args) {
        System.out.println("hello,String");
    }
}
  • 由于雙親委派機制找到的是 JDK 自帶的 String 類,在那個 String 類中并沒有 main() 方法

舉例 3 :

  • 代碼:在 java.lang 包下整個 ShkStart 類
package java.lang;

public class ShkStart {
    public static void main(String[] args) {
        System.out.println("hello!");
    }
}
  • 出于保護機制,java.lang 包下不允許我們自定義類

舉例 4 :

當我們加載jdbc.jar 用于實作資料庫連接的時候

  1. 首先我們需要知道的是 jdbc.jar是基于SPI介面進行實作的
  2. 所以在加載的時候,會進行雙親委派,最終從根加載器中加載 SPI核心類,然后再加載SPI介面類
  3. 接著在進行反向委托,通過執行緒背景關系類加載器進行實作類 jdbc.jar的加載,

5.3、雙親委派機制優勢

雙親委派機制的優勢

通過上面的例子,我們可以知道,雙親機制可以

  1. 避免類的重復加載
  2. 保護程式安全,防止核心API被隨意篡改
  3. 自定義類:java.lang.String 沒有屌用
  4. 自定義類:java.lang.ShkStart(報錯:阻止創建 java.lang開頭的類)

6、沙箱安全機制

  1. 自定義String類時:在加載自定義String類的時候會率先使用引導類加載器加載,而引導類加載器在加載的程序中會先加載jdk自帶的檔案(rt.jar包中java.lang.String.class),報錯資訊說沒有main方法,就是因為加載的是rt.jar包中的String類,
  2. 這樣可以保證對java核心源代碼的保護,這就是沙箱安全機制,

7、其他

如何判斷兩個class物件是否相同?

在JVM中表示兩個class物件是否為同一個類存在兩個必要條件:

  1. 類的完整類名必須一致,包括包名
  2. 加載這個類的ClassLoader(指ClassLoader實體物件)必須相同
  3. 換句話說,在JVM中,即使這兩個類物件(class物件)來源同一個Class檔案,被同一個虛擬機所加載,但只要加載它們的ClassLoader實體物件不同,那么這兩個類物件也是不相等的

對類加載器的參考

  1. JVM必須知道一個型別是由啟動加載器加載的還是由用戶類加載器加載的
  2. 如果一個型別是由用戶類加載器加載的,那么JVM會將這個類加載器的一個參考作為型別資訊的一部分保存在方法區中
  3. 當決議一個型別到另一個型別的參考的時候,JVM需要保證這兩個型別的類加載器是相同的

類的主動使用和被動使用

Java程式對類的使用方式分為:主動使用和被動使用,主動使用,又分為七種情況:

  1. 創建類的實體
  2. 訪問某個類或介面的靜態變數,或者對該靜態變數賦值
  3. 呼叫類的靜態方法
  4. 反射(比如:Class.forName("com.atguigu.Test"))
  5. 初始化一個類的子類
  6. Java虛擬機啟動時被標明為啟動類的類
  7. JDK7開始提供的動態語言支持:java.lang.invoke.MethodHandle實體的決議結果REF_getStatic、REF putStatic、REF_invokeStatic句柄對應的類沒有初始化,則初始化

除了以上七種情況,其他使用Java類的方式都被看作是對類的被動使用,都不會導致類的初始化,即不會執行初始化階段(不會呼叫 clinit() 方法和 init() 方法)

你只管學習,我來負責記筆記?? 關注公眾號! ,更多筆記,等你來拿,謝謝



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

標籤:Java

上一篇:第一章: 初始JVM

下一篇:華為OD(外包)社招技術二面,總結復盤

標籤雲
其他(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