Java基础加强-1
***************************************************
一、 枚举
1、 为什么要有枚举
问题:要定义星期几或性别的变量,该怎么定义?假设用1-7分别表示星期一到星期日,但有人可能会写成intweekday = 0;或即使使用常量方式也无法阻止意外。
枚举就是要让某个类型的变量的取值只能为若干个固定值中的一个,否则,编译器就会报错。枚举可以让编译器在编译时就可以控制源程序中填写的非法值,普通变量的方式在开发阶段无法实现这一目标。
2、 枚举的基本应用
举例:定义一个Weekday的枚举。
扩展:枚举类的values,valueOf,name,toString,ordinal等方法(记住,讲课时要先于自定义方法前介绍,讲课更流畅)
总结:枚举是一种特殊的类,其中的每个元素都是该类的一个实例对象,例如可以调用WeekDay.SUN.getClass().getName和WeekDay.class.getName()。
枚举类的实例对象个数是有限的,就是那些成员,可以在枚举类的构造方法中加入监控语句,看到这几个实例对象被创建出来的过程。
如果想在一个类中编写完各个枚举类和测试调用类,那么可以将枚举类定义成调用类的内部类。
3、 枚举就相当于一个类,其中也可以定义构造方法、成员变量、普通方法和抽象方法。枚举元素必须位于枚举体中的最开始部分,枚举元素列表的后要有分号与其他成员分隔。把枚举中的成员方法或变量等放在枚举元素的前面,编译器报告错误。
带构造方法的枚举
构造方法必须定义成私有的
如果有多个构造方法,该如何选择哪个构造方法?
枚举元素MON和MON()的效果一样,都是调用默认的构造方法。
带方法的枚举
定义枚举TrafficLamp
实现普通的next方法
实现抽象的next方法:每个元素分别是由枚举类的子类来生成的实例对象,这些子类采用类似内部类的方式进行定义。增加上表示时间的构造方法
枚举只有一个成员时,就可以作为一种单例的实现方式。
----------------------------------------------------------------------------------------------------------------------
讲构造方法枚举时,最好是定义一个带参数的构造方法和一个不带参数的构造方法,问大家默认调用的是哪个构造方法,然后提问想选择带参数的构造方法,该如何做呢?
如果枚举元素后面没有大括号对,那是不会生成内部类的。先看有内部类的效果,再看没有内部类的效果,eclipse自动删除原来的内部类。
写带有抽象方法的枚举步骤和技巧:
1. enumTrafficLamp{RED,GREEN,YELLOW}
2. enumTrafficLamp{RED,GREEN,YELLOW;public abstract next();}
3.enumTrafficLamp{RED{},GREEN{},YELLOW{};public abstract next();}
4.填充各个{}中的next方法。
----------------------------------------------------------------------------------------------------------------------
模拟交通灯的枚举示例代码:
package cn.itcast.day1;
public class EnumTest {
public static voidmain(String[] args){
new Thread(newRunnable(){
public void run(){
display(TrafficLamp.RED);
}
}).start();
new Thread(newRunnable(){
public void run(){
display(TrafficLamp.GREEN);
}
}).start();
//表示交通灯颜色的枚举类,包含构造方法和抽象方法
public enum TrafficLamp{
//当抽象方法存在,必须覆盖枚举类中的抽象方法
RED(25){
public TrafficLampnextLamp(){
return GREEN;
}
},GREEN(20){
public TrafficLampnextLamp(){
return YELLOW;
}
},YELLOW(5){
public TrafficLampnextLamp(){
return RED;
}
};
public abstractTrafficLamp nextLamp();
private int time;
private TrafficLamp(inttime){
this.time = time;
}
public int getTime(){
return time;
}
public void setTime(inttime){
this.time = time;
}
}
//交通灯的运行方法,在线程run中调用
public static voiddisplay(TrafficLamp state){
while (true) {
switch (state) {
case GREEN:
if(state.getTime() != 0) {
state.setTime(lampSleep(state, state.getTime()));
} else {
state =state.nextLamp();
}
break;
case RED:
if(state.getTime() != 0) {
state.setTime(lampSleep(state, state.getTime()));
} else {
state = state.nextLamp();
}
break;
case YELLOW:
if(state.getTime() != 0) {
state.setTime(lampSleep(state, state.getTime()));
} else {
state = state.nextLamp();
}
break;
}
}
}
public static intlampSleep(TrafficLamp state,int count){
try {
Thread.sleep(1000);
} catch(InterruptedException e) {
// TODOAuto-generated catch block
e.printStackTrace();
}
System.out.println(state.toString()+count);
count--;
return count;
}
}
----------------------------------------------------------------------------------------------------------------------
***************************************************
二、反射
importjava.lang.reflect.*;
importjava.util.Arrays;
public class ReflectTest {
public static voidmain(String[] args) throws Exception{
// TODO Auto-generatedmethod stub
String str = "hello";
//获取String类的字节码,三种方法:
Class cls1=String.class;
Class cls2=str.getClass();
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(int[].class.isPrimitive());//false
System.out.println(int[].class.isArray());//true
System.out.println(int.class== Integer.class);//flase
System.out.println(int.class == Integer.TYPE);//true
//-----------------------------------------------------
//获取构造器并构造对象
//获取对应类的构造方法,根据参数的不同来指定不同的构造方法
Constructor<String> constr =String.class.getConstructor(StringBuffer.class);
String string=constr.newInstance(new StringBuffer("abc"));
//如果需要调用无参构造方法,可以使用Class.newInstance();
System.out.println(string);
//-----------------------------------------------------
//反射:获取对象的成员变量
ReflectPoint r = newReflectPoint(3,5);
//getDeclaredField取出声明过的变量,包括public和private
Field fy = r.getClass().getDeclaredField("y");
System.out.println(fy.get(r));
Field fx =r.getClass().getDeclaredField("x");
fx.setAccessible(true);//暴力反射,将private成员设置成可见,才能被get获取
System.out.println(fx.get(r));
//----------------------------------------------------
//练习
changeStringValue(r);
System.out.println(r);
//----------------------------------------------------
//反射:获取对象的成员方法
Method sumMethod =r.getClass().getMethod("getSum",int.class,int.class);
//调用r对象的getSum方法,并且如果该方法是static的,参数可以是null
//sumMethod.invoke(null,6,7);
//参数类型不确定的情况,可以使用Object数组:sumMethod.invoke(r,newObject[]{每个参数的值});
int sum =(Integer)sumMethod.invoke(r,6,7);
System.out.println(sum);
//----------------------------------------------------
//反射:调用另一个类的main函数(类的静态方法)
//即:对接收数组参数的成员方法进行反射
//需要先设置运行对话框里先设置program argument的值(完整类名)
String startClassName =args[0];
Method mainMethod =Class.forName(startClassName).getMethod(
"main", String[].class);
//当String数组传入时会被拆包,因此需要使用newObject[]{String[]}
//或者(Object)强制转换,阻止对该数组的拆包。
System.out.println(mainMethod.invoke(null,
new Object[]{newString[]{"111","222","333"}}));
//---------------------------------------------------
//元与对象的区别
int[] a1 = new int[3];
int[] a2 = new int[4];
int[][] a3 = newint[2][3];
String[] a4 = new String[3];
System.out.println(a1.getClass() == a2.getClass());
//System.out.println(a1.getClass() == a3.getClass());
//System.out.println(a1.getClass() == a4.getClass());
//在看视频的时候,老师定义了几个数组,然后进行比较,在Eclipse中代码不报错,但是,我自己运行的时候,出现了如下错误:
//Incompatible operandtypes Class<capture#8-of ? extends int[]> and Class<capture#9-of ?extends int[][]>
//解析:查阅了网上的资料,解释如下:
//对于数组类型的变量编译器在编译时,其实已经知道类型了。也就是说,编译器认为这2个类型是不一样的。根本没有比较的需要。
//对于总是不成立的比较,其实是没有意义的。这个编译错误是 JDK 1.6的新功能
//而老师使用的恰好是JDK1.5的版本,我使用的是1.6版本。
System.out.println(a1.getClass().getSuperclass().getName());
System.out.println(a4.getClass().getSuperclass().getName());
Object obj1 =a1;
Object obj2 =a4;
//Object[] obj3 = a1;//编译失败,因为al里装的是int元而不是Object对象
Object[] obj4 = a3;
Object[] obj5 = a4;
//Arrays.asList(Object[] obj),而int是元数组而不是对象数组,因此不符合参数条件
//这是兼容JDK1.4的用法
System.out.println(Arrays.asList(a1));//hash值
System.out.println(Arrays.asList(a4));//元素值
//-----------------------------------------------------------
//数组的反射
//当有一个Object数组,元素值类型不一,就需要用到反射来取元素及元素类型
printArray(a4);
}
public static voidprintArray(Object obj){
Class class_ex =obj.getClass();
if(class_ex.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);
}
}
public static voidchangeStringValue(Object obj)throws Exception{
Field[] fields =obj.getClass().getFields();
for (Field field :fields) {
if(field.getType()==(String.class)) {
String newstr =field.get(obj).toString().replaceAll("b","a");
//将obj对象的field字段设置成newstr
field.set(obj,newstr);
}
}
}
}
class TestArgument{
public static void main(String[]args){
for (String string :args) {
System.out.println(string);
}
}
}
-----------------------------------------------------------------
public class ReflectPoint {
private int x;
public int y;
public Stringstr1="ball";
public Stringstr2="baseball";
public Stringstr3="itcast";
publicReflectPoint(int x,int y) {
this.x = x;
this.y = y;
}
@Override
public String toString(){
return "x="+x+" ;y= "+y+"\n"+"str1="+str1+"" +
" ;str2="+str2+" ;str3= "+str3;
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result+ x;
result = prime * result+ y;
return result;
}
@Override
public booleanequals(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;
}
public int getSum(inta,int b){
return a+b;
}
}
-----------------------------------------------------------------
HashCode与内存泄露
import java.util.*;
public class ReflectTest2 {
public static voidmain(String[] args){
Collection coll = newHashSet();
ReflectPoint r1 = newReflectPoint(3,3);
ReflectPoint r2 = newReflectPoint(5,5);
ReflectPoint r3 = newReflectPoint(3,3);
coll.add(r1);
coll.add(r2);
coll.add(r3);//覆盖了HashCode方法和equals方法,因此视为与r1相同元素
coll.add(r1);
System.out.println(coll.size());//2
r1.y=2;
coll.remove(r1);
System.out.println(coll.size());//2
//原因是:当涉及到HashCode值的字段发生了改变,那么再去搜索就找不到原来
//对应的HashCode的元素了,但是程序员察觉不到,就造成了内存泄露。
}
}