主頁 > 後端開發 > Java 泛型,你了解型別擦除嗎?

Java 泛型,你了解型別擦除嗎?

2020-10-10 05:27:34 後端開發

作者:frank909
https://blog.csdn.net/briblue/article/details/76736356

泛型,一個孤獨的守門者,

大家可能會有疑問,我為什么叫做泛型是一個守門者,這其實是我個人的看法而已,我的意思是說泛型沒有其看起來那么深不可測,它并不神秘與神奇,泛型是 Java 中一個很小巧的概念,但同時也是一個很容易讓人迷惑的知識點,它讓人迷惑的地方在于它的許多表現有點違反直覺,

文章開始的地方,先給大家奉上一道經典的測驗題,

List<String> l1 = new ArrayList<String>();
List<Integer> l2 = new ArrayList<Integer>();

System.out.println(l1.getClass() == l2.getClass());

請問,上面代碼最終結果輸出的是什么?不了解泛型的和很熟悉泛型的同學應該能夠答出來,而對泛型有所了解,但是了解不深入的同學可能會答錯,

正確答案是 true,

上面的代碼中涉及到了泛型,而輸出的結果緣由是型別擦除,先好好說說泛型,

泛型是什么?

泛型的英文是 generics,generic 的意思是通用,而翻譯成中文,泛應該意為廣泛,型是型別,所以泛型就是能廣泛適用的型別,

但泛型還有一種較為準確的說法就是為了引數化型別,或者說可以將型別當作引數傳遞給一個類或者是方法,

那么,如何解釋型別引數化呢?

public class Cache {
   Object value;

   public Object getValue() {
       return value;
   }

   public void setValue(Object value) {
       this.value = value;
   }

}

假設 Cache 能夠存取任何型別的值,于是,我們可以這樣使用它,

Cache cache = new Cache();
cache.setValue(134);
int value = https://www.cnblogs.com/javastack/p/(int) cache.getValue();
cache.setValue("hello");
String value1 = (String) cache.getValue();

使用的方法也很簡單,只要我們做正確的強制轉換就好了,

但是,泛型卻給我們帶來了不一樣的編程體驗,

public class Cache<T> {
   T value;

   public Object getValue() {
       return value;
   }

   public void setValue(T value) {
       this.value = https://www.cnblogs.com/javastack/p/value;
   }

}

這就是泛型,它將 value 這個屬性的型別也引數化了,這就是所謂的引數化型別,再看它的使用方法,

Cache<String> cache1 = new Cache<String>();
cache1.setValue("123");
String value2 = cache1.getValue();

Cache<Integer> cache2 = new Cache<Integer>();
cache2.setValue(456);
int value3 = cache2.getValue();

最顯而易見的好處就是它不再需要對取出來的結果進行強制轉換了,但,還有另外一點不同,

泛型除了可以將型別引數化外,而引數一旦確定好,如果類似不匹配,編譯器就不通過,
上面代碼顯示,無法將一個 String 物件設定到 cache2 中,因為泛型讓它只接受 Integer 的型別,

所以,綜合上面資訊,我們可以得到下面的結論,

  1. 與普通的 Object 代替一切型別這樣簡單粗暴而言,泛型使得資料的類別可以像引數一樣由外部傳遞進來,它提供了一種擴展能力,它更符合面向抽象開發的軟體編程宗旨,

  2. 當具體的型別確定后,泛型又提供了一種型別檢測的機制,只有相匹配的資料才能正常的賦值,否則編譯器就不通過,所以說,它是一種型別安全檢測機制,一定程度上提高了軟體的安全性防止出現低級的失誤,

  3. 泛型提高了程式代碼的可讀性,不必要等到運行的時候才去強制轉換,在定義或者實體化階段,因為 Cache<String> 這個型別顯化的效果,程式員能夠一目了然猜測出代碼要操作的資料型別,

下面的文章,我們正常介紹泛型的相關知識,

泛型的定義和使用

泛型按照使用情況可以分為 3 種,
1. 泛型類,
2. 泛型方法,
3. 泛型介面,

泛型類

我們可以這樣定義一個泛型類,

public class Test<T> {
   T field1;
}

尖括號 <> 中的 T 被稱作是型別引數,用于指代任何型別,事實上,T 只是一種習慣性寫法,如果你愿意,你可以這樣寫,

public class Test<Hello> {
   Hello field1;
}

但出于規范的目的,Java 還是建議我們用單個大寫字母來代表型別引數,常見的如:
1. T 代表一般的任何類,
2. E 代表 Element 的意思,或者 Exception 例外的意思,
3. K 代表 Key 的意思,
4. V 代表 Value 的意思,通常與 K 一起配合使用,
5. S 代表 Subtype 的意思,文章后面部分會講解示意,

如果一個類被 <T> 的形式定義,那么它就被稱為是泛型類,

那么對于泛型類怎么樣使用呢?

Test<String> test1 = new Test<>();
Test<Integer> test2 = new Test<>();

只要在對泛型類創建實體的時候,在尖括號中賦值相應的型別便是,T 就會被替換成對應的型別,如 String 或者是 Integer,你可以相像一下,當一個泛型類被創建時,內部自動擴展成下面的代碼,

public class Test<String> {
   String field1;
}

當然,泛型類不至接受一個型別引數,它還可以這樣接受多個型別引數,

public class MultiType <E,T>{
   E value1;
   T value2;

   public E getValue1(){
       return value1;
   }

   public T getValue2(){
       return value2;
   }
}

泛型方法

public class Test1 {

   public <T> void testMethod(T t){

   }
}

泛型方法與泛型類稍有不同的地方是,型別引數也就是尖括號那一部分是寫在回傳值前面的,<T> 中的 T 被稱為型別引數,而方法中的 T 被稱為引數化型別,它不是運行時真正的引數,

當然,宣告的型別引數,其實也是可以當作回傳值的型別的,

public  <T> T testMethod1(T t){
       return null;
}

泛型類與泛型方法的共存現象

public class Test1<T>{

   public  void testMethod(T t){
       System.out.println(t.getClass().getName());
   }
   public  <T> T testMethod1(T t){
       return t;
   }
}

上面代碼中,Test1<T> 是泛型類,testMethod 是泛型類中的普通方法,而 testMethod1 是一個泛型方法,而泛型類中的型別引數與泛型方法中的型別引數是沒有相應的聯系的,泛型方法始終以自己定義的型別引數為準,

所以,針對上面的代碼,我們可以這樣撰寫測驗代碼,

Test1<String> t = new Test1();
t.testMethod("generic");
Integer i = t.testMethod1(new Integer(1));

泛型類的實際型別引數是 String,而傳遞給泛型方法的型別引數是 Integer,兩者不想干,

但是,為了避免混淆,如果在一個泛型類中存在泛型方法,那么兩者的型別引數最好不要同名,比如,Test1<T> 代碼可以更改為這樣

public class Test1<T>{

   public  void testMethod(T t){
       System.out.println(t.getClass().getName());
   }
   public  <E> E testMethod1(E e){
       return e;
   }
}

泛型介面

泛型介面和泛型類差不多,所以一筆帶過,

public interface Iterable<T> {
}

通配符 ?

除了用 <T> 表示泛型外,還有 <?> 這種形式,? 被稱為通配符,

可能有同學會想,已經有了 <T> 的形式了,為什么還要引進 <?> 這樣的概念呢?

class Base{}

class Sub extends Base{}

Sub sub = new Sub();
Base base = sub;

上面代碼顯示,Base 是 Sub 的父類,它們之間是繼承關系,所以 Sub 的實體可以給一個 Base 參考賦值,那么

List<Sub> lsub = new ArrayList<>();
List<Base> lbase = lsub;

最后一行代碼成立嗎?編譯會通過嗎?

答案是否定的,

編譯器不會讓它通過的,Sub 是 Base 的子類,不代表 List<Sub> 和 List<Base> 有繼承關系,

但是,在現實編碼中,確實有這樣的需求,希望泛型能夠處理某一范圍內的資料型別,比如某個類和它的子類,對此 Java 引入了通配符這個概念,

所以,通配符的出現是為了指定泛型中的型別范圍,

通配符有 3 種形式,

  1. <?> 被稱作無限定的通配符,

  2. <? extends T> 被稱作有上限的通配符,

  3. <? super T> 被稱作有下限的通配符,

無限定通配符

public void testWildCards(Collection<?> collection){
}

上面的代碼中,方法內的引數是被無限定通配符修飾的 Collection 物件,它隱略地表達了一個意圖或者可以說是限定,那就是 testWidlCards() 這個方法內部無需關注 Collection 中的真實型別,因為它是未知的,所以,你只能呼叫 Collection 中與型別無關的方法,

我們可以看到,當 <?> 存在時,Collection 物件喪失了 add() 方法的功能,編譯器不通過,
我們再看代碼,

List<?> wildlist = new ArrayList<String>();
wildlist.add(123);// 編譯不通過

有人說,<?> 提供了只讀的功能,也就是它刪減了增加具體型別元素的能力,只保留與具體型別無關的功能,它不管裝載在這個容器內的元素是什么型別,它只關心元素的數量、容器是否為空?我想這種需求還是很常見的吧,

有同學可能會想,<?> 既然作用這么渺小,那么為什么還要參考它呢?

個人認為,提高了代碼的可讀性,程式員看到這段代碼時,就能夠迅速對此建立極簡潔的印象,能夠快速推斷原始碼作者的意圖,

<? extends T>

<?> 代表著型別未知,但是我們的確需要對于型別的描述再精確一點,我們希望在一個范圍內確定類別,比如型別 A 及 型別 A 的子類都可以,

public void testSub(Collection<? extends Base> para){

}

上面代碼中,para 這個 Collection 接受 Base 及 Base 的子類的型別,

但是,它仍然喪失了寫操作的能力,也就是說

para.add(new Sub());
para.add(new Base());

仍然編譯不通過,

沒有關系,我們不知道具體型別,但是我們至少清楚了型別的范圍,

<? super T>

這個和 <? extends T> 相對應,代表 T 及 T 的超類,

public void testSuper(Collection<? super Sub> para){
}

<? super T> 神奇的地方在于,它擁有一定程度的寫操作的能力,

public void testSuper(Collection<? super Sub> para){
   para.add(new Sub());//編譯通過
   para.add(new Base());//編譯不通過
}

通配符與型別引數的區別

一般而言,通配符能干的事情都可以用型別引數替換,
比如

public void testWildCards(Collection<?> collection){}

可以被

public <T> void test(Collection<T> collection){}

取代,

值得注意的是,如果用泛型方法來取代通配符,那么上面代碼中 collection 是能夠進行寫操作的,只不過要進行強制轉換,

public <T> void test(Collection<T> collection){
   collection.add((T)new Integer(12));
   collection.add((T)"123");
}

需要特別注意的是,型別引數適用于引數之間的類別依賴關系,舉例說明,

public class Test2 <T,E extends T>{
   T value1;
   E value2;
}
public <D,S extends D> void test(D d,S s){

   }

E 型別是 T 型別的子類,顯然這種情況型別引數更適合,
有一種情況是,通配符和型別引數一起使用,

public <T> void test(T t,Collection<? extends T> collection){

}

如果一個方法的回傳型別依賴于引數的型別,那么通配符也無能為力,

public T test1(T t){
   return value1;
}

型別擦除

泛型是 Java 1.5 版本才引進的概念,在這之前是沒有泛型的概念的,但顯然,泛型代碼能夠很好地和之前版本的代碼很好地兼容,

這是因為,泛型資訊只存在于代碼編譯階段,在進入 JVM 之前,與泛型相關的資訊會被擦除掉,專業術語叫做型別擦除,

通俗地講,泛型類和普通類在 java 虛擬機內是沒有什么特別的地方,回顧文章開始時的那段代碼

List<String> l1 = new ArrayList<String>();
List<Integer> l2 = new ArrayList<Integer>();

System.out.println(l1.getClass() == l2.getClass());

列印的結果為 true 是因為 List<String> 和 List<Integer> 在 jvm 中的 Class 都是 List.class,

泛型資訊被擦除了,

可能同學會問,那么型別 String 和 Integer 怎么辦?

答案是泛型轉譯,

public class Erasure <T>{
   T object;

   public Erasure(T object) {
       this.object = object;
   }

}

Erasure 是一個泛型類,我們查看它在運行時的狀態資訊可以通過反射,

Erasure<String> erasure = new Erasure<String>("hello");
Class eclz = erasure.getClass();
System.out.println("erasure class is:"+eclz.getName());

列印的結果是

erasure class is:com.frank.test.Erasure

Class 的型別仍然是 Erasure 并不是 Erasure<T> 這種形式,那我們再看看泛型類中 T 的型別在 jvm 中是什么具體型別,

Field[] fs = eclz.getDeclaredFields();
for ( Field f:fs) {
   System.out.println("Field name "+f.getName()+" type:"+f.getType().getName());
}

列印結果是

Field name object type:java.lang.Object

那我們可不可以說,泛型類被型別擦除后,相應的型別就被替換成 Object 型別呢?

這種說法,不完全正確,

我們更改一下代碼,

public class Erasure <T extends String>{
//  public class Erasure <T>{
   T object;

   public Erasure(T object) {
       this.object = object;
   }

}

現在再看測驗結果:

Field name object type:java.lang.String

我們現在可以下結論了,在泛型類被型別擦除的時候,之前泛型類中的型別引數部分如果沒有指定上限,如 <T> 則會被轉譯成普通的 Object 型別,如果指定了上限如 <T extends String> 則型別引數就被替換成型別上限,

所以,在反射中,

public class Erasure <T>{
   T object;

   public Erasure(T object) {
       this.object = object;
   }

   public void add(T object){

   }

}

add() 這個方法對應的 Method 的簽名應該是 Object.class,

Erasure<String> erasure = new Erasure<String>("hello");
Class eclz = erasure.getClass();
System.out.println("erasure class is:"+eclz.getName());

Method[] methods = eclz.getDeclaredMethods();
for ( Method m:methods ){
   System.out.println(" method:"+m.toString());
}

列印結果是

method:public void com.frank.test.Erasure.add(java.lang.Object)

也就是說,如果你要在反射中找到 add 對應的 Method,你應該呼叫 getDeclaredMethod("add",Object.class) 否則程式會報錯,提示沒有這么一個方法,原因就是型別擦除的時候,T 被替換成 Object 型別了,

型別擦除帶來的局限性

型別擦除,是泛型能夠與之前的 java 版本代碼兼容共存的原因,但也因為型別擦除,它會抹掉很多繼承相關的特性,這是它帶來的局限性,

理解型別擦除有利于我們繞過開發當中可能遇到的雷區,同樣理解型別擦除也能讓我們繞過泛型本身的一些限制,比如

正常情況下,因為泛型的限制,編譯器不讓最后一行代碼編譯通過,因為類似不匹配,但是,基于對型別擦除的了解,利用反射,我們可以繞過這個限制,

public interface List<E> extends Collection<E>{

    boolean add(E e);
}

上面是 List 和其中的 add() 方法的原始碼定義,

因為 E 代表任意的型別,所以型別擦除時,add 方法其實等同于

boolean add(Object obj);

那么,利用反射,我們繞過編譯器去呼叫 add 方法,

public class ToolTest {

   public static void main(String[] args) {
       List<Integer> ls = new ArrayList<>();
       ls.add(23);
//      ls.add("text");
       try {
           Method method = ls.getClass().getDeclaredMethod("add",Object.class);

           method.invoke(ls,"test");
           method.invoke(ls,42.9f);
       } catch (NoSuchMethodException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (SecurityException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (IllegalAccessException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (IllegalArgumentException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       } catch (InvocationTargetException e) {
           // TODO Auto-generated catch block
           e.printStackTrace();
       }

       for ( Object o: ls){
           System.out.println(o);
       }

   }

}

列印結果是:

23
test
42.9

可以看到,利用型別擦除的原理,用反射的手段就繞過了正常開發中編譯器不允許的操作限制,

泛型中值得注意的地方

泛型類或者泛型方法中,不接受 8 種基本資料型別,

所以,你沒有辦法進行這樣的編碼,

List<int> li = new ArrayList<>();
List<boolean> li = new ArrayList<>();

需要使用它們對應的包裝類,

List<Integer> li = new ArrayList<>();
List<Boolean> li1 = new ArrayList<>();

對泛型方法的困惑

public <T> T test(T t){
   return null;
}

有的同學可能對于連續的兩個 T 感到困惑,其實 <T> 是為了說明型別引數,是宣告,而后面的不帶尖括號的 T 是方法的回傳值型別,
你可以相像一下,如果 test() 這樣被呼叫

test("123");

那么實際上相當于

public String test(String t);

Java 不能創建具體型別的泛型陣列

這句話可能難以理解,代碼說明,

List<Integer>[] li2 = new ArrayList<Integer>[];
List<Boolean> li3 = new ArrayList<Boolean>[];

這兩行代碼是無法在編譯器中編譯通過的,原因還是型別擦除帶來的影響,

List<Integer> 和 List<Boolean> 在 jvm 中等同于List<Object> ,所有的型別資訊都被擦除,程式也無法分辨一個陣列中的元素型別具體是 List<Integer>型別還是 List<Boolean> 型別,

但是,

List<?>[] li3 = new ArrayList<?>[10];
li3[1] = new ArrayList<String>();
List<?> v = li3[1];

借助于無限定通配符卻可以,前面講過  代表未知型別,所以它涉及的操作都基本上與型別無關,因此 jvm 不需要針對它對型別作判斷,因此它能編譯通過,但是,只提供了陣列中的元素因為通配符原因,它只能讀,不能寫,比如,上面的 v 這個區域變數,它只能進行 get() 操作,不能進行 add() 操作,這個在前面通配符的內容小節中已經講過,

泛型,并不神奇

我們可以看到,泛型其實并沒有什么神奇的地方,泛型代碼能做的非泛型代碼也能做,

而型別擦除,是泛型能夠與之前的 java 版本代碼兼容共存的原因,

可量也正因為型別擦除導致了一些隱患與局限,

但,我還是要建議大家使用泛型,如官方檔案所說的,如果可以使用泛型的地方,盡量使用泛型,

畢竟它抽離了資料型別與代碼邏輯,本意是提高程式代碼的簡潔性和可讀性,并提供可能的編譯時型別轉換安全檢測功能,

型別擦除不是泛型的全部,但是它卻能很好地檢測我們對于泛型這個概念的理解程度,

我在文章開頭將泛型比作是一個守門人,原因就是他本意是好的,守護我們的代碼安全,然后在門牌上寫著出入的各項規定,及“xxx 禁止出入”的提醒,但是同我們日常所遇到的那些門衛一般,他們古怪偏執,死板守舊,我們可以利用反射基于型別擦除的認識,來繞過泛型中某些限制,現實生活中,也總會有調皮搗蛋者能夠基于對門衛們生活作息的規律,選擇性地繞開他們的監視,另辟蹊徑溜進或者溜出大門,然后揚長而去,剩下守衛者一個孤獨的身影,

所以,我說泛型,并不神秘,也不神奇,

推薦去我的博客閱讀更多:

1.Java JVM、集合、多執行緒、新特性系列教程

2.Spring MVC、Spring Boot、Spring Cloud 系列教程

3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程

4.Java、后端、架構、阿里巴巴等大廠最新面試題

覺得不錯,別忘了點贊+轉發哦!

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

標籤:Java

上一篇:專案經理說這種代碼必須重構,我同意了,這代碼是寫的是有多爛!

下一篇:《自拍教程50》Python adb隨機按鍵模擬2小時

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