JDK1.5java新特性

时间:2021-02-05 11:05:48

JDK1.5java增加的新特性:

自动装箱/拆箱      增强for     泛型      枚举      静态导入      可变参数

1 自动装箱/拆箱

* JDK1.5允许开发人员把一个基本类型直接赋给对应的包装类变量或者赋给Object类型的变量,这个过程称为自动装箱。

* 自动装箱和自动拆箱相反,即把包装类对象直接赋给一个对应的基本类型变量。

也就是基本类型与其对应的引用类型之间的转换,但是不能乱用,一般我们要采用基本类型,如果在使用时不注意,可能会造成程序低效率运行。

int ----- Integer
float ----- Float
double ---- Double
char ------- Character
boolean ---- Boolean
short ---- Short
long ---- Long
byte ---- Byte

比如:

Long  sum = 0;

for(long i =0;i<10000;i++){

  sum += i;

}

此处造成程序低效率就在于频繁的进行拆箱装箱操作。注意sum被定义成了Long而不是long。

还有两个高精度的类

  BigInteger

  BigDecimal


2 增强for

优缺点:

  优点:遍历更简单,在集合遍历中增强for就是代替迭代器的

  弊端:增强for的目标不能为null(即数组或集合不能为null),所以在遍历之前需要进行判断

格式:

  for(元素数据类型 变量:数组或集合){

      ....................................
      使用变量,该变量就是集合或者数组中的元素
  }

举例:

String[]  names = {"小明","小李","小王"};

for(String  name : names){

   System.out.println("hello "+name);

}


3 泛型

引入泛型的好处:

     1)去除了原先必须的冗长的强制类型转换

     2)能够在编译期检测出错误,不至于在运行时才发现,消除了潜在的错误,提高代码运行的可靠性。

     3)使用XJad.exe反编译工具可以看到,在编译期间泛型被擦除。

泛型基本应用:

1)泛型类

*把泛型定义在类上

*格式:public class 类名<泛型类型1.....>

*注意:泛型类型必须是引用类型

2)泛型方法

* 把泛型定义在方法上:方法可以接收任意类型

* 格式:public<泛型类型> 返回类型 方法名(泛型类型)

3)泛型接口

* 把泛型定义在接口上

* 格式:public interface 接口名<泛型类型1.....>

例1:一个简单的泛型类

package basic.demo;

public class FanTypeClass<T>{
  private T obj;

  public T getObj() {
    return obj;
  }

  public void setObj(T obj) {
    this.obj = obj;
  }
}

测试类:

package basic.demo;

public class FanTypeDemo {
  public static void main(String[] args) {

    FanTypeClass<String> obj1 = new FanTypeClass<String>();  //使用泛型
    obj1.setObj("hello");   //这时候set方法接收的参数必须是String,如果不传该类型,在编译期就会报错
    String s = obj1.getObj();  //这种方式下不需要强转
    System.out.println(s);

    FanTypeClass obj2 = new FanTypeClass();  //没使用泛型
    obj2.setObj(3);
    int num = (int) obj2.getObj();          //需要进行强转,如果不小心写错了就会产生ClassCastException,即类型转换异常

    String num = (String)obj2.getObj();    //这样写在编译期并不会报错,但是一旦执行就会产生类型转换异常
    System.out.println(num);
  }
}

例2:泛型方法的使用

package basic.demo;

public class FanTypeMethod {
  public <T> void show(T t){
    System.out.println(t);
  }
}

测试类:

package basic.demo;

public class FanTypeDemo {
  public static void main(String[] args) {

    FanTypeMethod ft = new FanTypeMethod();
    ft.show("hello");
    ft.show(100);
    ft.show(true);

  }

}

输出结果:

hello

100

true

例3 泛型接口

分为两种用法:子类实现时已知将来使用类型和子类实现时未知使用类型(这一种常见)

package basic.demo;

public interface FanTypeInter<T>{
  public abstract void show(T t);
}

package basic.demo;   //子类未知要使用的类型,故还是用泛型T

public class FanTypeInterImpl<T>  implements FanTypeInter<T> {
  @Override
  public void show(T t) {
    System.out.println(t);
  }
}

测试:

package basic.demo;

public class FanTypeDemo {
  public static void main(String[] args) {
    FanTypeInter<String> fti1 = new FanTypeInterImpl<String>();
    fti1.show("hello");

    FanTypeInter<Integer> fti2 = new FanTypeInterImpl<Integer>();
    fti2.show(100);

  }

}

泛型高级应用(通配符):

1)泛型通配符 <?>

*任意类型,如果没有明确,那么就是Object以及任意的java类

2)? extends E

* 向下限定,E及其子类

3)? super E

* 向上限定,E及其父类

例:理解高级应用:红色标注皆为错误用法

package basic.demo;

import java.util.ArrayList;
import java.util.List;

public class FanTypeSeniorDemo {
  public static void main(String[] args) {
    //泛型如果明确写,必须前后一致
    // List<Object> list1 = new ArrayList<Object>();
    // List<Object> list2= new ArrayList<Person>(); //error
    // List<Object> list3 = new ArrayList<Student>(); //error
    // List<Object> list4 = new ArrayList<Teacher>(); //error

    //?表示任意类型都是可以的
    // List<?> list1 = new ArrayList<Object>();
       // List<?> list2= new ArrayList<Person>();
    // List<?> list3 = new ArrayList<Student>();
    // List<?> list4 = new ArrayList<Teacher>();

    //? extends E 向下限定:E及其子类
    // List<? extends Person> list1 = new ArrayList<Object>(); //error
    // List<? extends Person> list2= new ArrayList<Person>();
    // List<? extends Person> list3 = new ArrayList<Student>();
    // List<? extends Person> list4 = new ArrayList<Teacher>();

    //? super E 向上限定 E及其父类
    List<? super Person> list1 = new ArrayList<Object>();
    List<? super Person> list2= new ArrayList<Person>();
    List<? super Person> list3 = new ArrayList<Student>(); //error
    List<? super Person> list4 = new ArrayList<Teacher>(); //error

  }

}
class Person{

}
class Student extends Person{

}
class Teacher extends Person{

}


4 枚举

 关键字Enum,用于定义一个枚举类

为什么需要枚举?
一些方法在运行时,它需要的数据不能是任意的,而必须是一定范围内的值,此类问题在JDK5以前采用自定义带有枚举功能的类解决,Java5以后可以直接使用枚举予以解决。

枚举类具有如下特性
  枚举类也是一种特殊形式的Java类。
  枚举类中声明的每一个枚举值代表枚举类的一个实例对象。
  与java中的普通类一样,在声明枚举类时,也可以声明属性、方法和构造函数,但枚举类的构造函数必须为私有的(这点不难理解)。
  枚举类也可以实现接口、或继承抽象类。
  JDK5中扩展了switch语句,它除了可以接收int, byte, char, short外,还可以接收一个枚举类型。
  若枚举类只有一个枚举值,则可以当作单例设计模式使用。

Java中声明的枚举类,均是java.lang.Enum类的子类,它继承了Enum类的所有方法。

常用方法:
  name():返回此枚举常量的名称,在其枚举声明中对其进行声明。
  ordinal():返回枚举常量的序数(它在枚举声明中的位置,其中初始常量序数为零)。也可以称为索引
  valueOf(Class enumClass, String name)
  values()     此方法虽然在JDK文档中查找不到,但每个枚举类都具有该方法,它遍历枚举类的所有枚举值非常方便。


5 静态导入

可以直接导入到方法的级别。

格式:import static 包名....类名.方法名;

注意:

  *方法必须是静态的

  *如果有多个同名的静态方法,容易不知道使用谁??这个时候要使用该特性,需要加前缀。由此可见,意义不大。所以一般不用,仅需知道就行。

例1:使用静态导入,使用静态方法时直接使用。

package basic.demo;

import static java.lang.Math.max;
import static java.lang.Math.min;

public class StaticImportDemo {
  public static void main(String[] args) {

    int num1 = 23;
    int num2 = 30;

    max(num1,num2);
    min(num1,num2);
  }
}

例2:编写重名的静态方法,然后一起在StaticImportDemo使用:

package basic.demo;

public class MyMath {
  public static int max(int num1,int num2){
    return num1>num2?num1:num2;
  }

  public static int min(int num1,int num2){
    return num1>num2?num2:num1;
  }
}

package basic.demo;

import static java.lang.Math.max;
import static java.lang.Math.min;

import static basic.demo.MyMath.max;
import static basic.demo.MyMath.min;  //使用都加前缀了,也就不需要静态导包了

public class StaticImportDemo {
  public static void main(String[] args) {

    int num1 = 23;
    int num2 = 30;

    java.lang.Math.max(num1,num2);  //必须加包名前缀,否则报错
    basic.demo.MyMath.min(num1,num2);//必须加包名前缀,否则报错,但是这样一加前缀就没有导包的意义了。。。。。
  }
}


6 可变参数

Effective Java不建议使用该特性,而且在java编程思想中也说了此特性的隐患。

*定义方法的时候不知道定义多少个参数,故引入可变参数的概念。

使用格式:

* 修饰符  返回值类型  方法名(数据类型...变量名){}

注意:

 * 这里的变量其实是一个数组

*如果一个方法有可变参数,并且有多个参数,那么,可变参数肯定是最后一个,否则编译器报错。

举例:

*Arrays工具类中的一个方法:

  public static <T>List<T> asList(T...a)

例1:简单使用:

package basic.demo;
public class ChangeParaDemo {
  public static void main(String[] args) {
    int[] data = {1,2,3,4};
    System.out.println(sum(data));

    int sum = sum(2,3,4,5);
    System.out.println(sum);

    sum = sum(2,5);
    System.out.println(sum);
  }

  private static int sum(int...num) {
    int result = 0;
    for(int i:num){
      result += i;
    }
    return result;
  }
}

例2 方法中有多个参数:java在编译器期间就强调可变参数必须在最后的位置。

package basic.demo;
public class ChangeParaDemo {
  public static void main(String[] args) {
    int sum = sum(100,1,2,3,4);
    System.out.println(sum);       //如果把可变参数放在方法参数的最后,结果是10

    int sum = sum(1,2,3,4,100);

    System.out.println(sum);   //如果可变参数放在前面,会报错
  }

  private static int sum(int special,int...num) {
    int result = 0;
    for(int i:num){
      result += i;
    }
    return result;
  }

  //error: 错误写法

  //出错原因,因为可变参数变量其实是一个数组,如果把可变参数放在前面,则special变量被视为数组的一部分,也就是说你只传了一个参数,和原方法参数个数不一致

  private static int sum(int...num,int special) {
    int result = 0;
    for(int i:num){
      result += i;
    }
    return result;
  }

}