• 1. 北京传智播客教育 www.itcast.cnJava基础加强讲师:黎活明
  • 2. 北京传智播客教育 www.itcast.cn 课程大纲eclipse的使用技巧 静态导入 可变参数与for循环增强 基本数据类型的自动拆箱与装箱 枚举 反射 JavaBean内省 beanutils工具包 注解 泛型 类加载器 代理
  • 3. 北京传智播客教育 www.itcast.cneclipse的使用技巧快捷键使用技巧:快捷键的位置:window-preferences-General-Keys: 最常用eclipse快捷键 内容提示 Alt+/ 添加块注释 Ctrl+Shift+/ 除去块注释 Ctrl+Shift+\ 复制行 Ctrl+Alt+向下键 更改为大写 Ctrl+Shift+X 更改为小写 Ctrl+Shift+Y 类和方法说明 F2
  • 4. 北京传智播客教育 www.itcast.cn静态导入import static语句导入一个类中的某个或所有静态方法 语法举例: import static java.lang.Math.random; import static java.lang.Math.*; public static void main(String[] args) { System.out.println(random()*100); } import static语句导入一个类中的静态成员变量 语法举例: public class Constant { public static String content = "itcast"; } import static cn.itcast.Constant.content;
  • 5. 北京传智播客教育 www.itcast.cn 增强for循环语法: for ( type 变量名:集合变量名 ) { … } 注意事项: 迭代变量必须在( )中定义! 集合变量可以是数组或实现了Iterable接口的集合类 举例: public static int add(int x,int ...args) { int sum = x; for(int arg:args) { sum += arg; } return sum; }
  • 6. 北京传智播客教育 www.itcast.cn可变参数可变参数的特点: 只能出现在参数列表的最后; ...位于变量类型和变量名之间,前后有无空格都可以; 调用可变参数的方法时,编译器为该可变参数隐含创建一个数组,在方法体中以数组的形式访问可变参数。
  • 7. 北京传智播客教育 www.itcast.cn基本数据类型的自动拆箱与装箱自动装箱: Integer num1 = 12; 自动拆箱: Int num = new Integer(23); 基本Boolean/Byte/Integer(数值范围:-128至127)数据类型的对象缓存: Integer num1 = 12; Integer num2 = 12; System.out.println(num1 == num2); Integer num3 = 129; Integer num4 = 129; System.out.println(num3 == num4);
  • 8. 北京传智播客教育 www.itcast.cn枚举使用枚举的关键技术如下:   1:enum关键字表示枚举类型,它的作用相当于类声明中的class关键字。   2:枚举类型不能有public的构造方法。   3:所有的枚举值都是public、static、final的,这些修饰符都是自动加上,无须程序员手动添加。   4:枚举之间用“,”分开,最好一个枚举值用分号“;”。   5:每一个枚举值是一个枚举类型的实例。   6:可以在枚举类型定义非枚举值变量,这些变量可以使用任何修饰符。   7:变量和方法的定义必须在枚举值后面定义。
  • 9. 北京传智播客教育 www.itcast.cn枚举的高级应用枚举就相当于一个类,其中也可以定义构造方法、成员变量、普通方法和抽象方法。 枚举元素必须位于枚举体中的最开始部分,枚举元素列表的后要有分号与其他成员分隔。把枚举中的成员方法或变量等放在枚举元素的前面,编译器报告错误。 带构造方法的枚举 构造方法必须定义成私有的 如果有多个构造方法,该如何选择哪个构造方法? 枚举元素MON和MON()的效果一样,都是调用默认的构造方法。 带方法的枚举 定义枚举TrafficLamp 实现普通的next方法 实现抽象的next方法:每个元素分别是由枚举类的子类来生成的实例对象,这些子类采用类似内部类的方式进行定义。 增加上表示时间的构造方法 枚举只有一个成员时,就可以作为一种单例的实现方式。
  • 10. 北京传智播客教育 www.itcast.cn反射的基石Class类Java程序中的各个Java类属于同一类事物,描述这类事物的Java类名就是Class。 对比提问:众多的人用一个什么类表示?众多的Java类用一个什么类表示? 人Person Java类Class 对比提问: Person类代表人,它的实例对象就是张三,李四这样一个个具体的人,Class类代表Java类,它的各个实例对象又分别对应什么呢? 对应各个类在内存中的字节码,例如,Person类的字节码,ArrayList类的字节码,等等。 一个类被类加载器加载到内存中,占用一片存储空间,这个空间里面的内容就是类的字节码,不同的类的字节码是不同的,所以它们在内存中的内容是不同的,这一个个的空间可分别用一个个的对象来表示,这些对象显然具有相同的类型,这个类型是什么呢? 如何得到各个字节码对应的实例对象( Class类型) 类名.class,例如,System.class 对象.getClass(),例如,new Date().getClass() Class.forName("类名"),例如,Class.forName("java.util.Date"); 九个预定义Class实例对象: 参看Class.isPrimitive方法的帮助 Int.class == Integer.TYPE 数组类型的Class实例对象 Class.isArray() 总之,只要是在源程序中出现的类型,都有各自的Class实例对象,例如,int[],void…
  • 11. 北京传智播客教育 www.itcast.cn反射 反射就是把Java类中的各种成分映射成相应的java类。例如,一个Java类中用一个Class类的对象来表示,一个类中的组成部分:成员变量,方法,构造方法,包等等信息也用一个个的Java类来表示,就像汽车是一个类,汽车中的发动机,变速箱等等也是一个个的类。表示java类的Class类显然要提供一系列的方法,来获得其中的变量,方法,构造方法,修饰符,包等信息,这些信息就是用相应类的实例对象来表示,它们是Field、Method、Contructor、Package等等。 一个类中的每个成员都可以用相应的反射API类的一个实例对象来表示,通过调用Class类的方法可以得到这些实例对象后,得到这些实例对象后有什么用呢?怎么用呢?这正是学习和应用反射的要点。
  • 12. 北京传智播客教育 www.itcast.cnConstructor类Constructor类代表某个类中的一个构造方法 得到某个类所有的构造方法: 例子:Constructor [] constructors= Class.forName("java.lang.String").getConstructors(); 得到某一个构造方法: 例子: Constructor constructor = Class.forName(“java.lang.String”).getConstructor(StringBuffer.class); //获得方法时要用到类型 创建实例对象: 通常方式:String str = new String(new StringBuffer("abc")); 反射方式: String str = (String)constructor.newInstance(new StringBuffer("abc")); //调用获得的方法时要用到上面相同类型的实例对象 Class.newInstance()方法: 例子:String obj = (String)Class.forName("java.lang.String").newInstance(); 该方法内部先得到默认的构造方法,然后用该构造方法创建实例对象。 该方法内部的具体代码是怎样写的呢?用到了缓存机制来保存默认构造方法的实例对象。
  • 13. 北京传智播客教育 www.itcast.cnField类Field类代表某个类中的一个成员变量 演示用eclipse自动生成Java类的构造方法 问题:得到的Field对象是对应到类上面的成员变量,还是对应到对象上的成员变量?类只有一个,而该类的实例对象有多个,如果是与对象关联,哪关联的是哪个对象呢?所以字段fieldX 代表的是x的定义,而不是具体的x变量。 示例代码: ReflectPoint point = new ReflectPoint(1,7); Field y = Class.forName("cn.itcast.corejava.ReflectPoint").getField("y"); System.out.println(y.get(point)); //Field x = Class.forName("cn.itcast.corejava.ReflectPoint").getField("x"); Field x = Class.forName("cn.itcast.corejava.ReflectPoint").getDeclaredField("x"); x.setAccessible(true); System.out.println(x.get(point)); 作业:将任意一个对象中的所有String类型的成员变量所对应的字符串内容中的"b"改成"a"。
  • 14. 北京传智播客教育 www.itcast.cnMethod类Method类代表某个类中的一个成员方法 得到类中的某一个方法: 例子: Method charAt = Class.forName("java.lang.String").getMethod("charAt", int.class); 调用方法: 通常方式:System.out.println(str.charAt(1)); 反射方式: System.out.println(charAt.invoke(str, 1)); 如果传递给Method对象的invoke()方法的第一个参数为null,这有着什么样的意义呢?说明该Method对象对应的是一个静态方法! jdk1.4和jdk1.5的invoke方法的区别: Jdk1.5:public Object invoke(Object obj,Object... args) Jdk1.4:public Object invoke(Object obj,Object[] args),即按jdk1.4的语法,需要将一个数组作为参数传递给invoke方法时,数组中的每个元素分别对应被调用方法中的一个参数,所以,调用charAt方法的代码也可以用Jdk1.4改写为 charAt.invoke(“str”, new Object[]{1})形式。
  • 15. 北京传智播客教育 www.itcast.cn内省了解JavaBeanJavaBean是一种特殊的Java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有的字段,且方法名符合某种命名规则。 如果要在两个模块之间传递多个信息,可以将这些信息封装到一个JavaBean中,这种JavaBean的实例对象通常称之为值对象(Value Object,简称VO)。这些信息在类中用私有字段来存储,如果读取或设置这些字段的值,则需要通过一些相应的方法来访问,大家觉得这些方法的名称叫什么好呢?JavaBean的属性是根据其中的setter和getter方法来确定的,而不是根据其中的成员变量。如果方法名为setId,中文意思即为设置id,至于你把它存到哪个变量上,用管吗?如果方法名为getId,中文意思即为获取id,至于你从哪个变量上取,用管吗?去掉set前缀,剩余部分就是属性名,如果剩余部分的第二个字母是小写的,则把剩余部分的首字母改成小的。 setId()的属性名id isLast()的属性名last setCPU的属性名是什么?CPU getUPS的属性名是什么?UPS 总之,一个类被当作javaBean使用时,JavaBean的属性是根据方法名推断出来的,它根本看不到java类内部的成员变量。 一个符合JavaBean特点的类可以当作普通类一样进行使用,但把它当JavaBean用肯定需要带来一些额外的好处,我们才会去了解和应用JavaBean!好处如下: 在Java EE开发中,经常要使用到JavaBean。很多环境就要求按JavaBean方式进行操作,别人都这么用和要求这么做,那你就没什么挑选的余地! JDK中提供了对JavaBean进行操作的一些API,这套API就称为内省。如果要你自己去通过getX方法来访问私有的x,怎么做,有一定难度吧?用内省这套api操作JavaBean比用普通类的方式更方便。
  • 16. 北京传智播客教育 www.itcast.cn内省综合案例PropertyDescriptor[] propertyDescriptors = Introspector.getBeanInfo(clazz).getPropertyDescriptors(); for(PropertyDescriptor propertydesc : propertyDescriptors){ Method method = propertydesc.getReadMethod();//获取属性的get方法 }
  • 17. 北京传智播客教育 www.itcast.cn Commons-beanutils将字符串转换成给定类型的值 ConvertUtils.convert(String value, Class clazz) 为bean中的指定属性赋值 BeanUtils.copyProperty(form, "name", "经济"); 将指定bean中的属性值拷贝到目标bean中同名的属性 BeanUtils.copyProperties(dest, orig) 将Map中的值拷贝到目标bean中与key同名的属性 PersonForm form = new PersonForm(); Map properties = new HashMap(); properties.put("id", 56); properties.put("name", "liming"); BeanUtils.populate(form, properties);
  • 18. 北京传智播客教育 www.itcast.cn了解注解及java提供的几个基本注解先通过@SuppressWarnings的应用让大家直观地了解注解: 通过System.runFinalizersOnExit(true);的编译警告引出@SuppressWarnings("deprecation") @Deprecated 直接在刚才的类中增加一个方法,并加上@Deprecated标注,在另外一个类中调用这个方法。 @Override public boolean equals(Reflect other)方法与HashSet结合讲解 总结: 注解相当于一种标记,在程序中加了注解就等于为程序打上了某种标记,没加,则等于没有某种标记,以后,javac编译器,开发工具和其他程序可以用反射来了解你的类及各种元素上有无何种标记,看你有什么标记,就去干相应的事。标记可以加在包,类,字段,方法,方法的参数以及局部变量上。 看java.lang包,可看到JDK中提供的最基本的annotation。
  • 19. 北京传智播客教育 www.itcast.cn自定义注解及其应用定义一个最简单的注解:public @interface MyAnnotation {} 把它加在某个类上:@MyAnnotation public class AnnotationTest{} 用反射进行测试AnnotationTest的定义上是否有@MyAnnotation 根据发射测试的问题,引出@Retention元注解的讲解,其三种取值:RetetionPolicy.SOURCE、RetetionPolicy.CLASS、RetetionPolicy.RUNTIME;分别对应:java源文件-->class文件-->内存中的字节码。 思考:@Override、@SuppressWarnings和@Deprecated这三个注解的属性值分别是什么? 演示和讲解@Target元注解 Target的默认值为任何元素,设置Target等于ElementType.METHOD,原来加在类上的注解就报错了,改为用数组方式设置{ElementType.METHOD,ElementType.TYPE}就可以了。 元注解以及其枚举属性值不用记,只要会看jdk提供那几个基本注解的API帮助文档的定义或其源代码,按图索骥即可查到,或者直接看java.lang.annotation包下面的类。
  • 20. 北京传智播客教育 www.itcast.cn为注解增加基本属性什么是注解的属性 一个注解相当于一个胸牌,如果你胸前贴了胸牌,就是传智播客的学生,否则,就不是。如果还想区分出是传智播客哪个班的学生,这时候可以为胸牌在增加一个属性来进行区分。加了属性的标记效果为:@MyAnnotation(color="red") 定义基本类型的属性和应用属性: 在注解类中增加String color(); @MyAnnotation(color="red") 用反射方式获得注解对应的实例对象后,再通过该对象调用属性对应的方法 MyAnnotation a = (MyAnnotation)AnnotationTest.class.getAnnotation(MyAnnotation.class); System.out.println(a.color()); 可以认为上面这个@MyAnnotation是MyAnnotaion类的一个实例对象 为属性指定缺省值: String color() default "yellow"; value属性: String value() default "zxx"; 如果注解中有一个名称为value的属性,且你只想设置value属性(即其他属性都采用默认值或者你只有一个value属性),那么可以省略value=部分,例如:@MyAnnotation("lhm")。
  • 21. 北京传智播客教育 www.itcast.cn为注解增加高级属性数组类型的属性 int [] arrayAttr() default {1,2,3}; @MyAnnotation(arrayAttr={2,3,4}) 如果数组属性中只有一个元素,这时候属性值部分可以省略大括 枚举类型的属性 Gender gender() ; @MyAnnotation(gender=Gender.MAN) 注解类型的属性: MetaAnnotation annotationAttr() default @MetaAnnotation("xxxx"); @MyAnnotation(annotationAttr=@MetaAnnotation(“yyy”) ) 可以认为上面这个@MyAnnotation是MyAnnotaion类的一个实例对象,同样的道理,可以认为上面这个@MetaAnnotation是MetaAnnotation类的一个实例对象,调用代码如下: MetaAnnotation ma = myAnnotation.annotationAttr(); System.out.println(ma.value());
  • 22. 北京传智播客教育 www.itcast.cn体验泛型Jdk 1.5以前的集合类中存在什么问题 ArrayList collection = new ArrayList(); collection.add(1); collection.add(1L); collection.add("abc"); int i = (Integer)arrayList.get(1);//编译要强制类型转换且运行时出错! Jdk 1.5的集合类希望你在定义集合时,明确表示你要向集合中装哪种类型的数据,无法加入指定类型以外的数据 ArrayList collection2 = new ArrayList(); collection2.add(1); /*collection2.add(1L); collection2.add(“abc”);*///这两行代码编译时就报告了语法错误 int i2 = collection2.get(0);//不需要再进行类型转换 泛型是提供给javac编译器使用的,可以限定集合中的输入类型,让编译器挡住源程序中的非法输入,编译器编译带类型说明的集合时会去除掉“类型”信息,使程序运行效率不受影响,对于参数化的泛型类型,getClass()方法的返回值和原始类型完全一样。由于编译生成的字节码会去掉泛型的类型信息,只要能跳过编译器,就可以往某个泛型集合中加入其它类型的数据,例如,用反射得到集合,再调用其add方法即可。
  • 23. 北京传智播客教育 www.itcast.cn了解泛型ArrayList类定义和ArrayList类引用中涉及如下术语: 整个称为ArrayList泛型类型 ArrayList中的E称为类型参数变量 整个ArrayList称为参数化的类型 ArrayList中的Integer称为类型参数变量的值或实际类型参数 ArrayList中的<>念着typeof ArrayList称为原始类型 参数化类型与原始类型的兼容性: 参数化类型可以引用一个原始类型的对象,编译报告警告,例如, Collection c = new Vector();//可不可以 原始类型可以引用一个参数化类型的对象,编译报告警告,例如, Collection c = new Vector();//原来的方法接受一个集合参数,新的类型也要能传进去 参数化类型不考虑类型参数的继承关系: Vector v = new Vector(); //错误!///不写没错,写了就是明知故犯 Vector v = new Vector(); //也错误! 编译器不允许创建类型变量的数组。即在创建数组实例时,数组的元素不能使用参数化的类型,例如,下面语句有错误: Vector vectorList[] = new Vector[10]; 思考题:下面的代码会报错误吗? Vector v1 = new Vector(); Vector v = v1;
  • 24. 北京传智播客教育 www.itcast.cn泛型中的?通配符问题: 定义一个方法,该方法用于打印出任意参数化类型的集合中的所有数据,该方法如何定义呢? 错误方式: public static void printCollection(Collection cols) { for(Object obj:cols) { System.out.println(obj); } /* cols.add("string");//没错 cols = new HashSet();//会报告错误!*/ } 正确方式: public static void printCollection(Collection cols) { for(Object obj:cols) { System.out.println(obj); } //cols.add("string");//错误,因为它不知自己未来匹配就一定是String cols.size();//没错,此方法与类型参数没有关系 cols = new HashSet(); } 总结: 使用?通配符可以引用其他各种参数化的类型,?通配符定义的变量主要用作引用,可以调用与参数化无关的方法,不能调用与参数化有关的方法。
  • 25. 北京传智播客教育 www.itcast.cn泛型中的?通配符的扩展限定通配符的上边界: 正确:Vector x = new Vector();//?号指定的类为Number的子类 错误:Vector x = new Vector(); 在应用泛型时可以使用extends限定符,并且可以用&来指定多个边界,如 void method(){} 限定通配符的下边界: 正确:Vector x = new Vector();//?号指定的类为Integer的父类 错误:Vector x = new Vector(); 提示: 限定通配符总是包括自己。 ?只能用作引用,不能用它去给其他变量赋值 Vector = new Vector(); Vector x = y; 上面的代码错误,原理与Vector x11 = new Vector();相似, 只能通过强制类型转换方式来赋值。
  • 26. 北京传智播客教育 www.itcast.cn泛型集合类的综合案例能写出下面的代码即代表掌握了Java的泛型集合类: HashMap hm = new HashMap(); hm.put("zxx",19); hm.put("lis",18); Set> mes= hm.entrySet(); for(Map.Entry me : mes) { System.out.println(me.getKey() + ":" + me.getValue()); } 对在jsp页面中也经常要对Set或Map集合进行迭代: ${entry.key}:${entry.value}
  • 27. 北京传智播客教育 www.itcast.cn通过反射获得泛型的参数化类型示例代码: Class GenericalReflection { private Vector dates = new Vector(); public void setDates(Vector dates) { this.dates = dates; } public static void main(String[] args) { Method methodApply = GenericalReflection.class.getDeclaredMethod("applyGeneric", Vector.class); ParameterizedType pType = (ParameterizedType) (methodApply .getGenericParameterTypes())[0]; System.out.println("setDates(" + ((Class) pType.getRawType()).getName() + "<" + ((Class) (pType.getActualTypeArguments()[0])).getName() + ">)" ); } } 泛型DAO的应用: public abstract class DaoBaseImpl implements DaoBase { protected Class clazz; public DaoBaseImpl() { Type type = this.getClass().getGenericSuperclass(); ParameterizedType pt = (ParameterizedType) type; this.clazz = (Class) pt.getActualTypeArguments()[0]; System.out.println("clazz = " + this.clazz); } } public class ArticleDaoImpl extends DaoBaseImpl
    implements ArticleDao { }
  • 28. 北京传智播客教育 www.itcast.cn类加载器与普通程序不同的是,Java程序(class文件)并不是本地的可执行程序。当运行Java程序时,首先运行JVM(Java虚拟机),然后再把Java class加载到JVM里头运行,负责加载Java class的这部分就叫做Class Loader。中文叫做类加载器。 Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个类负责加载特定位置的类:BootStrap,ExtClassLoader,AppClassLoader 类加载器也是Java类,因为其他是java类的类加载器本身也要被类加载器加载,显然必须有第一个类加载器不是不是java类,这正是BootStrap。 Java虚拟机中的所有类装载器采用具有父子关系的树形结构进行组织,在实例化每个类装载器对象时,需要为其指定一个父级类装载器对象或者默认采用系统类装载器为其父级类加载。
  • 29. 北京传智播客教育 www.itcast.cn 类装载器 一般类加载器的加载过程如下: 调用 findLoadedClass() 来查看是否存在已装入的类。 2>如果没有,那么获取class文件的原始字节。(通过IO从文件系统,来自网络的字节流等) 如果已有原始字节,调用 defineClass() 将它们转换成 Class 对象。 如果没有原始字节,然后调用 findSystemClass() 查看是否从本地文件系统获取类 如果 resolve 参数是 true,那么调用 resolveClass() 解析 Class 对象。 如果还没有类,返回 ClassNotFoundException。 否则,将类返回给调用程序。
  • 30. 北京传智播客教育 www.itcast.cn类加载器之间的父子关系和管辖范围图BootStrapExtClassLoaderAppClassLoaderItcastClassLoaderMyClassLoaderSystem classLoaderJRE/lib/rt.jarJRE/lib/ext/*.jarCLASSPATH指定的 所有jar或目录传智播客指定的 特殊目录
  • 31. 北京传智播客教育 www.itcast.cn类加载器的委托机制当Java虚拟机要加载一个类时,到底派出哪个类加载器去加载呢? 首先当前线程的类加载器去加载线程中的第一个类。 如果类A中引用了类B,Java虚拟机将使用加载类A的类装载器来加载类B。 还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类。 每个类加载器加载类时,又先委托给其上级类加载器。 当所有祖宗类加载器没有加载到类,回到发起者类加载器,还加载不了,则抛ClassNotFoundException,不是再去找发起者类加载器的儿子,因为没有getChild方法,即使有,那有多个儿子,找哪一个呢? 对着类加载器的层次结构图和委托加载原理,解释先前将ClassLoaderTest输出成jre/lib/ext目录下的itcast.jar包中后,运行结果为ExtClassLoader的原因。
  • 32. 北京传智播客教育 www.itcast.cn动态代理的工作原理图
  • 33. 北京传智播客教育 www.itcast.cn结束语 热烈欢迎各软件公司到传智播客公司预定和招聘软件开发人才! 谢谢!