java基礎(chǔ)學習筆記之反射
反射
反射:將類的屬性和方法映射成相應(yīng)的類。
反射基本使用
獲取Class類的三種方法:
- 類名.class
- 對象名.getClass()
- Class.forName("要加載的類名")
根據(jù)API寫就行了,大致流程就是:
- 用上述三種方式之一獲取特定類的Class類,即該類對應(yīng)的字節(jié)碼
- 調(diào)用Class對象的getConstructor(Class<?>... parameterTypes)獲取構(gòu)造方法對象
- 調(diào)用是構(gòu)造方法類Constructor的newInstance(Object... initargs)方法新建對象
- 調(diào)用Class對象的getMethod(String name, Class<?>... parameterTypes)獲取方法對象
- 調(diào)用方法對象類Method的invoke(Object obj, Object... args)方法,調(diào)用對象上相應(yīng)方法
用方法的參數(shù)類型唯一標識一個方法,依據(jù):方法的重載
數(shù)組的反射
下面這個例子主要說明幾點:
- 對于元素同類型的數(shù)組,同維數(shù)組,class一樣
- 不同維,class不同
- 不同維的,父類都是Object,一樣
- 基本類型以為數(shù)組不能直接轉(zhuǎn)換為Object[]
- java.util.Arrays的asList方法API看看
public class ReflectTest { public static void main(String[] args) { int [] a1 = new int[]{1,2,3}; int [] a2 = new int[5]; int [][] a3 = new int[2][3]; System.out.println(a1.getClass() == a2.getClass());//true System.out.println(a1.getClass());//class [I System.out.println(a3.getClass());//class [[I System.out.println(a1.getClass().getSuperclass() == a3.getClass().getSuperclass());//true System.out.println(a2.getClass().getSuperclass());//class java.lang.Object //下句編譯不通過:Error:(15, 42) java: 不可比較的類型: java.lang.Class<capture#1, 共 ? extends int[]>和java.lang.Class<capture#2, 共 ? extends int[][]> //System.out.println(a1.getClass() == a3.getClass()); Object []b3 = a3;//通過 //下句編譯不通過 Error:(17, 24) java: 不兼容的類型: int[]無法轉(zhuǎn)換為java.lang.Object[] //Object [] b1 = a1; String s1 = "abc"; System.out.println(Arrays.asList(a1));//[[I@1540e19d] System.out.println(Arrays.asList(s1));//[abc] } }
輸出:
true class [I class [[I true class java.lang.Object [[I@1540e19d] [abc]
亂入: hashcode與內(nèi)存泄露問題 參考java api:
- hashcode一旦生成,不要變
- 對象equals方法返回true,則hascode要一致
- 反之,equals方法返回false,hascode不一定互異
如果參與hascode計算的成員變量中途發(fā)生變化,則后面remove時失敗,造成內(nèi)存泄露
配置文件加載
類加載器加載只讀配置文件
類名.class.getClassLoader().getResourceAsStream(str);
類名.class.getResourceAsStream(str),實質(zhì)還是調(diào)用類加載器。 源碼截取(java.lang包下的Class.java):
public InputStream getResourceAsStream(String name) { name = resolveName(name); ClassLoader cl = getClassLoader0(); if (cl==null) { // A system class. return ClassLoader.getSystemResourceAsStream(name); } return cl.getResourceAsStream(name); }
關(guān)于路徑str,寫法有點講究。
- 不加斜杠,相對路徑: str = "config.properties";
- 加斜杠,從classpath的根路徑找: str = "/org/iot/ui/config.properties";
以前編譯java代碼時,有些conf/文件夾還要添加進依賴或者標記成source文件夾,里面明明都是xml文件,沒Java源碼。從這里,我現(xiàn)在知道了,是使用反射加載配置文件的緣故
內(nèi)省(Instropector) & JavaBean
JavaBean讀取屬性x的值的流程:變大寫、補前綴、獲取方法。
"x"-->"X"-->"getX"-->"MethodGetX"
自己用內(nèi)省操作
我目前沒用上,所以不貼代碼了,只附上核心類
簡單實現(xiàn): 使用java.beans.PropertyDescriptor類
麻煩實現(xiàn): 使用java.beans.Introspector類,遍歷getBeanInfo方法的返回值
JavaBean必須有一個不帶參數(shù)的構(gòu)造函數(shù)
使用BeanUtils工具包
- 字符串和整數(shù)轉(zhuǎn)換(對比(PropertyUtils)
- 屬性級聯(lián)操作
- 操作map
相關(guān)文章
Mybatis代碼生成器Mybatis Generator(MBG)實戰(zhàn)詳解
本文我們主要實戰(zhàn)Mybatis官方的代碼生成器:Mybatis Generator(MBG),掌握它以后,可以簡化大部分手寫代碼,我們只需要寫復雜邏輯代碼,需要的朋友可以參考下2023-05-05Netty之使用DelimiterBasedFrameDecoder進行消息分隔詳解
這篇文章主要介紹了Netty之使用DelimiterBasedFrameDecoder進行消息分隔詳解,在使用Netty進行TCP消息傳輸時,為了上層協(xié)議能夠?qū)ο⒄_區(qū)分,避免粘包和拆包導致的問題,一般可以通過消息定長、將回車換行符作為消息結(jié)束符,需要的朋友可以參考下2023-12-12Java concurrency之公平鎖(一)_動力節(jié)點Java學院整理
這篇文章主要為大家詳細介紹了Java concurrency之公平鎖的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06通過實例了解Java Integer類和int的區(qū)別
這篇文章主要介紹了通過實例了解Java Integer類和int的區(qū)別,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-03-03Spring Cloud Gateway 獲取請求體(Request Body)的多種方法
這篇文章主要介紹了Spring Cloud Gateway 獲取請求體(Request Body)的多種方法,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-01-01