黑马程序员_Java基础加强(1)

时间:2023-02-18 18:04:32

android培训java培训、期待与您交流!


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的元素了,但是程序员察觉不到,就造成了内存泄露。

    }

}