---------------------- android培训、java培训、期待与您交流! ----------------------
泛型的基本应用
泛型是java1.5中的一个新特性。没有使用泛型时,只要是对象,不管是什么类型的对象,都可以存储进同一个集合中。使用泛型集合,可以讲一个集合中的元素限定为某种特定类型,集合中只能存储该类型的对象,这样更加安全。并且当集合中获取一个元素时,编译器也可以知道这个对象的类型,因此可以不需要对对象进行强制类型转换了,这样更加方便。
泛型的内部原理
泛型是提供给java编译器使用的,可以限定集合中的输入类型,让编译器防止源程序中的非法输入。但是编译器完成编译工作以后,会去掉类型信息,使程序运行效率不受影响。对于参数化的泛型类型,调用getCass()方法返回的Class对象完全一样。如下:
ArrayList<Integer> arrInt = newArrayList<Integer>();
ArrayList<String> arrStr = newArrayList<String>();
arrInt.getClass() == arrStr.getClass();
结果返回的是true,说明编译之后,这两个对象对应的Class字节码文件相同。
由于编译的过程会去掉泛型的类型信息,所以只要跳过编译阶段,就可以往某个泛型集合中加入其它类型的对象,例如,用反射得到集合,再调用add()方法即可。
注意:在创建数组实例时,数组的元素不能使用泛型,例如,下面的语句会报错:
Vector<Integer>vector[] = new Vector<Integer>[10];
泛型的通配符扩展应用
使用“?”通配符可以引用其他各种参数化的类型,“?”通配符定义的变量主要用作引用,可以调用与参数化无关的方法,不能调用与参数有关的方法。
反省中“?”通配符的扩展,可以使“?”通配符继承某个类或者派生某个类的形式来给通配符指定边界,比如:
限定通配符的上边界:
正确:Vector<? extends Number> x = new Vector<Integer>();
错误:Vector<? extends Number> x = new Vector<String>();
限定通配符的下边界:
正确:Vector<? super Integer> x = new Vector<Number>();
错误:Vector<? super Integer> x = new Vector<Byte>();
自定义泛型方法
定义泛型要注意以下几点:
1.只有引用类型才能作为泛型方法的实际参数。
2.除了在应用泛型时可以使用extends、super限定符,在定义泛型时也可以使用extends、super限定符。并且还可以用“&”符号来指定多个边界,例如:
<Textends Serializable & cloneable> void method(){};
3.普通方法、静态方法和构造方法都可以使用泛型。
4.在泛型中可以同时有多个类型的参数,在定义他们时用逗号分开。例如:
publicstatic <K, V> V getValue(Kkey){return map.get(key);}
类加载器
程序中用到的某一个类,该类经过编译后生成了.class的字节码文件,再被加载进内存中,同时进行一些相应的处理工作的过程,都是由类加载器负责的。
java虚拟机中可以安装多个类加载器,系统默认三个主要的类加载器,分别是Boot
Strap、ExtClassLoader和AppClassLoader,每个类加载器负责加载特定位置的类。
类加载器也是java类,所以也需要被加载器所加载,显然必须有一个类加载器不是java类,这个加载器正是BootStrap,它不需要被别的类加载,它是嵌套在java虚拟机内核中的。当java虚拟机启动以后,这个加载器就默认已经存在了。
类加载器的委托机制:
当java虚拟机要加载一个类时,到底调用哪个类加载器去加载呢?
1.当前线程的类加载器去加载线程中的第一个类;
2.如果A类中引用了B类,java虚拟机将使用加载A类的类加载器来加载类B;
3.还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载指定的类。
每个类加载器加载类时,又先委托给其上级的类加载器,当所有的上级加载器都没有加载到指定类时,又回到发起者类加载器,如果还加载不到指定类,就会抛ClassNotFoundExce ption。
自定义类加载器
自定义的类加载器必须继承ClassLoader该抽象类。详见视频。
---------------------- android培训、java培训、期待与您交流! ----------------------