笑看天下无敌手 发表于 2024-6-29 02:33:41

程序人生——Java泛型和反射的使用发起

https://img-blog.csdnimg.cn/ca5e093b49a440438db5ff1c578acd1e.png


引出

程序人生——Java泛型和反射的使用发起
泛型和反射

发起93:Java的泛型是范例擦除的



[*]参加泛型优点:加强了参数范例的安全性,减少了范例的转换。Java的泛型在编译期有效,在运行期被删除,也就是说所有的泛型参数范例在编译后都会被清撤除。所以:1、泛型的class对象时是雷同的;2、泛型数组初始化时不能声明泛型范例;3、instanceof不允许存在泛型参数
发起94:不能初始化泛型参数和数组



[*]泛型范例在编译期被擦除,在类初始化时将无法获得泛型的具体参数,所以泛型参数和数组无法初始化,但是ArrayList却可以,因为ArrayList初始化是向上转型变成了Object范例;须要泛型数组办理办法:只声明,不再初始化,由构造函数完成初始化操纵
发起95:强制声明泛型的实际范例



[*]无法从代码中推断出泛型范例的环境下,即可强制声明泛型范例;方法:List list2 = ArrayUtils.asList();在输入前定义这是一个Integer范例的参数
发起96:不同的场景使用不同的泛型通配符



[*]Java泛型支持通配符(Wildcard),可以单独使用一个“?”表现任意类,也可以使用extends关键字表现某一个类(接口)的子范例,还可以使用super关键字表现某一个类(接口)的父范例。1、泛型结构只参加“读”操纵则限定上界(extends关键字);2、泛型结构只参加“写”操纵则限定下界(使用super关键字);3、如果一个泛型结构既用作“读”操纵也用作“写”操纵则使用确定的泛型范例即可,如List
发起97:警惕泛型是不能协变和逆变的



[*]Java的泛型是不支持协变和逆变的,只是能够实现协变和逆变)(协变和逆变是指宽范例和窄范例在某种环境下(如参数、泛型、返回值)替换或交换的特性。简朴地说,协变是用一个窄范例替换宽范例,而逆变则是用宽范例覆盖窄范例。子类覆写父类返回值范例比父范例变窄,则是协变;子类覆写父范例的参数范例变宽,则是逆变。数组支持协变,泛型不支持协变
发起98:发起采用的次序是List,List,List



[*]List是确定的某一个范例,编码者知道它是一个范例,只是在运行期才确定而已;
[*]List可以进行读写操纵,List<?>是只读范例,因为编译器不知道List中容纳的是什么范例的元素,无法增加、修改,但是能删除
发起99:严酷限定泛型范例采用多重界限



[*]使用“&”符号毗连多个泛型界限,如:
发起100:数组的真实范例必须是泛型范例的子范例



[*]有可能会抛出ClassCastException异常,toArray方法返回后会进行一次范例转换,Object数组转换成了String数组。由于我们无法在运行期获得泛型范例的参数,因此就须要调用者主动传入T参数范例
发起101:注意Class类的特殊性



[*]Java处理的根本机制:先把Java源文件编译成后缀为class的字节码文件,然后再通过ClassLoader机制把这些类文件加载到内存中,末了生成实例执行。Java使用一个元类(MetaClass)来形貌加载到内存中的类数据,这就是Class类,它是一个形貌类的类对象。Class类是“类中类”,具有特殊性:1、无构造函数,不能实例化,Class对象是在加载类时由Java虚拟机通过调用类加载器中的defineClass方法自动构建的;2、可以形貌根本范例,8个根本范例在JVM中并不是一个对象,一样平常存在于栈内存中,但是Class类仍然可以形貌它们,例如可以使用int.class表现int范例的类对象;3、其对象都是单例模式,一个Class的实例对象形貌一个类,而且只形貌一个类,反过来也建立,一个类只有一个Class实例对象。Class类是Java的反射入口,只有在获得了一个类的形貌对象后才能动态地加载、调用,一样平常获得一个Class对象有三种途径:1、类属性方式,如String.class;2、对象的getClass方法,如new String().getClass();3、forName方法重载,如Class.forName(“java.lang.String”)。获得了Class对象后,就可以通过getAnnotation()获得注解,通过个体Methods()获得方法,通过getConstructors()获得构造函数等
发起102:适时选择getDeclaredXXX和getXXX



[*]getMethod方法获得的是所有public访问级别的方法,包括从父类继承的方法,而getDeclaredMethod获得的是自身类的所有方法,包括公用方法、私有方法等,而且不受限于访问权限。Java之所以如许处理,是因为反射本意只是正常代码逻辑的一种增补,而不是让正常代码逻辑产生翻天覆地的改动,所以public的属性和方法最轻易获取,私有属性和方法也可以获取,但要限定本类。如果须要列出所有继承自父类的方法,须要先获得父类,然后调用getDeclaredMethods方法,之后持续递归
发起103:反射访问属性或方法是将Accessible设置为true



[*]通过反射方式执行方法时,必须在invoke之前查抄Accessible属性。而Accessible属性并不是我们语法层级理解的访问权限,而是指是否更轻易获得,是否进行安全查抄。Accessible属性只是用来判断是否须要进行安全查抄的,如果不须要则直接执行,这就可以大幅度地提升系统性能。经过测试,在大量的反射环境下,设置Accessible为true可以提升性能20倍以上
发起104:使用forName动态加载类文件



[*]forName只是加载类,并不执行任何代码)(动态加载(Dynamic Loading)是指在程序运行时加载须要的类库文件,一样平常环境下,一个类文件在启动时或首次初始化时会被加载到内存中,而反射则可以在运行时再决定是否要加载一个类,然后在JVM中加载并初始化。动态加载通常是通过Class.forName(String)实现。一个对象的生成必然会经过一下两个步骤:1、加载到内存中生成Class的实例对象;2、通过new关键字生成实例对象;**动态加载的意义:**加载一个类即表现要初始化该类的static变量,特别是static代码块,在这里我们可以做大量的工作,比如注册本身,初始化环境等,这才是我们重点关注的逻辑
发起105:动态加载不得当数组



[*]通过反射操纵数组使用Array类,不要采用通用的反射处理API)(如果forName要加载一个类,那它首先必须是一个类—8个根本范例排除在外,不是具体的类;其次,它必须具有可追索的类路径,否则会报ClassNotFoundException异常。在Java中,数组是一个非常特殊的类,虽然是一个类,但没有定义类路径。作为forName参数时会抛出ClassNotFoundException异常,缘故原由是:数组虽然是一个类,在声明时可以定义为String[],但编译器编译后会为不同的数组范例生成不同的类,所以要想动态创建和访问数组,根本的反射是无法实现的
发起106:动态代理可以使代理模式更加机动



[*]Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目的类生成代理,避免重复开发。静态代理是通过代理主题脚色(Proxy)和具体主题脚色(Real Subject)共同实现抽象主题脚色(Subject)的逻辑的,只是代理主题脚色把相关的执行逻辑委托给了具体主题脚色而已。动态代理须要实现InvocationHandler接口,必须要实现invoke方法,该方法完成了对真实方法的调用
发起107:使用反射增加装饰模式的普适性



[*]装饰模式(Decorator Pattern)的定义是“动态地给一个对象添加一些额外的职责。就增加功能来说,装饰模式相比于生成子类更为机动”。比力通用的装饰模式,只须要定义被装饰的类及装饰类即可,装饰行为由动态代理实现,实现了对装饰类和被装饰类的完全解耦,提供了系统的扩展性
发起108:反射让模板方法模式更强大



[*]决定使用模板方法模式时,请尝试使用反射方式实现,它会让你的程序更机动、更强大)(模板方法模式(Template Method Pattern)的定义是:定义一个操纵中的算法骨架,将一些步骤耽误到子类中,使子类不改变一个算法的结构即可重定义该算法的某些特定步骤。简朴说,就是父类定义抽象模板作为骨架,此中包括根本方法(是由子类实现的方法,而且在模板方法被调用)和模板方法(实现对根本方法的调度,完成固定的逻辑),它使用了简朴的继承和覆写机制。使用反射后,不须要定义任何抽象方法,只需定义一个根本方法辨别器即可加载复合规则的根本方法
发起109:不须要太多关注反射效率



[*]反射效率低是个真命题,但因为这一点而不使用它就是个假命题)(反射效率相对于正常的代码执行确实低很多(经测试,相差15倍左右),但是它是一个非常有效的运行期工具类
深入认识JVM

JVM内存分配,类加载

Java进阶(1)——JVM的内存分配 & 反射Class类的类对象 & 创建对象的几种方式 & 类加载(何时进入内存JVM)& 注解 & 反射+注解的案例
https://img-blog.csdnimg.cn/2117b24a590d4033a418a2ec9363e6f4.png
创建对象的4种方法总结

Java进阶(4)——结合类加载JVM的过程理解创建对象的几种方式:new,反射Class,克隆clone(拷贝),序列化反序列化
https://img-blog.csdnimg.cn/1169c52d34fe4d06b861f6558ac2dcc1.png
垃圾回收GC

https://img-blog.csdnimg.cn/ddd39783039647a78c079ba05243f233.png
Java进阶(垃圾回收GC)——理论篇:JVM内存模型 & 垃圾回收定位清除算法 & JVM中的垃圾回收器
简介:本篇博客介绍JVM的内存模型,对比了1.7和1.8的内存模型的变化;介绍了垃圾回收的语言发展;论述了定位垃圾的方法,引用计数法和可达性分析发以及垃圾清除算法;然后介绍了Java中的垃圾回收器,由串行、到并行再到并发,末了到G1的演变;末了给出了垃圾回收器的对比和使用指引。
JVM调优,Arthas使用



[*]Java进阶(JVM调优)——阿里云的Arthas的使用 & 安装和使用 & 死锁查找案例,重新加载案例,慢调用分析
[*]Java进阶(JVM调优)——JVM调优参数 & JDK自带工具使用 & 内存溢出和死锁问题案例 & GC垃圾回收
https://img-blog.csdnimg.cn/3aaaf686f0c645d9882292494053605c.png
https://img-blog.csdnimg.cn/8ed33fdc7e274bae8922604965834bc9.png
https://img-blog.csdnimg.cn/7a998fdbba9444518402ef1a919d66ee.png
认识多线程

创建多线程方法+了解线程池

Java进阶(5)——创建多线程的方法extends Thread和implements Runnable的对比 & 线程池及常用的线程池
https://img-blog.csdnimg.cn/01302e5f99ae446ea84c04d4a895ea7a.png
多线程下-1非原子性问题即办理

Java进阶(6)——抢购问题中的数据不安全(非原子性问题)& Java中的synchronize和ReentrantLock锁使用 & 死锁及其产生的条件
https://img-blog.csdnimg.cn/bf0963dc7e104662bc3e49666faba6c2.png
再论线程,创建、生命周期

Java进阶(再论线程)——线程的4种创建方式 & 线程的生命周期 & 线程的3大特性 & 集合中的线程安全问题
重要内容:
1.线程创建的方式,继承Thread类,实现Runable接口,实现Callable接口,采用线程池;
2.线程生命周期: join():运行竣事再下一个, yield():暂时让出cpu的使用权,deamon():守护线程,末了竣事,sleep():如果有锁,不会让出;
3.线程3大特性,原子性,可见性,有序性;
4.list集合中线程安全问题,hash算法问题;
总结

程序人生——Java泛型和反射的使用发起

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: 程序人生——Java泛型和反射的使用发起