黑马程序员_java高新技术(1)枚举、反射、内省
来源:互联网 发布:混乱之治mac 编辑:程序博客网 时间:2024/06/05 04:54
------- android培训、java培训、期待与您交流! ----------
JDK 5.0
静态导入,自动装箱拆箱,增强for循环,
可变参数,枚举,泛型,元数据
day01***Eclipse/枚举
*Workspace与project
会切换工作间与导入项目
*Perspective(透视图)与view
windows下有reset perspective重置透视图
*设置整个workspace的javac与java
*设置单个工程的javac与java
*快捷键的绑定与代码模块
*利用快捷键提高工作效率
在应用中具体举例讲解。
@@快捷键
Ctrl+1 快速改正
Alt+选中 看源码(没有源码则粘贴源码进去,文件夹-External Folder),(java)文件-External File)
Alt+/ 联想功能
Alt+左右 看源码时的前后顺序
Alt+上下 移动选中的代码
ctrl+Alt+上下 复制选中的代码
ctrl+T 查看类的继承关系
f2 f2进入提示框,查看方法的说明
ctrl+shift+X/大写 ctrl+shift+Y/小写
ctrl+shift+f 格式化format代码
ctrl+shift+o 导包
ctrl+shift+/ 加注释
ctrl+shift+\ 除注释
ctrl+shift+L 查看所有快捷键
ctrl+shift+T 直接看源码(自己输入)
Alt+shift+s 自动生成一些源代码source等
选中代码Alt+shift+M ==> extract Method 提取方法
@@断点
f5:step into跳入
f6:step over跳过
f7:step return 跳回
drop to fram :跳到当前方法的第一行
resume:跳到下一个断点(没有下一个则运行完整个程序)
watch:观察变量或表达式的值
断点注意问题:
1,断点调试完成后,要在breakpoints视图中清除所有断点
2,断点调试完成后,一定记得结束运行断点的JVM
@@eclipse的使用技巧
02 eclipce及ide开发工具的简介
myeclipse是eclipse的插件,使eclipse可以进行web项目开发
现在myeclipse整合了eclipse,包含了eclipse工具。
myeclipse是用java开发的
进程中的javaw.exe 开启的就是图形化的java控制台,而不像java.exe开启的是dos命令行。
专业术语(面试)
java ee
ide--> Integrated Development Environment 集成开发环境
jms
jmx
jndi
03 eclipse工程管理与快捷键配置。
window--preferences常用
一个相对独立的项目就是一个工程,一个工程中涉及的多个java文件,资源文件等用一个工程进行管理。
开发工具能对所有的源文件集中管理,记住每个源文件的位置和相互关系。
工程中有哪几个源文件,启动类是哪个,启动参数设置等配置信息在工程中都记录。
一个workspace可以包含多个project。如果要为eclipse再配置一套环境选项,可以再创建一个work是。
资源文件就是配置文件。
file--switch workspace 切换工作间
一个工作间有自己的特色环境配置,其下的工程与工作间的环境配置一样。
java工程的包名 com.机构名.内容 name:类名(大写)
快捷键的使用技巧:General-keys,设置 Alt+/ 进行内容提示,要注意解除Alt+/ 原来的绑定关系。直接输入alt+/
就可以找到它的绑定关系,删除用 remove binding 。
04 透视图 与程序调试
透视图:window --Show View
透视图其实就是界面的布局。不同的透视图,包含不同的视图(view),每个视图的位置、大小也不同。
断点
f5 :step into
f6: step over
f7: step return
调试:双击设置断点,右键 debug as ...打开debug视图 选中参数,右键watch
drop to frame: 跳到当前方法的第一行
resume: 跳到下一个断点。(如果没有下一个,则运行完整个程序)
watch:观察变量或表达式的值
注意:
调试完后,要在breakpoints视图中清除所有断点。
调试完后,一定记得结束运行断点的jvm。
05 编译与运行环境
java--compiler 编译时的环境
java--Installed JREs 运行时的环境
单独工程 右键--Properties java Compiler编译环境
--Run as --其它--JRE 查看单独一个工程的运行环境
高版本的java能否运行低版本的javac编译的程序?能:1.6jre的java,能运行1.5的javac编译的程序。
低版本的java能否运行高版本的javac编译的程序?不能:1.5jre的java,不能运行1.6编译的程序。
06 java模板代码 window--pre--java--Editor Templates
代码模块的设置java-editor-Templates,new,
name:tryf
pattern:try{
line_selection选中的内容
}finally{
cursor光标
}
使用模板 选中代码 右键 Surround With 循环/try异常/同步...
07 导入已有的工程:
先将工程文件拷贝到工作间下
然后file--import--general--Existig Projects Workspace--目录
jre的库设置
包和库的设置 选中工程--右键--Build Path--其它--Libraries --Add Library(多个jar包)--user Library--自己命名库,增加多个jar包
08 @@静态导入(JDK1.5后)与编译器语法设置
import 导入一个类或某个包中所有的类
import static 导入一个类中的某个静态方法或所有静态方法。
//import static java.lang.Math.max;
import static java.lang.Math.*;
System.out.println(max(3, 6));可以省略类名
System.out.println(abs(3 - 6));
09 @@可变参数与Overload面试题
可变参数的特点:
可变参数你就把它看成数组
只能出现的参数列表的最后面;
...位于变量类型和变量名之间,前后有无空格都可以
调用可变参数的方法,编译器为该可变参数隐含创建一个数组,在方法体中以数组的形式方法可变参数。
可以以数组的形式来调用可变参数。
public class VarableParameter {
public static void main(String[] args){
System.out.println(add(2,3,4));
System.out.println(add(1,2,3,4,5));
}
public static int add(int x,int...args){
int sum = x;
for(int i =0;i<args.length;i++){
sum += args[i];
}
return sum;
}
}
可变参数注意问题:
int nums[] = {1,2,3,5};
//把数组当成一个对象放进了list中了
list = Arrays.asList(nums);
System.out.println(list);//打印[[I@7ced01]地址值
Integer num[] = {1,2,3,5};
//把数组中的每个元素都当成一个对象放进了list中了
list = Arrays.asList(num);
System.out.println(list);//打印[1, 2, 3, 5]
overload重载 Vs override 覆盖
override可以翻译为覆盖,从字面就可以知道,它是覆盖了一个方法并且对其重写,以求达到不同的作用。对我们来说最熟悉的覆盖就是对接口方法的实现,在接口中一般只是对方法进行了声明,而我们在实现时,就需要实现接口声明的所有方法。除了这个典型的用法以外,我们在继承中也可能会在子类覆盖父类中的方法。
在覆盖要注意以下的几点:
1、覆盖的方法的标志必须要和被覆盖的方法的标志完全匹配,才能达到覆盖的效果;
2、覆盖的方法的返回值必须和被覆盖的方法的返回一致;
3、覆盖的方法所抛出的异常必须和被覆盖方法的所抛出的异常一致,或者是其子类;
4、被覆盖的方法不能为private,否则在其子类中只是新定义了一个方法,并没有对其进行覆盖。
overload对我们来说可能比较熟悉,可以翻译为重载,它是指我们可以定义一些名称相同的方法,通过定义不同的输入参数来区分这些方法,然后再调用时,VM就会根据不同的参数样式,来选择合适的方法执行。
在使用重载要注意以下的几点:
1、在使用重载时只能通过不同的参数样式。例如,不同的参数类型,不同的参数个数,不同的参数顺序;
2、不能通过访问权限、返回类型、抛出的异常进行重载;
3、方法的异常类型和数目不会对重载造成影响;
4、对于继承来说,如果某一方法在父类中是访问权限是priavte,那么就不能在子类对其进行重载,如果定义的话,也只是定义了一个新方法,而不会达到重载的效果。
10 @@增强for循环
增强for循环注意问题:增强for只适合取数据,不能改变数据。
for(type 变量名 : 集合变量名){...}
注意:迭代变量必须在()中定义
集合变量可以是数组或实现了Iterable接口的集合类。
11 @@自动装箱与拆箱(JDK1.5后)
Integer iObj = 3;//自动装箱
System.out.println(iObj + 12);//自动拆箱
Integer i1 = 13;
Integer i2 = 13;
//在装箱成Integer时,如果在-128~127;范围内,就会缓存以来。
System.out.println(i1==i2);//ture
Integer i3 = 134;
Integer i4 = 134;
//在装箱成Integer时,如果超过-128~127;范围,就不会缓存以来。
System.out.println(i3==i4);//false
这是一种享元模式。flyweight。(word中的字符,OS中的图标。。。等)
享元模式下:
word文档中的每个字符
把26个字符提前实例化变成对象,word中的每个字符只是调用某个字符对象的方法,参数是坐标。
通过方法 i(int x,int y)//调用字符 i 的方法,参数是字符想要显示的坐标
非享元模式下:
每次写下一个字符就是一次实例化,对每个字符进行实例化
i(char ch,int x,int y)//大量的创建实例,浪费内存资源。
Integer i5 = Integer.valueOf(127);
Integer i6 = Integer.valueOf(127);
System.out.println(i5==i6);//true
Integer i7 = Integer.valueOf(128);
Integer i8 = Integer.valueOf(128);
System.out.println(i7==i8);//false
12@@枚举
为什么要有枚举
问题:要定义星期几或性别的变量,该怎么定义?假设1-7分别表示星期一到星期七,但有人会写 0;
枚举就是要让某个类型的变量的取值只能为若干个固定值中的一个,否则,编译器就会报错。
枚举可以让编译器在编译时就可以控制程序中的非法值,普通变量的方式在开发阶段无法实现这一功能。
用普通类如何实现枚举功能,定义一个Weekday的类来模拟枚举功能
*私有的构造方法
*每个元素分别用一个公有的静态成员变量表示
*可以有若干公有方法或抽象方法,要提供nextDay()方法必须是抽象的。
枚举的基本应用
*举例:定义一个WeekDay的枚举。
*扩展:枚举类的values,valueOf,getName,toString,ordinal等方法
*总结:枚举是一个特殊的类,其中的每个元素都是该类的一个实例化对象。
例如可以调用WeekDay.SUN.getClass().getName和WeekDay.class.getName()
枚举也可以实现接口、或者继承抽象类。
JDK5.0中扩展了switch语句,它除了接受int,byte,char,short,还可以接受一个枚举类型。
//测试枚举的常用方法
public void test2(){
System.out.println(Grand.C.name());//枚举常量的名字 C
System.out.println(Grand.C.ordinal());//枚举常量的序数 2
String str = "B";
//Grand g = Grand.valueOf(Grand.class, str);
Grand g = Grand.valueOf(str); //开发中常用!
System.out.println(g); //B
Grand gs[] = Grand.values(); //返回枚举类Grand中所有的枚举常量
for(Grand g1 : gs)
System.out.println(g1); //A B C D E
}
@@枚举入门
public static void main(String[] args) {
WeekDay weekDay2 = WeekDay.FRI;//该句话编译时,会调用WeekDay的七个构造函数
System.out.println(weekDay2);FRI
System.out.println(weekDay2.name());//FRI
System.out.println(weekDay2.ordinal());//5 位置
System.out.println(weekDay2.toString());//FRI
System.out.println(weekDay2.getClass());//class cn.itcast.day1.EnumTest$WeekDay
System.out.println(WeekDay.valueOf("SUN").toString());//SUN
System.out.println(WeekDay.values().length);//7
}
//格式
(1)最简单
public enum WeekDay{
SUN,MON,TUE,WED,THU,FRI,SAT
}
(2)带构造方法的枚举,构造方法必须私有
public enum WeekDay{//相当于内部类
SUN(1),MON(),TUE,WED,THU,FRI,SAT;//必须放第一行
private WeekDay(){System.out.println("first");}
private WeekDay(int day){System.out.println("second");}
}
(3)带抽象方法的枚举
public enum TrafficLamp{
RED {//生成一个类文件EnumTest$TrafficLamp$1.class
@Override
public TrafficLamp nextLamp() {
return GREEN;
}
},GREEN {//生成一个类文件EnumTest$TrafficLamp$2.class
@Override
public TrafficLamp nextLamp() {
return YELLOW;
}
},YELLOW {//生成一个类文件EnumTest$TrafficLamp$3.class
@Override
public TrafficLamp nextLamp() {
return RED;
}
};
public abstract TrafficLamp nextLamp();
}
(4)带构造方法和抽象方法的枚举
public enum TrafficLamp{
RED(30) {//生成一个类文件EnumTest$TrafficLamp$1.class
@Override
public TrafficLamp nextLamp() {
return GREEN;
}
},GREEN (45){//生成一个类文件EnumTest$TrafficLamp$2.class
@Override
public TrafficLamp nextLamp() {
return YELLOW;
}
},YELLOW (5){//生成一个类文件EnumTest$TrafficLamp$3.class
@Override
public TrafficLamp nextLamp() {
return RED;
}
};
public abstract TrafficLamp nextLamp();
private int time;
private TrafficLamp(int time){this.time = time;}
}
(5)带构造方法和普通方法的枚举
public enum TrafficLamp{
RED(30) {//生成一个类文件EnumTest$TrafficLamp$2.class
@Override
public TrafficLamp nextLamp() {
return GREEN;
}
},GREEN (45){//生成一个类文件EnumTest$TrafficLamp$2.class
@Override
public TrafficLamp nextLamp() {
return YELLOW;
}
},YELLOW (5);//不生成类文件
public TrafficLamp nextLamp(){
return null;
}
private int time;
private TrafficLamp(int time){this.time = time;}
}
*枚举就相当于一个类,其中也可以定义构造方法、成员变量、普通方法、抽象方法
*枚举元素必须位于枚举体中的最开始部分,枚举元素列表的最后面有分号与其他成员隔开。
把枚举中的成员方法或变量等放在枚举前面,会编译失败。
*带构造方法的枚举
构造方法必须私有
可以有多个构造方法
枚举元素MON和MON() 效果一样,都是调用默认的构造方法。
*带方法的枚举
实现普通的方法
实现抽象的方法:每个元素分别是有枚举类的子类来生成的实例对象,这些子类采用类似内部类的方式进行定义。
*枚举只有一个成员时,就可以作为一种单例的实现方式。
***反射/JavaBean
17 反射
人->Person
Java类->Class
@@Class类
*Class类代表Java类
对应各个类在内存中的字节码,例如,Person类的字节码,ArrayList类的字节码,等
一个类被类加载器加载到内存中,占用一片存储空间。这个空间里面的内容可分别用一个个的对象来表示。
这些类具有相同的类型,就是Class类。
*@@实例化如何得到各个字节码对应的实例对象(Class对象)
1,类名.class 例如System.class
2,对象.getClass 例如 new Date().getClass()
3,Class.forName("类名"), 例如 Class.forName("java.util.Date")
Person p1 = new Person();
1 Class cls1 = Date.class
Class cls2 = Person.class
2 p1.getClass();
3 Class.forName("java.lang.String");
*@@Class 类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。
每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。
基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。
int.class.isPrimitive();//true 判定指定的 Class 对象是否表示一个基本类型。
Integer.class.isPrimitive();false
int.class == Integer.TYPE;//true
int.class == Integer.class;//false
void.class.isPrimitive();//true
Void.class.isPrimitive();//false
void.class == Void.TYPE;//true
void.class == Void.class;//false。Void 类是一个不可实例化的占位符类,它持有对表示 Java 关键字 void 的 Class 对象的引用。
Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的。
@@方法摘要
<U> Class<? extends U>
asSubclass(Class<U> clazz)
强制转换该 Class 对象,以表示指定的 class 对象所表示的类的一个子类。
static Class<?> forName(String className)
返回与带有给定字符串名的类或接口相关联的 Class 对象。
static Class<?> forName(String name, boolean initialize, ClassLoader loader)
使用给定的类加载器,返回与带有给定字符串名的类或接口相关联的 Class 对象。
String getName()
以 String 的形式返回此 Class 对象所表示的实体(类、接口、数组类、基本类型或 void)名称。
boolean isPrimitive() 共有9种为true,8种基本类型和 void
判定指定的 Class 对象是否表示一个基本类型。
有九种预定义的 Class 对象,表示八个基本类型和 void。这些类对象由 Java 虚拟机创建,与其表示的基本类型同名,
即 boolean、byte、char、short、int、long、float 和 double。
这些对象仅能通过下列声明为 public static final 的变量访问,也是使此方法返回 true 的仅有的几个 Class 对象。
例 public static void main(String[] args) throws ClassNotFoundException {
String str1 = "abc";
Class cls1 = str1.getClass();
Class cls2 = String.class;
Class cls3 = Class.forName("java.lang.String");
System.out.println(cls1 == cls2);//true
System.out.println(cls1 == cls3);//true
System.out.println(cls1.isPrimitive());//false
System.out.println(int.class.isPrimitive());//true
System.out.println(Integer.class.isPrimitive());false
System.out.println(int.class == Integer.class);//false
System.out.println(int.class == Integer.TYPE);//true
System.out.println(int[].class.isPrimitive());//false
System.out.println(int[].class.isArray());//true 判定此 Class 对象是否表示一个数组类。
System.out.println(void.class.isPrimitive());//true
System.out.println(void.class == Void.TYPE);//true
System.out.println(void.class == Void.class);//false,Void 类是一个不可实例化的占位符类,它持有对表示 Java 关键字 void 的 Class 对象的引用。
}
总之,只要在源程序中出现的类型,都有各自的Class实例对象。例如int[],void
*反射就是把Java类中的各种成分映射成相应的java类。
表示java类的Class类显然要提供一系列的方法,来获得其中的变量、方法、构造方法、
修饰符、包等信息。这些信息就是用相应类的实例对象来表示,它们是Field/Methed/Constructor/Package等
*一个类中的每个成员都可以用相应的反射API类的一个实例对象来表示,通过调用Class类的方法可以得到这些实例对象。
@@Constructor类 构造方法对应的类
得到某个类所有的构造方法
Constructor[] constructors = Class.forName("java.lang.String").getConstructors();
得到具体某一个构造方法//得到的是参数为StringBuffer的构造方法。
Constructor consturctor = Class.forName("java.lang.String").getConstructor(StringBuffer.class);
创建带参数的实例对象
通常 String str = new String(new StringBuffer("abc"));
反射 String str = (String)constructor.newInstance(new StringBuffer("abc"));
创建不带参数的实例对象,直接用java类.newInstance();
StringBuilder str3 = (StringBuilder)StringBuilder.class.newInstance();
System.out.println(str3.append("abc"));//abc
@@Field类 成员变量对应的类
Object get(Object obj)
返回指定对象上此 Field 表示的字段的值
Class<?> getType()
返回一个 Class 对象,它标识了此 Field 对象所表示字段的声明类型。
ReflectPoint pt1 = new ReflectPoint(3,5);
Field fieldY = pt1.getClass().getField("y");//得到public成员对应的类对象
System.out.println(fieldY.get(pt1));
Field fieldX = pt1.getClass().getDeclaredField("x");//得到private成员对应的类对象
fieldX.setAccessible(true);//暴力反射,强制访问类的私有成员。
System.out.println(fieldX.get(pt1));
作业:将任何一个对象的所有String类型的成员变量所对应的字符串内容中的“b”改为“a”
private static void changeStringValue(Object obj) throws IllegalArgumentException, IllegalAccessException {
// TODO Auto-generated method stub
Field[] fields = obj.getClass().getFields();
for(Field field : fields){
if(field.getType() == String.class){
String oldValue = (String)field.get(obj);
String newValue = oldValue.replaceAll("b","a");
field.set(obj, newValue);
}
}
}
@@Method类 成员方法对应的类
Method类代表某个类中的一个成员方法
*得到类中的一个方法
Method charAt = Class.forName("java.lang.String").getMethod("charAt",int.class);
*调用方法 String str = new String("abc");
通常:str.charAt(1);
反射:charAt.invoke(str,1);//b
CharAt.invoke(str,new Object[]{1});//jdk1.4时的方法,把参数对象放进Object数组中
如果传递给Method对象的invoke方法的第一个参数为null,这有着什么样的意义呢?
说明该Method对象对应是一个静态方法!
@@用反射执行main方法
目标:写一个程序,这个程序能够根据用户提供的类名,去执行该类中的main方法。
jdk1.5 整个数组是一个参数,而jdk1.4数组中的每个元素对应一个参数。
1.5兼容1.4,所以不能用mainMethod.invoke(null,new String[]{xxx}),会被当1.4的方法处理。
*解决办法
mainMethod.invoke(null,new Object[]{new String[]{xxx}});
mainMethod.invoke(null,(Object)new String[]{xxx});编译器会做特殊处理!认为是main方法。
//main方法的调用
//通常方式 TestArguments.main(new String[]{"1","2","3"});
//反射方式
String startClass = args[0];//将 最开始执行类 传进来
Method mainMethod = Class.forName(startClass).getMethod("main",String[].class);
mainMethod.invoke(null, new Object[]{new String[]{"jdk","1.5"}});
mainMethod.invoke(null,(Object)new String[]{"jdk","1.5"});
class TestArguments{
public static void main(String[] args){
for(String arg : args){
System.out.println(arg);
}
}
@@数组的反射1
数组所属的java类是同一个字节码,前提 数组的维度和数组元素的类型都相同!
int [] a1 = new int[]{1,2,3};
int [] a2 = new int[4];
int[][] a3 = new int[2][3];
String [] a4 = new String[]{"a","b","c"};
System.out.println(a1.getClass() == a2.getClass());//true
//System.out.println(a1.getClass() == a4.getClass());//编译失败
System.out.println(a1.getClass().getName());//[I 表示该java类是 一维数组,元素类型是int
System.out.println(a1.getClass().getSuperclass().getName());//java.lang.Object
Object aObj1 = a1;//int[] 是Object类型 Object中装的是一个 int型一维数组
Object aOvj2 = a4;//String[] 是Object类型 Object中装的是一个String型一维数组
//Object[] aObj3 = a1;//编译失败,int基本类型不是Object类型 Object中不能装一个 基本类型的值。
Object[] aObj4 = a3;//int[] 类型是Object类型 每个Object中装的是 一个int型的一维数组
Object[] aObj5 = a4;//String 类型是Object类型 每个Object中装的是一个 String类型的值
//Arrays中的方法原型 public static <T> List<T> asList(T... a)
System.out.println(Arrays.asList(a1));
//[[I@ca0b6] 表示List<int[]> list = Arrays.asList(int[] al) 转换成的list集合中只有一个元素 即 al (一个一维的int类型数组),
、 说明int(基本数据类型)不能传入泛型中! 基本数据类型数据不能传进Object类型中;
System.out.println(Arrays.asList(a4));//[a, b, c] List<String> list = Arrays.asList(a4)集合中有3个元素,
、 String类型可以传入泛型中。String类型可以装进Object类型中!
泛型在1.5以前用Object表示
1.5后用<T>表示
泛型中不能传入基本数据类型的数据!
System.out.println(a1);//[I@ca0b6 相当于a1.getClass().getName()+"@"+Integer.toHexString(a1.hashCode())
System.out.println(a4);//[Ljava.lang.String;@10b30a7
Class类的getName()方法
Element Type Encoding
boolean Z (特殊)
byte B
char C
class or interface Lclassname; (特殊)
double D
float F
int I
long J (特殊)
short S
类或接口名 classname 是上面指定类的二进制名称。
示例:
String.class.getName()
returns "java.lang.String"
byte.class.getName()
returns "byte"
(new Object[3]).getClass().getName()
returns "[Ljava.lang.Object;"
(new int[3][4][5][6][7][8][9]).getClass().getName()
returns "[[[[[[[I"
@@数组的反射2
java.lang.reflect 下的Array全是静态方法
private static void printObject(Object obj){
Class clazz = obj.getClass();
if(clazz.isArray()){ //**重要方法
int len = Array.getLength(obj); //**重要方法
for(int i = 0;i<len;i++){
System.out.println(Array.get(obj,i));//**重要方法
}
}else{
System.out.println(obj);
}
}
printObject(a4);//a b c
printObject("xyz");//xyz
printObject('a');//a
printObject(123);//123
printObject(true);//true
怎么得到一个数组中的元素类型?
int[] a = new int[]{1,2,3};
//System.out.println(a[0].getClass().getName());//编译失败
//System.out.println(a[1].getClass().getName());//编译失败
System.out.println(a.getClass().getName());// [I
Object[] a = new Object[]{"a",1};
System.out.println(a[0].getClass().getName()); // java.lang.String
System.out.println(a[1].getClass().getName());// java.lang.Integer
System.out.println(a.getClass().getName());// [Ljava.lang.Object;
没法得到一个数组的元素类型,只能得到其中某个元素的类型。
@@hashCode的用处
ArrayList和HashSet的比较 及 hashCode的分析。
内存泄露:对象没有被释放,一直占用内存资源。
在元素类中覆写hashCode和equals方法后
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + x;
result = prime * result + y;
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
ReflectPoint other = (ReflectPoint) obj;
if (x != other.x)
return false;
if (y != other.y)
return false;
return true;
}
Collection collections = new HashSet();
ReflectPoint pt1 = new ReflectPoint(3,3);
ReflectPoint pt2 = new ReflectPoint(5,5);
ReflectPoint pt3 = new ReflectPoint(3,3);
collections.add(pt1);
collections.add(pt2);
collections.add(pt3);
collections.add(pt1);
//如果改变了pt1的y值,那么就无法成功删除pt1了,会造成内存泄露.
pt1.y = 7;
collections.remove(pt1);
System.out.println(collections.size());
@@反射的作用==实现框架的功能
*我做房子卖给用户,有用户自己安装门窗和空调,我做的房子就是框架,
用户需要使用我的框架,把门窗插入进我提供的框架中。
框架与工具类的区别,工具类被用户调用,而框架调用用户提供的类!
*框架要解决的核心问题
我在写框架(房子)的时候,你这个用户还在上小学,还不会写程序,我写的框架程序怎么调用
你以后写到的类呢?
因为在写程序时无法知道要被调用的类名,所以,在程序中无法直接new某个类的实例对象,而要用反射的方式做。
你做的门调用锁,锁是工具,你做的门被房子调用,房子是框架,房子和锁都是被人提供。
//@@不用类加载器的方式
config.properties(工程的目录下)文件中的内容 className=java.util.ArrayList
InputStream ips = new FileInputStream("config.properties");
Properties props = new Properties();
props.load(ips);
ips.close();
String className = props.getProperty("className");
Collection collections = (Collection)Class.forName(className).newInstance();
//以上所有等于Collection collections = new ArrayList();
@@用类加载器的方式管理资源和配置文件
getRealPath() 安装目录
配置文件的路径设置==》 安装目录/配置文件内部路径
一定要记住完整的路径,但完整的路径不是硬编码,而是运算出来的。
InputStream ips = ReflectTest2.class.getClassLoader().getResourceAsStream("cn/itcast/day1/config.properties");
//或者 InputStream ips = ReflectTest2.class.getResourceAsStream("config.properties");//省略中间加载器,默认ReflectTest2的class文件的路径。效果如上!
Properties props = new Properties();
props.load(ips);
ips.close();
String className = props.getProperty("className");
Collection collections = (Collection)Class.forName(className).newInstance();
@@加载配置文件小结
//1==不用类加载器(配置资源在javaenhance_my工程的下面)
//InputStream ips = new FileInputStream("config.properties");
//2==用类加载器的方式加载配置文件(不省略中间获取加载器的步骤,所以资源路径名要写全)(配置资源在javaenhance_my工程中的cn/itcast/day1下面)
//InputStream ips = ReflectTest2.class.getClassLoader().getResourceAsStream("cn/itcast/day1/config.properties");
//3==省略中间获取加载器的步骤,则资源路径 默认为ReflectTest2的class文件的路径。效果如上!
InputStream ips = ReflectTest2.class.getResourceAsStream("config.properties");
//4==将配置文件放在新建的包中,在cn/itcast/day1 文件中建立一个资源文件包 cn.itcast.day1.resource
InputStream ips = ReflectTest2.class.getResourceAsStream("resource/config.properties");
//5==用绝对路径 String name=“/...”;前面有个/表示 :配置文件的路径与前面一个类(Re..2)的class文件路径不相关,所以要写全配置文件的路径名。
//InputStream ips = ReflectTest2.class.getResourceAsStream("/cn/itcast/day1/resource/config.properties");
@@内省Introspector
为什么学习内省?开发框架时,经常需要使用java对象的属性来封装程序的数据,
每次都使用反射技术完成此操作过于麻烦,所以sun开发了一套API,专门用于操作java对象的属性。
内省访问JavaBean属性的两种方式:
1,通过PropertyDescriptor类操作Bean的属性
2,通过Introspector类获得Bean对象的BeanInfo,
然后通过BeanInfo来获取属性的描述器(PropertyDescriptor),
通过这个属性描述器就可以获取某个属性对应的getter/setter方法,然后通过反射来调用这些方法。
@@由内省引出的JavaBean JavaBean就是特殊一点的java类
*JavaBean是一种特殊的Java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有的字段,且方法名符合某种命名规则。
JavaBean的属性 是根据方法得来的。属性要私有private,get和set方法要公有public!
比如一个方法是 setAge()-->属性 age
*如果要在2个模块之间传递多个信息,可以将这些信息封装到一个JavaBean中,这种JavaBean(也叫java类)的实例对象通常称之为值对象(Value Object)。
如果去掉set get后 剩下的字符串第二个字母是小写,那么属性名全小写;
剩下的字符串第二个字母是大写,那么属性名全大写;
@@一个JavaBean中的属性的定义
public class Person { //一个JavaBean类
private String name ; //字段
private String password ; //字段
private int age ; //字段
/*
* 一个JavaBean的属性有哪些,由(并且只由)它的get或set方法(有一个方法就行了)决定的。
* 如果只有上面三个私有成员变量,它们只能叫做字段,
* 但是如果有了下面的公有的get或set方法,它们就可以叫做属性了。
*
* 所以判断一个JavaBean有哪些属性,应该通过它有哪些get或set方法来判断。
* 另外:任何类都继承了Object类,而Object类中有一个getName()方法,所有会有一个name属性。
*
*本类共五个属性:ab age class name password (系统默认顺序)
* */
//此函数使得Person类多了一个属性,叫做ab
public String getAb() {
return null;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
@@使用内省API操作bean的属性
public class Demo1 {
//得到bean的所有属性:可以含继承来的,或不含继承来的
@Test
public void test1() throws Exception{
Person p = new Person();
BeanInfo beanInfo_all = Introspector.getBeanInfo(Person.class);
//对Person进行内省,也内省了它继承的类
PropertyDescriptor[] pds_all =beanInfo_all.getPropertyDescriptors();
System.out.println(pds_all.length);// 5
for(PropertyDescriptor pd : pds_all){
System.out.println(pd.getName());
//Person这个JavaBean类的所有属性值,包括继承下来的
//:ab age class name password
}
BeanInfo beanInfo_self = Introspector.getBeanInfo(Person.class,Object.class);
//在断点Object.class之下,对Person.class进行内省,得到的是Person类自己的属性!
PropertyDescriptor[] pds_self =beanInfo_self.getPropertyDescriptors();
System.out.println(pds_self.length);// 4
for(PropertyDescriptor pd : pds_self){
System.out.println(pd.getName());
//Person这个JavaBean类的所有属性值,不包括继承下来的
//:ab age name password
}
}
//只想得到bean的特定属性: age
@Test
public void test2() throws Exception{
Person p = new Person();
PropertyDescriptor pd = new PropertyDescriptor("age",Person.class);
//得到属性的写方法:为属性赋值
Method method_set = pd.getWriteMethod();//public void setAge(int age)
method_set.invoke(p, 45);
//获取属性的值:
Method method_get = pd.getReadMethod();//public int getAge()
System.out.println(method_get.invoke(p, null));
}
//高级点的内容:获取当前操作的属性的类型
@Test
public void test3() throws Exception{
Person p = new Person();
PropertyDescriptor pd = new PropertyDescriptor("age",Person.class);
//得到属性的类型
System.out.println(pd.getPropertyType());// int
}
}
@@BeanUtils工具包
用BeanUtils工具包方便些!
好处:自动进行类型转换,BeanUtils以字符串的形式对JavaBean进行操作!
BeanUtils.getProperty(pt1,"x");
BeanUtils.setProperty(pt1,"x","9");//自动进行类型转换,将String "9"换成int 9
好处:PropertyUtils以属性本身的类型进行操作!
PropertyUtils.setProperty(pt1, "x", 9);//不进行类型转换。
System.out.println(PropertyUtils.getProperty(pt1, "x"));
好处:支持操作成员变量的内部数据。操作属性链的功能!
BeanUtils.setProperty(pt1,"birthday.time", "111");
System.out.println(BeanUtils.getProperty(pt1,"birthday.time"));
好处:拷贝属性,将JavaBean转换成Map对象,
/*//java7的新特性
Map map = (name:"wzq",age:18);
BeanUtils.setProperty(map, "name", "cdd");*/
Person bean = new Person();
BeanUtils.populate(bean, map);//将Map填充进JavaBean
------- android培训、java培训、期待与您交流! ----------
- 黑马程序员_java高新技术(1)枚举、反射、内省
- java高新技术_枚举反射内省(黑马程序员)
- 黑马程序员—java高新技术_枚举&反射&内省
- 黑马程序员--高新技术--枚举、反射、注解、类加载器、内省
- 黑马程序员_java高新技术_枚举与反射
- 黑马程序员_java高新技术 枚举
- 黑马程序员_java高新技术--枚举
- 黑马程序员_Java高新技术之 内省 JavaBean
- 黑马程序员_Java高新技术_内省&JavaBean
- 黑马程序员_Java高新技术:反射
- 黑马程序员 Java高新技术 枚举,内省
- 黑马程序员 高新技术<二> 反射和内省
- 黑马程序员 Java高新技术--反射和内省
- 黑马程序员_Java高新技术——反射和内省(第8篇)
- 黑马程序员_java的反射与内省
- 黑马程序员_java高新技术(2)枚举
- 黑马程序员_Java高新技术之枚举
- 黑马程序员_Java高新技术_枚举
- 汉诺塔算法
- android phonelistener 使用MediaRecorder监听电话电话.并上传服务器
- 删除特殊字符和限定用户输入长度
- 自定义Dilalog
- mysql是如何启动和关闭的
- 黑马程序员_java高新技术(1)枚举、反射、内省
- VC++(MFC)实现SQL Server的远程连接(ADO实现)
- 什么是HTML和网页文件
- 类加载器
- Linux下使用Shell脚本改变当前工作路径
- Ubuntu 系统托盘输入法图标消失的解决办法
- MyEclipse默认标签TODO,XXX,FIXME和自定义标签的使用
- 瓦片地图坐标与触摸坐标的转换
- ucGUI鼠标拖动实现(DockDrop实现)