原文链接:公众号狂神说
视频教程[狂神说B站]: https://www.bilibili.com/video/BV12J41137hu
如有侵权,联系立删
Annotation是从JDK5.0开始引入的新技术).
Annotation的作用 :
Annotation的格式:
注解是以"@注释名"在代码中存在的, 还可以添加一些参数值,例如:@SuppressWarnings(value=“unchecked”).
Annotation在哪里使用?
可以附加在package , class , method , field等上面,相当于给他们添加了额外的辅助信息,我们可以通过反射机制编程实现对这些元数据的访问。
@Override : 定义在java.lang.Override中,此注释只适用于修辞方法, 表示一个方法声明打算重写超类中的另一个方法声明.
@Deprecated :定义在java.lang.Deprecated中,此注释可以用于修辞方法,属性,类,表示不鼓励程序员使用这样的元素,通常是因为它很危险或者存在更好的选择.
@SuppressWarnings : 定义在java.lang.SuppressWarnings中,用来抑制编译时的警告信息,与前两个注释有所不同,你需要添加一-个参数才能正确使用,这些参数都是已经定义好了的,我们选择性的使用就好了.
@SuppressWarnings(“l”)
@SuppressWarnings(“unchecked”)
@SuppressWarnings(value={“unchecked”,“deprecation”})
public class Demo01 extends Object{
@Override//重写方法
public String toString() {
return super.toString();
}
@Deprecated//表示弃用方法
public static void test(){
}
@SuppressWarnings("all")//抑制警告
public static void test01(){
int age;
}
public static void main(String[] args) {
test();
test01();
}
}
import java.lang.annotation.*;
@MyAnnotation
public class Demo02 {
void test(){
}
}
//定义一个注解
//Target 表示我们的注解可以用在哪些地方.
@Target(value = {ElementType.METHOD, ElementType.TYPE})
//Retention表示我们的注解在什么地方还有效。
// runtime>class>sources
@Retention(value = RetentionPolicy.RUNTIME)
//Documented表示是否将我们的注解生成在Javadoc中
@Documented
//Inherited子类可以继承父类的注解
@Inherited
@interface MyAnnotation{ }
import java.lang.annotation.*;
public class Demo03 {
public static void main(String[] args) {
}
//注解可以显示赋值,如果没有默认值 ,我们就必须给注解赋值
@MyAnnotation2(name = "赵大宝",age = 12)
public void test(){}
@MyAnnotation3("baobao")//参数只有一个,且参数名为value
public void test1(){}
}
@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation2{
//注解的参数:参数类型+参数名();
String name() default "";
int age() default 0;
int id() default -1;// 如果默认值为-1,代表不存在。
String[] schools() default {"清华大学,辽宁大学"}
}
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation3{
//只有一个参数时,参数名为value时,使用时不需参数名
String value();
}
Reflection (反射)是Java被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息,并能***直接操作任意对象的内部属性及方法***。
Class C= Class forName("java.lang String")
加载完类之后,在堆内存的方法区中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就包含了完整的类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一-面镜子, 透过这个镜子看到类的结构,所以我们形象的称之为:反射。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-efxKF0g6-1590302922298)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590050829899.png)]
public class Demo01 {
public static void main(String[] args) throws ClassNotFoundException {
/*通过反射获取类的Class对象,
*一个类只有一个Class对象,所以c1,c2,c3的hashcode相同
*一个类被加载后,整个类的结构都会被封装在Class对象中
*/
Class c1 = Class.forName("com.reflection.User");
System.out.println(c1.getName());//com.reflection.User
Class c2 = Class.forName("com.reflection.User");
System.out.println(c2.hashCode());
Class c3 = Class.forName("com.reflection.User");
System.out.println(c3.hashCode());
}
}
//实体类:pojo,entity,只有属性
class User{
private String name;
private int age;
public User(){
}
public User(String name,int age){
this.age = age;
this.name = name;
}
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
public final Class getClass()
Class类的常用方法
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Ec1yEKjO-1590302922318)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590052537141.png)]
获取Class类的实例
a)若已知具体的类, 通过类的class属性获取, 该访法最为安全可靠,程序性能最高。
Class class = Person.class;
b)已知某个类的实例, 调用该实例的getClass()方法获取Class对象
Class class = person.getClass();
c)已知一 个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取可能抛出ClassNotFoundException
Class class = Class forName("demo01. Student");
d)内置基本数据类型可以直接用类名.Type
e)还可以利用ClassLoader我们之 后讲解
public class Demo02 {
public static void main(String[] args) throws ClassNotFoundException {
Person s1 = new Student();
System.out.println("这个人是"+s1.name);
//方式一:通过对象获取
Class c1 = s1.getClass();
System.out.println(c1.hashCode());
//方式二:通过forname获取
Class c2 = Class.forName("com.reflection.Student");
System.out.println(c2.hashCode());
//通过类名.class获得
Class c3 = Student.class;
System.out.println(c3.hashCode());
//方式四:基本内置类型的包装类都有一个TYPE属性
Class c4 = Integer.TYPE;
System.out.println(c4);
//获得父类类型
Class c5 = c1.getSuperclass();
System.out.println(c5);
}
}
class Person{
public String name;
public Person(){}
public Person(String name){
this.name = name;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
'}';
}
}
class Student extends Person{
public Student(){
this.name = "Student";
}
}
class Teacher extends Person{
public Teacher(){
this.name = "Teacher";
}
}
哪些类型可以有Class对象?
class: 外部类成员(成员内部类,静态内部类),局部内部类,匿名内部类。
interface: 接口
[]:数组
enum:枚举
annotation:注解@interface
primitive type:基本数据类型
void
import java.lang.annotation.ElementType;
public class Demo03 {
public static void main(String[] args) {
Class c1 = Object.class; //类 .
Class c2 = Comparable.class; //按口
Class c3 = String[].class; //- -维数组
Class c4 = int[][].class; //二维数组
Class c5 = Override.class; //注解
Class c6 = ElementType.class; //枚举
Class c7 = Integer.class; //基本数据类型
Class c8 = void.class; //void
Class c9 = Class.class; //Class
System.out.println(c1);//class java.lang.Object
System.out.println(c2);//interface java.lang.Comparable
System.out.println(c3);//class [Ljava.lang.String;
System.out.println(c4);//class [[I
System.out.println(c5);//interface java.lang.Override
System.out.println(c6);//class java.lang.annotation.ElementType
System.out.println(c7);//class java.lang.Integer
System.out.println(c8);//void
System.out.println(c9);//class java.lang.Class
//只要元素类型与维度一样,就是同一个Class.
int[] a = new int[10];
int[] b = new int[100];
System.out.println(a.getClass().hashCode());//1163157884
System.out.println(b.getClass().hashCode());//1163157884
}
}
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-nctbmRul-1590302922344)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590112714558.png)]
类的加载与ClassLoader的理解
加载:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后生成一个代表这个类的java.lang.Class对象.
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DLX1D3VW-1590302922346)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590113106107.png)]
链接:将Java类的二进制代码合并到JVM的运行状态之中的过程。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-yNUOa4yI-1590302922349)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590113167821.png)]
初始化:
public class Demo04 {
public static void main(String[] args) {
A a=new A();
System.out.println(A.m);//100
/*
1.加载到内存,产生一个类对应Class对象
2.链接,链接结束后m=◎
3.初始化
(){
System. out . println( "A类静态代码块初始化") ;
m = 300;
m=100;
}
m=100
*/
}
}
class A {
static {
System.out.println("A类静态代码块初始化");
m = 300;
}
static int m = 100;
public A() {
System.out.println("A类的参构造初始化");
}
}
public class Demo05 {
static {
System.out.println("main类被加载");
}
public static void main(String[] args) throws ClassNotFoundException {
//1.主动引用
//Son son = new Son();
/*
* main类被加载
* 父类被加载
* 子类被加载
* */
//反射也会产生主动引用
//Class.forName ("com.reflection.Son");
/*
* main类被加载
* 父类被加载
* 子类被加载
* */
//不会产生类的引用的方法
//System.out.println(Son.b);
/*
* main类被加载
* 父类被加载
* 2
* */
//Son[] array = new Son[5];
/*main类被加载*/
System.out.println(Son.M) ;
/*
* main类被加载
* 1
* */
}
}
class Father {
static int b = 2;
static {
System.out.println("父类被加载");
}
}
class Son extends Father {
static {
System.out.println("子类被加载");
}
static int m = 100;
static final int M = 1;
}
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qdsOBXFP-1590302922352)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590114543252.png)]
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-UG5ZxZUu-1590302922355)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590114571976.png)]
package com.reflection;
public class Demo06 {
public static void main(String[] args) throws ClassNotFoundException {
//获取系统的类加载器
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
System.out.println(systemClassLoader);//sun.misc.Launcher$AppClassLoader@18b4aac2
//获取系统类加载器的父类加载器-->扩展类加载器
ClassLoader parent = systemClassLoader.getParent();
System.out.println(parent);//sun.misc.Launcher$ExtClassLoader@4554617c
//获取扩展类加载器的父类加载器- ->根加载器(C/c++)
ClassLoader grantparent = parent.getParent();
System.out.println(grantparent);//null
//测试当前类是哪个加载器加载的
ClassLoader classLoader = Class.forName("com.reflection.Demo06").getClassLoader();
System.out.println(classLoader);//sun.misc.Launcher$AppClassLoader@18b4aac2
//测试JDK内置的类是谁加载的
ClassLoader classLoader1 = Class.forName("java.lang.Object").getClassLoader();
System.out.println(classLoader1);//null
//如何获得系统类加载器可以加载的路径
System.out.println(System.getProperty("java.class.path"));
/*
* D:\Environment\java\jre\lib\charsets.jar;
* D:\Environment\java\jre\lib\deploy.jar;
* D:\Environment\java\jre\lib\ext\access-bridge-64.jar;
* D:\Environment\java\jre\lib\ext\cldrdata.jar;
* D:\Environment\java\jre\lib\ext\dnsns.jar;
* D:\Environment\java\jre\lib\ext\jaccess.jar;
* D:\Environment\java\jre\lib\ext\jfxrt.jar;
* D:\Environment\java\jre\lib\ext\localedata.jar;
* D:\Environment\java\jre\lib\ext\nashorn.jar;
* D:\Environment\java\jre\lib\ext\sunec.jar;
* D:\Environment\java\jre\lib\ext\sunjce_provider.jar;
* D:\Environment\java\jre\lib\ext\sunmscapi.jar;
* D:\Environment\java\jre\lib\ext\sunpkcs11.jar;
* D:\Environment\java\jre\lib\ext\zipfs.jar;
* D:\Environment\java\jre\lib\javaws.jar;
* D:\Environment\java\jre\lib\jce.jar;
* D:\Environment\java\jre\lib\jfr.jar;
* D:\Environment\java\jre\lib\jfxswt.jar;
* D:\Environment\java\jre\lib\jsse.jar;
* D:\Environment\java\jre\lib\management-agent.jar;
* D:\Environment\java\jre\lib\plugin.jar;
* D:\Environment\java\jre\lib\resources.jar;
* D:\Environment\java\jre\lib\rt.jar;
* D:\project\java-design\out\production\design02;
* D:\IDEA\IntelliJ IDEA 2020.1.1\lib\idea_rt.jar
*/
}
}
package com.reflection;
import sun.reflect.misc.MethodUtil;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Member;
import java.lang.reflect.Method;
//获得类的信息
public class Demo07 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {
Class c1 = Class.forName("com.reflection.User");
//获得包名+类名
User user = new User();
Class c2 = user.getClass();
//获得类的信息
System.out.println(c1.getName());//获得包名+类名
System.out.println(c1.getSimpleName());//获得类名
//获得类的属性
System.out.println("=======================");
Field[] fields = c1.getFields();//获取类的公开属性和父类的公开属性
fields = c1.getDeclaredFields();//获取类的任何属性
for (Field field : fields) {
System.out.println(field);
}
//获得指定属性的值
Field name = c1.getDeclaredField("name");
System.out.println(name);
//获得类的方法
System.out.println("=========================");
Method[] methods = c1.getMethods();//获得本类和父类的所有public方法
for (Method method : methods) {
System.out.println("methods " + method);
}
System.out.println("=========================");
Method[] decmethods = c1.getDeclaredMethods();//获得本类的所有方法
for (Method method : decmethods) {
System.out.println("decmethods " + method);
}
//获得指定方法
//需要传参数的原因:存在重载,参数可找到指定的方法
System.out.println("=========================");
Method getName = c1.getMethod("getName", null);
Method setName = c1.getMethod("setName", String.class);
System.out.println(getName);
System.out.println(setName);
//获得构造器
System.out.println("=========================");
Constructor[] constructors = c1.getConstructors();
for (Constructor constructor : constructors) {
System.out.println("getConstructors " + constructor);
}
System.out.println("=========================");
Constructor[] constructors1 = c1.getDeclaredConstructors();
for (Constructor constructor : constructors1) {
System.out.println("getDeclaredConstructors " + constructors1);
}
//获得指定的构造器
Constructor getDeclaredConstructor = c1.getDeclaredConstructor(String.class, int.class);
System.out.println("指定构造器" + getDeclaredConstructor);
}
}
创建类的对象:调用Class对象的newInstance()方法
1)类必须有一个无参数的构造器。
思考?难道没有无参的构造器就不能创建对象了吗?
只要在操作的时候明确的调用类中的构造器,并将参数传递进去之后,才可以实例化操作。步骤如下:
1)通过Class类的getDeclaredConstructor(Class … parameterTypes)取得本类的指定形参类型的构造器
2)向构造器的形参中传递一一个对象数组进去,里面包含了构造器中所需的各个参数。
3)通过Constructor实例化对象
通过反射,调用类中的方法,通过Method类完成。
①通过Class类的getMethod(String name,Clas…parameterTypes)方法取得一个Method对象,并设置此方法操作时所需要的参数类型。
②之后使用Object invoke(Object obj, Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-s51WirZQ-1590302922357)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\1590121229558.png)]
Object invoke(Object obj, Object ... args)
package com.reflection;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
//动态创建对象,通过反射
public class Demo08 {
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
//获得class对象
Class c1 = Class.forName("com.reflection.User");
//创建一个对象
System.out.println("============================");
User user = (User)c1.newInstance();//本质是调用了类的无参构造器
System.out.println(user);
//通过构造器创建对象
System.out.println("============================");
Constructor constructor = c1.getDeclaredConstructor(String.class,int.class);
User user2 = (User)constructor.newInstance("打爆",22);
System.out.println(user2);
//通过反射调用普通方法
//通过反射获取一个方法
System.out.println("============================");
Method setName = c1.getDeclaredMethod("setName",String.class);
//invoke:激活的意思
//参数:对象,方法的值
setName.invoke(user,"韩威");
System.out.println(user.getName());
System.out.println("============================");
//通过反射操作属性
User user3 = (User)c1.newInstance();
Field name = c1.getDeclaredField("name");
//不能直接操作私有属性,我们需要关闭程序的安全监测,属性或方法的setAccessible(true)
name.setAccessible(true);
name.set(user3,"小宝");
System.out.println(user3.getName());
}
}
package com.reflection;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Demo09 {
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {
test1();//5ms
test2();//4114ms
test3();//1483ms
}
public static void test1(){
User user = new User();
long start = System.currentTimeMillis();
for (int i = 0; i < 1000000000; i++) {
user.getName();
}
long end = System.currentTimeMillis();
System.out.println(end-start+"ms");
}
public static void test2() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
User user = new User();
Class c1 = user.getClass();
Method getName = c1.getDeclaredMethod("getName",null);
long start = System.currentTimeMillis();
for (int i = 0; i < 1000000000; i++) {
getName.invoke(user,null);
}
long end = System.currentTimeMillis();
System.out.println(end-start+"ms");
}
public static void test3() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
User user = new User();
Class c1 = user.getClass();
Method getName = c1.getDeclaredMethod("getName",null);
getName.setAccessible(true);
long start = System.currentTimeMillis();
for (int i = 0; i < 1000000000; i++) {
getName.invoke(user,null);
}
long end = System.currentTimeMillis();
System.out.println(end-start+"ms");
}
}
package com.reflection;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Map;
public class Demo10 {
public void test01 (Map<String,User> map, List<User> list){
System.out.println("test01");
}
public Map<String, User> test02(){
System.out.println("test02");
return null;
}
public static void main(String[] args) throws NoSuchMethodException {
Method method = Demo10.class.getMethod("test01",Map.class,List.class);
Type[] genericParameterTypes= method.getGenericParameterTypes();
for (Type genericParameterType : genericParameterTypes) {
System.out.println("参数范型"+genericParameterType);
if (genericParameterType instanceof ParameterizedType){
Type[] actualTypeAnguments=((ParameterizedType)(ParameterizedType) genericParameterType).getActualTypeArguments();
for (Type actualTypeAngument : actualTypeAnguments) {
System.out.println("实际参数范型"+actualTypeAngument);
}
}
}
Method method1 = Demo10.class.getMethod("test02",null);
Type getGenericReturnType= method1.getGenericReturnType();
if (getGenericReturnType instanceof ParameterizedType) {
Type[] actualTypeArguments = ((ParameterizedType) getGenericReturnType).getActualTypeArguments();
for (Type actualTypeArgument : actualTypeArguments) {
System.out.println("返回值范型" + actualTypeArgument);
}
}
}
}
package com.reflection;
import com.sun.xml.internal.ws.api.ha.StickyFeature;
import java.lang.annotation.*;
import java.lang.reflect.Field;
import java.security.cert.CRLReason;
public class Demo11 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
Class c1 = Class.forName("com.reflection.Student2");
//通过反射获取注解
Annotation[] annotations = c1.getAnnotations();
for (Annotation annotation : annotations) {
System.out.println(annotation);//@com.reflection.Table(value=db_student)
}
//获得注解中的value值
Table table = (Table)c1.getAnnotation(Table.class);
String value = table.value();
System.out.println(value);//db_student
//获得类指定的注解
Field f = c1.getDeclaredField("name");
Filed annotation=f.getAnnotation(Filed.class);
System.out.println(annotation.columnName());//db_name
System.out.println(annotation.length());//3
System.out.println(annotation.type());//varchar
}
}
@Table("db_student")
class Student2{
@Filed(columnName = "db_id",type = "int",length = 10)
private int id;
@Filed(columnName = "db_age",type = "int",length = 10)
private int age;
@Filed(columnName = "db_name",type = "varchar",length = 3)
private String name;
public Student2(){}
public Student2(int id, int age, String name) {
this.id = id;
this.age = age;
this.name = name;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
//类名的注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface Table{
String value();
}
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface Filed{
String columnName();
String type();
int length();
}
~~~java
e;
@Filed(columnName = "db_name",type = "varchar",length = 3)
private String name;
public Student2(){}
public Student2(int id, int age, String name) {
this.id = id;
this.age = age;
this.name = name;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
//类名的注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface Table{
String value();
}
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface Filed{
String columnName();
String type();
int length();
}
B站地址