首页
统计
关于
Search
1
Sealos3.0离线部署K8s集群
1,073 阅读
2
类的加载
737 阅读
3
Spring Cloud OAuth2.0
725 阅读
4
SpringBoot自动装配原理
689 阅读
5
集合不安全问题
582 阅读
笔记
Java
多线程
注解和反射
JVM
JUC
设计模式
Mybatis
Spring
SpringMVC
SpringBoot
MyBatis-Plus
Elastic Search
微服务
Dubbo
Zookeeper
SpringCloud
Nacos
Sentinel
数据库
MySQL
Oracle
PostgreSQL
Redis
MongoDB
工作流
Activiti7
Camunda
消息队列
RabbitMQ
前端
HTML5
CSS
CSS3
JavaScript
jQuery
Vue2
Vue3
Linux
容器
Docker
Kubernetes
Python
登录
Search
标签搜索
Java
CSS
mysql
RabbitMQ
JavaScript
Redis
JVM
Mybatis-Plus
Camunda
多线程
CSS3
Python
Spring Cloud
注解和反射
Activiti
工作流
SpringBoot
Mybatis
Spring
html5
蘇阿細
累计撰写
388
篇文章
累计收到
4
条评论
首页
栏目
笔记
Java
多线程
注解和反射
JVM
JUC
设计模式
Mybatis
Spring
SpringMVC
SpringBoot
MyBatis-Plus
Elastic Search
微服务
Dubbo
Zookeeper
SpringCloud
Nacos
Sentinel
数据库
MySQL
Oracle
PostgreSQL
Redis
MongoDB
工作流
Activiti7
Camunda
消息队列
RabbitMQ
前端
HTML5
CSS
CSS3
JavaScript
jQuery
Vue2
Vue3
Linux
容器
Docker
Kubernetes
Python
页面
统计
关于
搜索到
8
篇与
的结果
2021-08-23
类的加载
加载将class字节码文件加载到内存中,并将这些静态数据转换为方法区的运行时数据结构,然后生成一个代表这个类的java.lang.Class对象链接将Java类的二进制代码合并到JVM的运行状态之中的过程验证:确保加载的类信息符合JVM规范准备:正式为类变量(static)分配内存并设置类变量默认初始值的阶段,这些内存都将在方法区中进行分配解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程初始化执行类构造器<clinit>()方法的过程,类构造器<clinit>()方法是由编译器自动收集类中所有类变量的赋值动作和静态代码块中的赋值语句产生的。(类构造器构造类的信息,不是构造该类对象的构造器)当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确加锁和同步
2021年08月23日
737 阅读
0 评论
0 点赞
2020-11-28
动态创建对象执行方法
动态创建对象执行方法1、创建类的对象:调用Class对象的newInstance()方法类必须有无参构造器需要构造器的访问权限 //获得Class对象 Class c1 = Class.forName("com.sw.reflection.User"); //构造一个对象 User user = (User) c1.newInstance(); //本质是调用类的无参构造 System.out.println(user);2、在没有无参构造器的情况下,只要在操作的时候明确的调用类中的构造器并将参数传递进去,就可以实例化操作通过Class类的getDeclaredConstructor(Class ... parameterTypes)取得本类的指定形参类型的构造器向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数通过Constructor实例化对象 //获得Class对象 Class c1 = Class.forName("com.sw.reflection.User"); //通过构造器创建对象 Constructor constructor = c1.getDeclaredConstructor(int.class, String.class, int.class); User user1 = (User) constructor.newInstance(1, "孙笑川", 30); System.out.println(user1);3、调用指定的方法通过反射调用类中的方法,通过Method类完成通过Class类的getMethod(String name,parameterTypes)方法取得一个Method对象,并设置此方法操作时所需要的参数类型之后使用Object invoke(Object obj,Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息Object invoke(Object obj,Object[] args)Object对应原方法的返回值,若原方法无返回值,则返回null若原方法为静态方法,此时形参Object obj可为null若原方法形参列表为空,则Object[] args为null若原方法声明为private,则需要在调用invoke方法前,显示调用方法对象的setAccessible(true)方法,关闭程序的安全检查setAccessibleMethod和Field、Constructor对象都有setAccessible方法其作用是启动和禁止访问安全检查的开关参数值为true时指反射的对象在使用时应该取消Java语言访问检查提高反射的效率使原本无法访问的private私有成员也可以访问参数值为fasle指反射的对象应执行Java语言访问检查(默认值为false) //获得Class对象 Class c1 = Class.forName("com.sw.reflection.User"); //通过反射调用普通方法 User user2 = (User) c1.newInstance(); //通过反射获取一个方法 Method setName = c1.getDeclaredMethod("setName", String.class); //invoke 激活 //.invoke(对象,"值") setName.invoke(user2,"孙笑川"); System.out.println(user2.getName());整个实例Demo:package com.sw.reflection; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; /** * @Author suaxi * @Date 2020/11/28 14:31 * 通过反射,动态的创建对象 */ public class Test09 { public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException { //获得Class对象 Class c1 = Class.forName("com.sw.reflection.User"); //构造一个对象 //User user = (User) c1.newInstance(); //本质是调用类的无参构造 //System.out.println(user); //通过构造器创建对象 //Constructor constructor = c1.getDeclaredConstructor(int.class, String.class, int.class); //User user1 = (User) constructor.newInstance(1, "孙笑川", 30); //System.out.println(user1); //通过反射调用普通方法 User user2 = (User) c1.newInstance(); //通过反射获取一个方法 Method setName = c1.getDeclaredMethod("setName", String.class); //invoke 激活 //.invoke(对象,"值") setName.invoke(user2,"孙笑川"); System.out.println(user2.getName()); //通过反射操作属性 User user3 = (User) c1.newInstance(); Field name = c1.getDeclaredField("name"); //不能直接操作私有属性,需关掉程序的安全检测,属性/方法.setAccessible(true); name.setAccessible(true); name.set(user3,"药水哥"); System.out.println(user3.getName()); } }
2020年11月28日
77 阅读
0 评论
0 点赞
2020-11-27
类加载器
类加载器1、作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口2、类缓存:标准的JavaSE类加载器可以按要求查找类,但一旦某个类被加载到类加载器中,它将维持加载(缓存)一段时间,JVM垃圾回收机制可以回收这些Class对象3、类加载器作用是用来把类(class)装载进内存,JVM规范定义了如下类型的类加载器package com.sw.reflection; /** * @Author suaxi * @Date 2020/11/27 21:29 */ public class Test07 { public static void main(String[] args) throws ClassNotFoundException { //获取系统类的加载器 ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader(); System.out.println(systemClassLoader); //获取系统类加载器的父类加载器 --->扩展类加载器 ClassLoader parent = systemClassLoader.getParent(); System.out.println(parent); //获取扩展类加载器的父类加载器 --->根加载器(C/c++编写,无法直接获取) ClassLoader parent1 = parent.getParent(); System.out.println(parent1); //测试当前类是哪个加载器加载的 ClassLoader classLoader = Class.forName("com.sw.reflection.Test07").getClassLoader(); System.out.println(classLoader); //测试JDK内置的类是谁加载的 classLoader = Class.forName("java.lang.Object").getClassLoader(); System.out.println(classLoader); //获取系统类加载器可以加载的路径 System.out.println(System.getProperty("java.class.path")); } }
2020年11月27日
69 阅读
0 评论
0 点赞
2020-11-27
类初始化
什么时候会发生类初始化?1、主动引用(一定会发生类的初始化)当虚拟机启动时,先初始化main方法所在的类new一个对象时调用类的静态成员(final常量除外)和静态方法使用java.lang.reflection包的方法对类进行反射调用当初始化一个类,如果其父类没有被初始化,则会先初始化它的父类2、被动引用(不会发生类的初始化)当访问一个静态域时,只有真正声明这个域的类才会被初始化。如:通过子类引用父类的静态变量,不会导致子类被初始化通过数组定义类引用引用常量(常量在链接阶段就已经存入调用类的常量池中了)package com.sw.reflection; import java.lang.reflect.Array; /** * @Author suaxi * @Date 2020/11/27 17:21 * 测试类什么时候会初始化 */ public class Test06 { static { System.out.println("Main方法被加载"); } public static void main(String[] args) throws ClassNotFoundException { //1、主动引用 //Son son = new Son(); //2、反射(也会产生主动引用) //Class.forName("com.sw.reflection.Son"); //不会产生类的 引用的方法 //System.out.println(Son.b); //数组 //Son[] s = new Son[5]; //常量 System.out.println(Son.M); } } class Father{ static int b = 2; static { System.out.println("父类被加载"); } } class Son extends Father{ static { System.out.println("子类被加载"); m =300; } static int m = 100; static final int M = 1; }主动引用反射子类引用父类的静态变量数组常量(子类中的final类)
2020年11月27日
65 阅读
0 评论
0 点赞
2020-11-27
类的加载与ClassLoader
类的加载与ClassLoader1、加载:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,之后生成一个代表这个类的java.lang.Class对象2、链接:将Java类的二进制代码合并到JVM的运行状态之中的过程验证:确保加载的类信息符合JVM规范准备:正式为类变量(static)分配内存并设置类变量默认初始值的阶段,注:这些内存都将在方法区中进行分配解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程3、初始化:执行类构造器()方法的过程。类构造器()方法是由编译器自动收集类中所有类变量的赋值动作和静态代码块中的语句所产生的(类构造器是构造类信息的,不是构造该类对象的构造器)当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化虚拟机会保证一个类的()方法在多线程环境中被正确加锁和同步package com.sw.reflection; /** * @Author suaxi * @Date 2020/11/27 16:56 */ public class Test05 { public static void main(String[] args) { A a = new A(); System.out.println(a.m); /* 1、加载到内存,产生一个Class对象 2、链接,链接结束后 m=0 3、初始化 <clinit>{ System.out.println("A类静态代码块初始化"); m = 300; } m = 100; */ } } class A{ static { System.out.println("A类静态代码块初始化"); m = 300; } static int m = 100; public A(){ System.out.println("A类的无参构造初始化"); } }
2020年11月27日
179 阅读
0 评论
0 点赞
2020-11-27
反射
动态语言在运行时可以改变其结构的语言,即在运行时代码可以根据某些条件改变自身结构主要的动态语言有:Object-C、C#、JavaScript、PHP、Python等静态语言与动态语言相反,如:C、C++、Java等Java具有一定的动态性,即反射机制获取类似动态语言的特性Java Reflection(反射)Reflection是Java被视为动态语言的关键,反射机制允许程序在执行期间借助Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。Class c = Class.forName("java.lang.String")加载完类之后,在堆内存的方法中就产生了一个Class类型的对象,这个对象就包含了完整的类的结构信息。可以通过这个对象看到类的结构——反射。正常方式:引入需要的”包类“名称 --->通过new实例化 --->取得实例化对象反射方式:实例化对象 --->getClass()方法 --->得到完整的”包类“名称package com.sw.reflection; /** * @Author suaxi * @Date 2020/11/27 15:58 */ public class Test01 extends Object{ public static void main(String[] args) throws ClassNotFoundException { //通过反射获取类的Class对象 Class<?> c1 = Class.forName("com.sw.reflection.Test01"); System.out.println(c1); //一个类在内存中只有一个Class对象 //一个类被加载后,类的整个结构都会被封装在Class对象中 Class<?> c2 = Class.forName("com.sw.reflection.Test01"); Class<?> c3 = Class.forName("com.sw.reflection.Test01"); System.out.println(c2.hashCode()); System.out.println(c3.hashCode()); } } //实体类 class User{ int id; String name; int age; public User() { } public User(int id, String name, int age) { this.id = id; this.name = name; this.age = age; } public int getId() { return id; } public void setId(int id) { this.id = id; } 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; } @Override public String toString() { return "User{" + "id=" + id + ", name='" + name + '\'' + ", age=" + age + '}'; } }
2020年11月27日
62 阅读
0 评论
0 点赞
2020-11-27
Class类
在Object类中定义了以下方法,此方法将被所有子类继承public final Class getClass()以上的方法返回的类型是一个Class类,是Java反射的源头Class本身也是一个对象Class对象只能由系统创建一个加载的类在JVM中只会有一个Class实例一个Class对象对应的是一个加载到JVM中的一个.class文件每个类的实例都会记得自己是由哪一个Class生成的通过Class可以完整的得到一个类中的所有被加载的结构Class类是Reflection的根源Class类的常用方法方法名说明static ClassforName(String name)返回指定类名name的对象Object newInstance()调用缺省构造函数,返回Class对象的一个实例getName()返回此Class对象所表示的实体的名称Class getSuperClass()返回当前Class对象的父类的Class对象Class[] getinterfaces()返回当前Class对象的接口ClassLoader getClassLoader()返回该类的类加载器Constructor[] getConstructors返回一个包含某些Constructor对象的数组Method getMethod(String name,Class.. T)返回一个Method对象,此对象的形参类型为paramTypeField[] getDeclaredFields()返回Field对象的一个数组获取Class类的实例1、若已知具体的类,通过类的class属性获取,该方法最为安全可靠,程序性能最高Class c = Person.class;2、已知某个类的实例,调用该实例的getClass()方法获取Class对象Class c = person.getClass();3、已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,可能抛出ClassNotFoundException异常Class c = Class.forName("com.sw.xxx.xx");4、内置基本数据类型可以直接用类名.TYPE5、还可以用ClassLoader获取package com.sw.reflection; /** * @Author suaxi * @Date 2020/11/27 16:17 * Class类的创建方式 */ public class Test02 { public static void main(String[] args) throws ClassNotFoundException { Person p = new Student(); System.out.println("孙笑川是"+p.getName()); //方式一:通过对象获得 Class c1 = p.getClass(); System.out.println(c1.hashCode()); //方式二:forname获得 Class c2 = Class.forName("com.sw.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{ String name; public Person() { } public Person(String name) { this.name = name; } public String getName() { return name; } public void setName(String name) { this.name = name; } @Override public String toString() { return "Person{" + "name='" + name + '\'' + '}'; } } class Student extends Person{ public Student() { this.name = "学生"; } }
2020年11月27日
47 阅读
0 评论
0 点赞
2020-11-27
自定义注解
使用@interface自定义注解时,自动继承了java.lang.annotation.Annotation接口@interface用来声明一个注解,格式:public @interface 注解名{定义的内容}其中的每一个方法实际上是声明了一个配置参数方法的名称就是参数的名称返回值类型就是参数的类型(返回值只能说基本类型Class、String、int等)可以通过default来声明参数的默认值如果只有一个参数成员,参数名一般设置为value注解元素必须要有值,我们定义注解元素时,经常使用空字符串,0作为默认值package com.annotation.demo01; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * @Author suaxi * @Date 2020/11/27 11:10 * 自定义注解 */ public class Test01 { //注解可以显示赋值,如果没有默认值,我们就必须给注解赋值 @MyAnnotation(name = "孙笑川",schools = {"xx大学,xx小学"}) public void test(){ } //@MyAnnotation2(value = "药水哥") @MyAnnotation2("药水哥")//注解中默认值为value的时候value可以省略不写 public void test2(){ } } @Target({ElementType.TYPE,ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME) @interface MyAnnotation{ //注解的参数:参数类型 + 参数名() String name() default ""; int age() default 0; int id() default -1; //如果默认值为-1,表示不存在 String[] schools(); } @Target({ElementType.TYPE,ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME) @interface MyAnnotation2{ //如果只有一个参数成员,参数名一般设置为value String value(); }
2020年11月27日
61 阅读
0 评论
0 点赞