Java基礎 - 反射(2)

文章目錄

  • 示例
    • 5. 通過反射獲得類的private、 protected、 默認訪問修飾符的屬性值。
    • 6. 通過反射獲得類的private方法。
    • 7. 通過反射實現一個工具BeanUtils, 可以將一個對象屬性相同的值賦值給另一個對象

接上篇:

示例

5. 通過反射獲得類的private、 protected、 默認訪問修飾符的屬性值。

import java.lang.reflect.Field;class Student {private String privateField = "私有屬性值";protected String protectedField = "受保護屬性值";String defaultField = "默認訪問屬性值"; // 包級私有public String publicField = "公有屬性值";
}
public class ReflectionExample5 {public static void main(String[] args) {Student student = new Student();try {// 獲取類的 Class 對象Class<?> clazz = Student.class;// ================== 訪問私有屬性 ==================Field privateField = clazz.getDeclaredField("privateField");privateField.setAccessible(true); // 解除私有訪問限制String privateValue = (String) privateField.get(student);System.out.println("privateField: " + privateValue);// ================== 訪問受保護屬性 ==================Field protectedField = clazz.getDeclaredField("protectedField");protectedField.setAccessible(true); // 無需繼承關系即可訪問String protectedValue = (String) protectedField.get(student);System.out.println("protectedField: " + protectedValue);// ================== 訪問默認(包級私有)屬性 ==================Field defaultField = clazz.getDeclaredField("defaultField");defaultField.setAccessible(true);String defaultValue = (String) defaultField.get(student);System.out.println("defaultField: " + defaultValue);// ================== 訪問公有屬性 ==================// 方式 1:getDeclaredField + setAccessible(強制訪問)Field publicField1 = clazz.getDeclaredField("publicField");publicField1.setAccessible(true); // 即使公有也強制解除限制(非必須)String publicValue1 = (String) publicField1.get(student);System.out.println("publicField(強制訪問): " + publicValue1);// 方式 2:直接通過 getField 獲取(不推薦,僅用于對比)Field publicField2 = clazz.getField("publicField");String publicValue2 = (String) publicField2.get(student);System.out.println("publicField(正常訪問): " + publicValue2);} catch (NoSuchFieldException e) {System.err.println("字段不存在: " + e.getMessage());} catch (IllegalAccessException e) {System.err.println("訪問權限失敗: " + e.getMessage());}}
}

關鍵操作說明

  1. getDeclaredField() 方法
    - 作用:獲取類中聲明的所有字段(包括 private/protected/默認/public)
    - 需要手動調用 setAccessible(true) 來突破非 public 屬性的訪問限制。
  2. setAccessible(true)
    - 方法:Field.setAccessible(true)
    - 意義:解除 Java 的訪問控制檢查,允許操作非公有字段。
    - 安全性警告:此操作會繞過 Java 的封裝機制,僅建議在框架等需要深度操作時使用。
  3. 字段值與對象實例的關聯
    - 對于 實例字段:必須傳入具體對象實例(如 student),通過 field.get(object) 獲取值。
    - 對于 靜態字段:可直接傳入 null,如 field.get(null)。

6. 通過反射獲得類的private方法。

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;class MyClass3 {private String privateMethod(String param) {return "私有方法被調用,參數: " + param;}
}
public class ReflectionExample6 {public static void main(String[] args) {try {// 創建類的實例MyClass3 myObject = new MyClass3();// 獲取Class對象Class<?> clazz = MyClass3.class;// 獲取私有方法,需指定方法名和參數類型Method method = clazz.getDeclaredMethod("privateMethod", String.class);// 解除訪問限制(關鍵步驟)method.setAccessible(true);// 調用方法,傳入實例及參數String result = (String) method.invoke(myObject, "Hello");// 輸出結果System.out.println("調用結果: " + result);} catch (NoSuchMethodException e) {System.err.println("方法未找到: " + e.getMessage());} catch (IllegalAccessException e) {System.err.println("非法訪問: " + e.getMessage());} catch (InvocationTargetException e) {System.err.println("方法內部錯誤: " + e.getCause());}}
}

7. 通過反射實現一個工具BeanUtils, 可以將一個對象屬性相同的值賦值給另一個對象

import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;class BeanUtils {/*** 將源對象的屬性值復制到目標對象(淺拷貝)* @param source 源對象* @param target 目標對象*/public static void copyProperties(Object source, Object target) {if (source == null || target == null) {throw new IllegalArgumentException("源對象和目標對象不能為 null");}// 獲取源對象和目標對象的所有字段(包含父類字段)List<Field> sourceFields = getAllFields(source.getClass());List<Field> targetFields = getAllFields(target.getClass());for (Field sourceField : sourceFields) {// 查找目標對象中與源對象字段同名的字段Field targetField = findField(targetFields, sourceField.getName());if (targetField == null) continue;// 檢查類型是否兼容(支持基本類型和包裝類型)if (!isTypeCompatible(sourceField.getType(), targetField.getType())) continue;try {// 設置字段可訪問性sourceField.setAccessible(true);targetField.setAccessible(true);// 復制值Object value = sourceField.get(source);targetField.set(target, value);} catch (IllegalAccessException e) {// 處理無法訪問的異常System.err.println("字段復制失敗: " + sourceField.getName() + " -> " + targetField.getName());}}}/*** 獲取類及其父類的所有字段(非靜態)*/private static List<Field> getAllFields(Class<?> clazz) {List<Field> fields = new java.util.ArrayList<>();while (clazz != null && clazz != Object.class) {fields.addAll(Arrays.stream(clazz.getDeclaredFields()).filter(f -> !isStatic(f)).collect(Collectors.toList()));clazz = clazz.getSuperclass();}return fields;}/*** 根據字段名從字段列表中查找字段*/private static Field findField(List<Field> fields, String fieldName) {return fields.stream().filter(f -> f.getName().equals(fieldName)).findFirst().orElse(null);}/*** 判斷字段是否為靜態*/private static boolean isStatic(Field field) {return java.lang.reflect.Modifier.isStatic(field.getModifiers());}/*** 判斷源類型與目標類型是否兼容*/private static boolean isTypeCompatible(Class<?> sourceType, Class<?> targetType) {// 處理基本類型與包裝類型的兼容(例如 int -> Integer)if (sourceType.isPrimitive()) {return targetType.isPrimitive() ? sourceType.equals(targetType) : getWrapperType(sourceType).equals(targetType);} else if (targetType.isPrimitive()) {return getWrapperType(targetType).equals(sourceType);} else {return targetType.isAssignableFrom(sourceType);}}/*** 獲取基本類型對應的包裝類型*/private static Class<?> getWrapperType(Class<?> primitiveType) {if (primitiveType == int.class) return Integer.class;if (primitiveType == long.class) return Long.class;if (primitiveType == boolean.class) return Boolean.class;if (primitiveType == byte.class) return Byte.class;if (primitiveType == char.class) return Character.class;if (primitiveType == short.class) return Short.class;if (primitiveType == double.class) return Double.class;if (primitiveType == float.class) return Float.class;return primitiveType;}
}// 父類
class Person {protected String name;private int age;public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}// 子類
class Employee extends Person {private double salary;private Integer departmentId;public double getSalary() {return salary;}public void setSalary(double salary) {this.salary = salary;}public Integer getDepartmentId() {return departmentId;}public void setDepartmentId(Integer departmentId) {this.departmentId = departmentId;}
}// 目標類
class EmployeeDTO {private String name;private int age;        // 基本類型private Double salary;  // 包裝類型private Integer departmentId;public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public Double getSalary() {return salary;}public void setSalary(Double salary) {this.salary = salary;}public Integer getDepartmentId() {return departmentId;}public void setDepartmentId(Integer departmentId) {this.departmentId = departmentId;}
}public class ReflectionExample7 {public static void main(String[] args) {// 準備數據Employee emp = new Employee();emp.name = "張三";emp.setAge(30);emp.setSalary(15000.5);emp.setDepartmentId(101);// 目標對象EmployeeDTO dto = new EmployeeDTO();// 復制屬性BeanUtils.copyProperties(emp, dto);// 驗證結果System.out.println("DTO.name: " + dto.getName());System.out.println("DTO.age: " + dto.getAge());System.out.println("DTO.salary: " + dto.getSalary());System.out.println("DTO.departmentId: " + dto.getDepartmentId());}
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/900948.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/900948.shtml
英文地址,請注明出處:http://en.pswp.cn/news/900948.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

FCOS目標檢測

一、模型框架 FCOS采用的網絡架構和RetinaNet一樣&#xff0c;都是采用FPN架構&#xff0c;如圖2所示&#xff0c;每個特征圖后是檢測器&#xff0c;檢測器包含3個分支&#xff1a;classification&#xff0c;regression和center-ness。 對于特征圖Fi∈RHWC&#xff0c;其相對…

Java基礎 - 泛型(常見用法)

文章目錄 泛型類泛型方法泛型類派生子類示例 1&#xff1a;子類固定父類泛型類型&#xff08;StringBox 繼承自 Box<String>&#xff09;示例 2&#xff1a;子類保留父類泛型類型&#xff08;AdvancedBox<T> 繼承自 Box<T>)示例 3&#xff1a;添加子類自己的…

YOLO學習筆記 | YOLOv8環境搭建全流程指南(2025.4)

===================================================== github:https://github.com/MichaelBeechan CSDN:https://blog.csdn.net/u011344545 ===================================================== YOLOv8環境搭建 一、環境準備與工具配置1. Conda虛擬環境搭建2. CUDA與…

【 Beautiful Soup (bs4) 詳解】

引言 Beautiful Soup 是 Python 最流行的 HTML/XML 解析庫&#xff0c;能夠從復雜的網頁文檔中高效提取數據。以下是其核心知識點及示例代碼。 一、庫簡介 1. 核心模塊 BeautifulSoup&#xff1a;主類&#xff0c;用于構建文檔樹結構Tag&#xff1a;表示 HTML/XML 標簽的對象…

傅利葉發布首款開源人形機器人N1:開發者可實現完整復刻

2025年4月11日&#xff0c;上海——通用機器人公司傅利葉正式發布首款開源人形機器人 Fourier N1&#xff0c;并同步開放涵蓋物料清單、設計圖紙、裝配指南、基礎操作軟件在內的完整本體資源包。作為傅利葉 “Nexus 開源生態矩陣” 的首個落地項目&#xff08;“N1” 即 “Nexu…

視覺目標檢測大模型GAIA

中國科學院自動化研究所智能感知與計算研究中心攜手華為等領軍企業&#xff0c;共同推出面向產業應用的視覺目標檢測全流程解決方案——GAIA智能檢測平臺。該研究成果已獲CVPR 2021會議收錄&#xff08;論文鏈接&#xff1a; 論文地址&#xff1a;https://arxiv.org/pdf/2106.…

前端時間同步利器:React + useEffect 實現高性能動態時鐘

前言 在你奮筆疾敲代碼的瞬間&#xff0c;是不是突然一低頭&#xff0c;發現時間像偷偷跑路的變量&#xff0c;一眨眼就從上午飄到下午&#xff1f;飯沒吃、會沒開、工位也快被前端貓霸占了。仿佛你寫的不是代碼&#xff0c;而是“時間穿梭機”。別慌&#xff0c;咱們今天就來…

前端動畫性能優化

前端動畫性能優化全攻略&#xff1a;告別卡頓與高CPU占用 一、動畫性能問題現狀分析 1.1 性能問題現象 動畫幀率低于60FPS時出現明顯卡頓滾動/縮放操作時響應延遲CPU占用率長期超過70%移動端設備發熱嚴重 1.2 核心問題根源 瀏覽器渲染流程中的性能瓶頸主要出現在&#xff1…

springboot中如何處理跨域

什么是跨域 跨域&#xff08;Cross-Origin&#xff09;是瀏覽器出于安全考慮&#xff0c;對不同源的資源訪問施加的限制機制。其核心原因是同源策略&#xff08;Same-Origin Policy&#xff09;&#xff0c;即瀏覽器僅允許協議&#xff08;Protocol&#xff09;、域名&#xf…

js實現生肖宜忌展示

實現效果圖如下 實現邏輯&#xff1a; 1.錄入屬相列表&#xff08;列表順序不可調整&#xff09;&#xff1b; 2.錄入各屬相相宜、相忌屬相&#xff1b; 3.輸入年份后&#xff0c;根據屬相列表獲取到正確的屬相&#xff1b; 4.根據獲取的屬相去展示宜、忌屬相&#xff1b; 5.打…

3DMAX筆記-UV知識點和烘焙步驟

1. 在展UV時&#xff0c;如何點擊模型&#xff0c;就能選中所有這個模型的uv 2. 分多張UV時&#xff0c;不同的UV的可以設置為不同的顏色&#xff0c;然后可以通過顏色進行篩選。 3. 烘焙步驟 擺放完UV后&#xff0c;要另存為一份文件&#xff0c;留作備份 將模型部件全部分成…

AI 重構 Java 遺留系統:從靜態方法到 Spring Bean 注入的自動化升級

在當今快速發展的軟件行業中&#xff0c;許多企業都面臨著 Java 遺留系統的維護和升級難題。這些老舊系統往往采用了大量靜態方法&#xff0c;隨著業務的不斷發展&#xff0c;其局限性日益凸顯。而飛算 JavaAI 作為一款強大的 AI 工具&#xff0c;為 Java 遺留系統的重構提供了…

【從一個 TypeScript 報錯理解 ES6 模塊的三種導入方式】

從一個 TypeScript 報錯理解 ES6 模塊的三種導入方式 在日常開發中&#xff0c;我們經常遇到模塊導入導出的場景。最近在處理一個項目時&#xff0c;遇到了一個有趣的問題&#xff1a;對于只有默認導出的模塊&#xff0c;我們該使用哪種導入方式&#xff1f;這個問題引發了對 …

安徽京準:NTP網絡時鐘服務器功能及同步模式的介紹

安徽京準&#xff1a;NTP網絡時鐘服務器功能及同步模式的介紹 安徽京準&#xff1a;NTP網絡時鐘服務器功能及同步模式的介紹 1、NTP網絡時鐘服務器概念&#xff1a; NTP時鐘服務器&#xff0c;表面意思是時間計量工具的服務設備&#xff0c;其在現代工業中是用于對客戶端設備…

JMeter從入門到荒廢-常見問題匯總

啟動某個ThreadGroup的時候&#xff0c;啟動不了 現象 點擊start按鈕的時候&#xff0c;結果樹和匯總報告都沒有任何數據。 同時&#xff0c;點擊右上角的error log 發現有錯誤信息&#xff1a; 錯誤信息如下&#xff1a; 2025-04-09 10:03:48,009 ERROR o.a.j.g.a.ActionR…

Elasticsearch 學習規劃

Elasticsearch 學習規劃 明確學習目標與動機 場景化需求分析 - **S**&#xff1a;掌握Elasticsearch架構體系&#xff0c;熟練使用Elasticsearch 進行數據分析,Elasticsearch結合java 項目落地案例 - **M**&#xff1a;搜索和Elasticsearch相關GitHub項目 - **A**&#xff1a;每…

核心案例 | 湖南汽車工程職業大學無人機操控與編隊技術實驗室

核心案例 | 湖南汽車工程職業大學無人機操控與編隊技術實驗室 為滿足當今無人機行業應用需求&#xff0c;推動無人機技術的教育與實踐深度融合&#xff0c;北京卓翼智能科技有限公司旗下品牌飛思實驗室與湖南汽車工程職業大學強強聯手&#xff0c;共同建設無人機操控與編隊技術…

【Android】Android 獲取當前前臺應用包名與自動化控制全流程實踐筆記(適配 Android 10+)

一、前言 在 Android 系統中&#xff0c;獲取當前運行的前臺應用、返回桌面、跳轉權限設置、關閉其他應用等行為&#xff0c;往往受到系統的嚴格限制。隨著 Android 版本的提升&#xff08;特別是 Android 10 之后&#xff0c;即 API 29&#xff09;&#xff0c;很多傳統方法已…

Sentinel核心源碼分析(上)

文章目錄 前言一、客戶端與Spring Boot整合二、SphU.entry2.1、構建責任鏈2.2、調用責任鏈2.2.1、NodeSelectorSlot2.2.2、ClusterBuilderSlot2.2.3、LogSlot2.2.4、StatisticSlot2.2.5、AuthoritySlot2.2.6、SystemSlot2.2.7、FlowSlot2.2.7.1、selectNodeByRequesterAndStrat…

淺談「分詞」:原理 + 方案對比 + 最佳實踐

在文本搜索、自然語言處理、智能推薦等場景中&#xff0c;「分詞」 是一個基礎但至關重要的技術點。無論是用數據庫做模糊查詢&#xff0c;還是構建搜索引擎&#xff0c;分詞都是提高效率和準確度的核心手段。 &#x1f50d; 一、什么是分詞&#xff1f; 分詞&#xff08;Tok…