java基礎(chǔ)之反射和泛型以及注解
java基礎(chǔ)之反射和泛型以及注解
泛型擦除
泛型擦除: 泛型只在編譯時期有效,編譯后的字節(jié)碼文件中不存在泛型信息。
聲明泛型集合,集合兩端類型必須一致。類型也可以用包裝類型,泛型的類型必須是引用類型,不能為基本類型。
實(shí)現(xiàn)公用的類和方法,對公用的業(yè)務(wù)進(jìn)行抽取。
泛型方法/泛型類/泛型接口
public class GenericTest
{
/**
* 泛型聲明,定義泛型方法
* @param <T>
* @param <K>
* @param t
* @param k
*/
public <T, K> K save(T t, K k)
{
return null;
}
@Test
public void testMethod() throws Exception
{
//使用泛型方法: 在使用泛型方法的時候,確定泛型的類型
save("hello", 1);
}
}
泛型類:
public class GenericTest<T>
@Test
public void testMethod() throws Exception
{
//使用泛型方法: 在使用泛型方法的時候,確定泛型的類型
//save("hello", 1);
//泛型類如何使用:在創(chuàng)建泛型類的時候確定
GenericTest<String> demo = new GenericTest<String>();
demo.save("hello", 1);
}
泛型中的extends 和super的意義:
Extends:定義List<? Extends String>;傳入的參數(shù)?必須是String類型的子類,否則會報錯;
Super:定義List<? Super String>;傳入的參數(shù)必須是String類型的父類,否則會報錯;
Type : 接口,任何類型默認(rèn)的接口!
反射
反射可以在運(yùn)行時期動態(tài)創(chuàng)建對象,獲取對象的屬性,方法
/**
* @ClassName: App
* @Description: 反射技術(shù)
* @author lqw
* @date 2016-5-13 下午01:33:55
*
*/
public class App
{
@Test
public void testInfo() throws Exception
{
//類全名
String sql = "com.hbmy.reflect.demo2.Admin";
//得到類的字節(jié)碼
Class<?> clazz = Class.forName(sql);
/**
* 創(chuàng)建對象1: 通過默認(rèn)構(gòu)造函數(shù)創(chuàng)建(簡寫)
*/
Admin admin = (Admin) clazz.newInstance();
/**
* 創(chuàng)建對象2: 通過無參構(gòu)造器創(chuàng)建對象
*/
Constructor<?> constructors = clazz.getDeclaredConstructor();
constructors.newInstance();
/**
* 創(chuàng)建對象3:通過有參構(gòu)造器創(chuàng)建對象
*/
Constructor<?> constructor = clazz.getDeclaredConstructor(String.class);
Admin admin2 = (Admin) constructor.newInstance("zhangsan");
// System.out.println(admin);
}
/**
* 獲取屬性名稱、值
* getDeclaredFields: 獲取所有的包含私有的屬性名稱
* getFields:只能訪問public的屬性
*/
@Test
public void testNameAndValue() throws Exception
{
//類全名
String sql = "com.hbmy.reflect.demo2.Admin";
//得到類的字節(jié)碼
Class<?> clazz = Class.forName(sql);
Admin admin = (Admin) clazz.newInstance();
// Method[] methods = clazz.getMethods();
// for (Method method : methods)
// {
// //設(shè)置強(qiáng)制訪問
// method.setAccessible(true);
// //名稱
// String name = method.getName();
// System.out.println(name);
//
// }
// Field[] fields = clazz.getFields();//打印出來的結(jié)果只有money
Field[] fields = clazz.getDeclaredFields();
for (Field field : fields)
{
//設(shè)置強(qiáng)制訪問
field.setAccessible(true);
//名稱
String name = field.getName();
Object value = field.get(admin);
System.out.println(name + value);
}
}
/**
* 反射獲取方法
*/
@Test
public void testGetMethods() throws Exception
{
//類全名
String sql = "com.hbmy.reflect.demo2.Admin";
//得到類的字節(jié)碼
Class<?> clazz = Class.forName(sql);
Admin admin = (Admin) clazz.newInstance();
/*
* 獲取方法對象
*/
Method declaredMethod = clazz.getDeclaredMethod("getId");
/**
* 調(diào)用方法
*/
Object return_value = declaredMethod.invoke(admin);
System.out.println(return_value);
Method[] methods = clazz.getDeclaredMethods();
for (Method method : methods)
{
method.setAccessible(true);
String name = method.getName();
System.out.println(name);
}
}
注解
注解的作用
1、 告訴編譯器如何去運(yùn)行
2、 簡化(取代)配置文件
public class App
{
@Override
public String toString()
{
return super.toString();
}
@SuppressWarnings({"unused","unchecked"})
public void save()
{
List list = null;
}
@Deprecated
public void save1()
{
}
}
自定義注解: 通過自定義注解可以給類,字段,方法加上描述信息。
public @interface Author
{
/**
* 注解屬性
* 1.修飾符為默認(rèn)或者public
* 2.不能有主體
* 3. 如果注解名稱為value,使用的時候可以省略名稱,直接給值
*/
String name() default "lqw";
//帶默認(rèn)值得注解
int age() default 23;
String remark();
}
元注解
元注解就是注解的注解
指定注解的可用范圍
@Target({
TYPE,
FIELD,
METHOD,
PARAMETER,
CONSTRUCTOR,
LOCAL_VARIABLE})
注解的生命周期
/**
* 元注解2: 指定注解的生命周期
* RetentionPolicy.SOURCE 只在源碼級別有效
* RetentionPolicy.CLASS 只在類的字節(jié)碼級別有效 默認(rèn)值
* RetentionPolicy.RUNTIME 只在運(yùn)行時期有效
*/
@Retention(RetentionPolicy.SOURCE)
最后總結(jié)一句:注解和反射其實(shí)不難,只要不畏懼,注解其實(shí)看看源碼也就那么回事。至于反射嘛。可以這么說,無反射,則無框架,幾乎所有的框架都是通過反射實(shí)現(xiàn)的。說白了,反射也就是通過加載類的字節(jié)碼去獲取類里面的方法和屬性,其實(shí)框架也是這么實(shí)現(xiàn)的。
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關(guān)文章
Spring Boot 3.x 全新的熱部署配置方式詳解(IntelliJ ID
這篇文章主要介紹了Spring Boot 3.x 全新的熱部署配置方式(IntelliJ IDEA 2023.1),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-07-07
解決springboot啟動Logback報錯ERROR in ch.qos.logback.cla
這篇文章主要介紹了解決springboot啟動Logback報錯ERROR in ch.qos.logback.classic.joran.action.ContextNameAction - Failed to rena問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-04-04
SpringBoot中Mybatis + Druid 數(shù)據(jù)訪問的詳細(xì)過程
Spring Boot 底層都是采用 SpringData 的方式進(jìn)行統(tǒng)一處理各種數(shù)據(jù)庫,SpringData也是Spring中與SpringBoot、SpringCloud 等齊名的知名項(xiàng)目,下面看下SpringBoot Mybatis Druid數(shù)據(jù)訪問的詳細(xì)過程,感興趣的朋友一起看看吧2021-11-11
java用重定向方法從文件中讀入或?qū)懭霐?shù)據(jù)
這篇文章主要為大家詳細(xì)介紹了用重定向方法從文件中讀入或?qū)懭霐?shù)據(jù),具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-03-03
Spring Boot外部化配置實(shí)戰(zhàn)解析
這篇文章主要介紹了Spring Boot外部化配置實(shí)戰(zhàn)解析,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-06-06
詳解Springboot整合ActiveMQ(Queue和Topic兩種模式)
這篇文章主要介紹了詳解Springboot整合ActiveMQ(Queue和Topic兩種模式),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04

