高新技术---枚举及部分JDK1.5新特性

第一讲     枚举

一、概述

这里说的枚举,不是集合vector的特有枚举迭代器,而是JDK1.5的一个新特性。之所以单独拿它开刷,是这个知识点比较重要,同时相对来说比较难理解一些。

为什么要有枚举

问题:要定义星期几或性别的变量,该怎么定义?假设用1-7分别表示星期一到星期日,但有人可能会写成int weekday = 0;或即使使用常量方式也无法阻止意外。

枚举就是要让某个类型的变量的取值只能为若干个固定值中的一个,否则,编译器就会报错。枚举可以让编译器在编译时就可以控制源程序中填写的非法值,普通变量的方式在开发阶段无法实现这一目标。

示例:

[java] view plaincopy

  1. /*
  2. * 用普通类如何实现枚举功能,定义一个Weekday的类来模拟枚举功能。
  3. 1、私有的构造方法
  4. 2、每个元素分别用一个公有的静态成员变量表示
  5. 3、可以有若干公有方法或抽象方法。采用抽象方法定义nextDay就将大量的if.else语句转移成了一个个独立的类。
  6. */
  7. package cn.itheima;
  8. public abstract class WeekDay {
  9. private WeekDay(){}
  10. public final static WeekDay SUN=new WeekDay(){
  11. public WeekDay nextDay(){
  12. return MON;
  13. }
  14. };
  15. public final static WeekDay MON=new WeekDay(){
  16. public WeekDay nextDay(){
  17. return SUN;
  18. }
  19. };
  20. public abstract WeekDay nextDay();
  21. public String toString(){
  22. return this==SUN?"SUM":"MON";
  23. }
  24. }

二、枚举的基本应用

1、通过enum关键字定义枚举类,枚举类是一个特殊的类,每个元素都是该类的一个实例对象。

2、用枚举类规定值,如上面的WeekDay类。以后用此类型定义的值只能是这个类中规定好的那些值,若不是这些值,编译器不会通过。

3、好处:在编译时期就会发现错误,表明值不符合,减少了运行时期的错误。

4、如果调用者想打印枚举类中元素的信息,需由编写此类的人定义toString方法。

注:枚举类是一个class,而且是一个不可被继承的final类,其中的元素都是类静态常量。

5、常用方法:

构造器:

1)构造器只是在构造枚举值的时候被调用。

2)构造器只有私有private,绝不允许有public构造器。这样可以保证外部代码无法重新构造枚举类的实例。因为枚举值是public static final的常量,但是枚举类的方法和数据域是可以被外部访问的。

3)构造器可以有多个,调用哪个即初始化相应的值。

非静态方法:(所有的枚举类都继承了Enum方法)

1)String toString() ;//返回枚举量的名称

2)int ordinal() ;//返回枚举值在枚举类中的顺序,按定义的顺序排

3)Class getClass() ;//获取对应的类名

4) String name();//返回此枚举常量的名称,在其枚举声明中对其进行声明。

静态方法:

1)valueOf(String e) ;//转为对应的枚举对象,即将字符串转为对象

2)values() ;//获取所有的枚举对象元素

示例:

[java] view plaincopy

  1. package cn.itheima;
  2. public class EnumDemo {
  3. public static void main(String[] args) {
  4. WeekDay weekDay=WeekDay.MON;
  5. System.out.println(weekDay);//输出枚举常量名
  6. System.out.println(weekDay.name());//输出对象名
  7. System.out.println(weekDay.getClass());//输出对应类
  8. System.out.println(weekDay.toString());//输出枚举对象名
  9. System.out.println(weekDay.ordinal());//输出此对象在枚举常量的次序
  10. System.out.println(WeekDay.valueOf("WED"));//将字符串转化为枚举常量
  11. System.out.println(WeekDay.values().length);//获取所以的枚举元素,并打印其长度
  12. }
  13. //定义枚举内部类
  14. public enum WeekDay{
  15. SUN(1),MON,TUE,WED,THI,FRI,SAT;//分号可有可无,但如果下面还有方法或其他成员时,分号不能省。
  16. //而且当有其他方法时,必须在这些枚举变量的下方。
  17. //无参构造器
  18. private WeekDay(){
  19. System.out.println("First");
  20. }
  21. //带参数的构造器
  22. private WeekDay(int day){
  23. System.out.println("Second");
  24. }
  25. }
  26. }

三、枚举的高级应用

1、枚举就相当于一个类,其中也可以定义构造方法、成员变量、普通方法和抽象方法。

2、枚举元素必须位于枚举体中的最开始部分,枚举元素列表的后要有分号与其他成员分隔。把枚举中的成员方法或变量等放在枚举元素的前面,编译器报告错误。

3、带构造方法的枚举

1)构造方法必须定义成私有的

2)如果有多个构造方法,该如何选择哪个构造方法?

3)枚举元素MON和MON()的效果一样,都是调用默认的构造方法。

4、带方法的枚举

如:

[java] view plaincopy

  1. /*
  2. * 抽象的枚举方法
  3. * 此时枚举中的常量需要子类来实现,这是可以利用内部类的方式来定义枚举常量
  4. * 带方法的枚举
  5. 1)定义枚举TrafficLamp
  6. 2)实现普通的next方法
  7. 3)实现抽象的next方法:每个元素分别是由枚举类的子类来生成的实例对象,这些子类
  8. 4)用类似内部类的方式进行定义。
  9. 5)增加上表示时间的构造方法
  10. * */
  11. package cn.itheima;
  12. public class EnumTest {
  13. public enum TrafficLamp{
  14. RED(30){
  15. public TrafficLamp nextLamp(){
  16. return GREEN;
  17. }
  18. },
  19. GREEN(30){
  20. public TrafficLamp nextLamp(){
  21. return YELLOW;
  22. }
  23. },
  24. YELLOW(5){
  25. public TrafficLamp nextLamp(){
  26. return RED;
  27. }
  28. };
  29. private int time;
  30. //构造器
  31. private TrafficLamp(int time){
  32. this.time=time;}
  33. //抽象方法
  34. public abstract TrafficLamp nextLamp();
  35. }
  36. }

小结:

1、匿名内部类比较常用

2、类的方法返回的类型可以是本类的类型

3、类中可定义静态常量,常量的结果就是自己这个类型的实例对象

4、枚举只有一个成员时,就可以作为一种单例的实现方式。

注:

1、所有的枚举都继承自java.lang.Enum类。由于Java不支持多继承,所以枚举对象不能再继承其他类。

2、switch语句支持int,char,enum类型,使用枚举,能让我们的代码可读性更强。

第二讲     JDK1.5其他部分新特性

一、静态导入

1、写法:

import staticjava.util.Arrays.*;//导入的是Arrays这个类中的所以静态成员。

import staticjava.lang.System.*//导入了Ssytem类中所以静态成员。

没加static导入的是类,加上static导入的全是某一个类中所以的静态成员。这样写在调用该类的静态方法时可以不用再写类名。如:Arrays.sort(数组);就可以直接写sort(数组);

2、注意:

当导入的两个类中有同名成员时,需要在成员前加上相应的类名。

当类名重名时,需要指定具体的包名。当方法重名时,指定具体所属的对象或者类。

示例:

[java] view plaincopy

  1. import java.util.*;
  2. import static java.util.Arrays.*;
  3. import static java.lang.System.*;
  4. class  StaticImport //extends Object
  5. {
  6. public static void main(String[] args)
  7. {
  8. out.println("haha");//打印输出时就可以直接省略书写System.
  9. int[] arr = {3,1,5};
  10. sort(arr);//使用Arrays工具类的方法sort时就可以省略书写Array.
  11. int index = binarySearch(arr,1);//半分查找也是一样可以省略
  12. out.println("Index="+index);
  13. //当没有指定继承时,所以类默认继承了Object,
  14. //因为toString方法都具备,所以为了区分,必须写上具体调用者
  15. out.println(Arrays.toString(arr));
  16. }
  17. }

二、增强for循环

1、格式:

for(数据类型变量名 :被遍历的集合(collection)或者数组) {执行语句}

2、说明

a、对集合进行遍历。只能获取集合元素。但是不能对集合进行操作。可以看作是迭代器的简写形式。

b、迭代器除了遍历,还可以进行remove集合中元素的动作。如果使用ListIterator,还可以在遍历过程中对集合进行增删改查的操作。

3、传统for和高级for的区别:

高级for有一个局限性。必须有被遍历的目标(集合或数组)。

传统for遍历数组时有索引。

建议在遍历数组的时候,还是希望使用传统for。因为传统for可以定义角标。

注意:变量类型前可加修饰符,如final(可被局部内部类访问到)。

示例:

[java] view plaincopy

  1. import java.util.*;
  2. class For
  3. {
  4. public static void main(String[] args)
  5. {
  6. //定义一个ArrayList集合
  7. ArrayList<String> al = new ArrayList<String>();
  8. al.add("abc1");
  9. al.add("abc2");
  10. al.add("abc3");
  11. for(String s : al)
  12. {
  13. System.out.println(s);//用高级for遍历集合
  14. }
  15. //传统for与高级for遍历数组
  16. int[] arr = {3,5,1};
  17. for(int x=0; x<arr.length; x++)
  18. {
  19. System.out.println(arr[x]);
  20. }
  21. for(int i : arr)
  22. {
  23. System.out.println("i:"+i);
  24. }
  25. //定义一个HashMap集合
  26. HashMap<Integer,String> hm = new HashMap<Integer,String>();
  27. hm.put(1,"a");
  28. hm.put(2,"b");
  29. hm.put(3,"c");
  30. //keySet取出方式的高级for遍历
  31. Set<Integer> keySet = hm.keySet();
  32. for(Integer i : keySet)
  33. {
  34. System.out.println(i+"::"+hm.get(i));
  35. }
  36. //entrySet取出方式的高级for遍历
  37. for(Map.Entry<Integer,String> me : hm.entrySet())
  38. {
  39. System.out.println(me.getKey()+"------"+me.getValue());
  40. }
  41. }
  42. }

三、可变参数:(方法的重载)VariableParameter

如果一个方法在参数列表中传入多个参数,个数不确定,那么每次都要复写该方法。这时可以用数组作为形式参数。但是在传入时,每次都需要定义一个数组对象,作为实际参数。在JDK1.5版本后,就提供了一个新特性:伤感的句子可变参数。

用…这三个点表示,且这三个点位于变量类型和变量名之间,前后有无空格皆可。

可变参数其实就是数组参数的简写形式。不用每一次都手动的建立数组对象。只要将要操作的元素作为参数传递即可。隐式将这些参数封装成了数组。

在使用时注意:可变参数一定要定义在参数列表的最后面。

示例:

[java] view plaincopy

  1. class  ParamMethodDemo
  2. {
  3. public static void main(String[] args)
  4. {
  5. show("haha",2,3,4,5,6);
  6. }
  7. public static void show(String str,int... arr)//...就表示可变参数
  8. {
  9. System.out.println(arr.length);
  10. }
  11. }

四、基本数据类型的自动拆箱与装箱

1、自动装箱:Integer iObj = 3;

2、自动拆箱:iObj + 2;

3、对于基本数据类型的说明:整数在-128 ~ 127之间的数,包装成Integer类型对象,会存入常量池中的缓存,再创建一个对象的时候,如果其值在这个范围内,就会直接到常量池中寻找,因为这些小数值使用的频率很高,所以缓存到常量池中,淘宝开店被调用时就方便很多。

4、享元模式(flyweight):

1)概述:有很多个小的对象,他们有很多相同的属性,把属性相同的部分变为同一个对象,这些属性称为内部状态。那些不同的属性,把他们变成方法的参数,称之为外部的状态。这种优化内存,只创建一个对象的模式,称之为享元模式。例如:Integer对象中对象取值范围在-128~127时,其值相同的对象想等,因为这些小的数被缓存在一个池中,被调用的频率较高,所以通常在池中取,从而导致对象相同。这就是典型的享元设计模式。

2)应用:

(1)word中输入英文字母,可创建26对象,每个对象值是出现的位置不同(坐标),所以可用一个对象调用位置的方法:如,字母i:i.display(intx,inty),将高度重复使用的char类型的字母i封装成一个对象使用。

(2)图标:window下的文件夹图标,只有名称这个属性不同,包含了很多其他相同的属性,那么可以应用享元模式。

3)valueOf(int x):Integer中静态方法,将一个整数转换为Integer,即把基本数据类型变为包装类。

时间: 2024-10-27 07:13:21

高新技术---枚举及部分JDK1.5新特性的相关文章

jdk1.5新特性和jdk1.7新特性

jdk1.5新特性 1.自动装箱和自动拆箱 自动装箱,把基本类型的值变成对象类型 Integer a = 2; 自动拆箱,把对象类型变成基本类型 int b = new Integer(2); 而不需要调用intValue int b = new Integer(2).intValue(); 包装类有 Integer Boolean Float Double Short Byte 2.枚举 java枚举具有简单和安全性. 3.泛型 泛型在编译的时候可以控制类型,比如说List<String>

黑马程序员——java高新---JDK1.5新特性和反射

------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 一.JDK1.5新特性 ——>静态导入 import和import static之间的区别: 1.import是导入一个类或某个包中所有的类. 2.import static是导入一个类中的某个静态成员或所有的静态成员. 注意: 1.当导入的两个类中有同名成员时,需要在成员前加上相应的类名. 2.当类名重名时,需要指定具体的包名. 3.方法重名时,需要指定具体所属的对象或者类. 代码示例: 1

黑马程序员------Java中jdk1.5新特性

Java培训.Android培训.iOS培训..Net培训.期待与您交流! JDK1.5新特性: 为什么会出现新特性: 新的技术出现是为了解决老的问题,Java语言为了提高开发者的开发效率,对之前的某些不利于提高效率的技术进行改进. 静态导入: 静态导入:可以导入某个类下的静态方法,静态导入后,可以不写类名而直接使用此类下的静态方法. 语法:import static 包名.类名.静态方法 代码示例: package com.itheima.day1; /** * 静态导入 * @author

JDK1.8新特性——Collector接口和Collectors工具类

JDK1.8新特性——Collector接口和Collectors工具类 摘要:本文主要学习了在Java1.8中新增的Collector接口和Collectors工具类,以及使用它们在处理集合时的改进和优化. 部分内容来自以下博客: https://www.jianshu.com/p/7eaa0969b424 流式处理 JDK1.8中新增的流式处理提供了一种高效且易于使用的处理数据的方式,它可以对集合执行非常复杂的查找.过滤和映射数据等操作,极大的简化了对于集合的使用.借助流式处理,可以像使用S

接口、接口优点、接口的属性和方法特性、接口与继承的区别、接口与抽象类的区别、匿名实现类、JDK1.8新特性、打印类名称

接口里的属性,必须是公共的/默认的&静态的&Final&初始化后的属性: 接口里所有方法都是抽象的. 接口和继承---先天有的用继承,后天学习的用接口: 接口特性-------- 1.接口不可以被实例化: 2.实现类必须实现接口中所有方法,否则实现类必须是抽象类 3.实现类可以实现多个接口,来弥补Java不能多继承: 4.接口中的变量都是静态常量:(静态的-可以通过类的名字点-直接调用:) ----------------------------------------------

jdk1.5新特性

package cn.itcast.jdk15; import java.util.ArrayList; /* jdk1.5新特性之-----自动装箱与自动拆箱. java是面向对象 的语言,任何事物都可以使用类进行描述,sun就使用了 一些类描述java中八种基本数据类型数据 基本数据类型 包装类型 byte Byte short Short int Integer long Long float Float double Double boolean Boolean char Charact

jdk1.5新特性之-------&gt;可变参数

/* jdk1.5新特性之------->可变参数 需求: 定义一个函数做加法功能(函数做几个数据 的加法功能是不确定). 可变参数的格式: 数据类型... 变量名 可变参数要 注意的细节: 1. 如果一个函数 的形参使用上了可变参数之后,那么调用该方法的时候可以传递参数也可以不传递参数. 2. 可变参数实际上是一个数组对象. 3. 可变参数必须位于形参中的最后一个参数. 4. 一个函数最多只能有一个可变 参数,因为可变参数要位于形参中最后一个位置上. */ public class Demo4

59. jdk1.5新特性之----增强for循环

/*jdk1.5新特性之----增强for循环:底层是一个迭代器 作用:简化迭代器书写格式 使用范围:实现了Iterable接口的对象或者数组对象 格式:    for(变量类型  变量名 :遍历目标){        //代码块    }    注意:    1.因为for in循环底层还是迭代器,所以在循环的时候我们不能修改迭代对象的长度    2.因为我们没有实例化迭代器对象,所以我们不能使用迭代器中的一些方法*/ 基本运用实例代码: public class Demo2 { public

jdk1.8新特性之lambda表达式

lambda表达式其实就是指一个匿名函数,应用最广泛的就是匿名内部类.在jdk1.8之前,我们定义一个匿名内部类可能需要写一大坨代码,现在有了lambda之后,可以写的很简洁了.但不是说lambda只能用来简化匿名内部类,从lambda的实际作用和表现上来看,就是一个变量指代一个代码块.而能够使用lambda表达式的一个前提要求是,该变量必须实现某个函数式接口.啥是函数式接口?参考jdk1.8新特性之函数式接口.看例子: 1.函数式接口 * Copyright (c) 1994, 2013, O