14、泛型与其他API

十四、泛型与其他API

1、系统结构图(xmind)

——泛型:

——2.其他API


2、tips

——1.泛型

1.JDK1.5的集合类希望在定义集合时,明确表明你要向集合中装入那种类型的数据,无法加入指定类型以外的数据

2.泛型是提供给javac编译器使用的可以限定集合中的输入类型说明的集合时,会去掉“类型”信息,使程序运行效率不受影响,对参数化的泛型类型,getClass()方法的返回值和原始类型完全一样。

3.由于编译生成的字节码会去掉泛型的类型信息,只要能跳过编译器,就可以往某个泛型集合中加入其它类型的数据,如用反射得到集合,再调用add方法即可。

4.没有使用泛型时,只要是对象,不管是什么类型的对象,都可以存储进同一个集合中。使用泛型集合,可以将一个集合中的元素限定为一个特定类型,集合中只能存储同一个类型的对象,这样更安全;并且当从集合获取一个对象时,编译器也可以知道这个对象的类型,不需要对对象进行强制类型转换,这样更方便。泛型就是把原来的类名进行了延长!在JDK 1.5中,你还可以按原来的方式将各种不同类型的数据装到一个集合中,但编译器会报告unchecked警告。

5.通常在集合框架中很常见,只要见到<>就要定义泛型。其实<>就是用来接收类型的。当使用集合时,将集合中要存储的数据类型作为参数传递到<>中即可。

6.编译器不允许创建泛型变量的数组。即在创建数组实例时,数组的元素不能使用参数化的类型,如:Vector<Integer>vectorList[] = new Vector<Integer>[10];//错误的。

7.Collection<?>  a可以与任意参数化的类型匹配,但到底匹配的是什么类型,只有以后才知道,所以:a=newArrayList<Integer>和a=new ArrayList<String>都可以,但a.add(new Date())或a.add(“abc”)都不行。

8.语法格式:

       class Utils<XX>
      {
             private XX s;
             public void setxx(XX s)
            {
                 this.s=s;
             }
             public XX getXX()
             {
                 return s;
             }
       }


总结:

对泛型的定义:

第一、定义泛型:当又不确定的类型需要传入到集合中,需要定义泛型。

第二、定义泛型类:如果类型确定后,所操作的方法都是属于此类型,则定义泛型类。

第三、定义泛型方法:如果定义的方法确定了,里面所操作的类型不确定,则定义泛型方法。


——2.编译器的泛型

1、编译器判断范型方法的实际类型参数的过程称为类型推断,类型推断是相对于知觉推断的,其实现方法是一种非常复杂的过程。

2、根据调用泛型方法时实际传递的参数类型或返回值的类型来推断,具体规则如下:

      1)当某个类型变量只在整个参数列表中的所有参数和返回值中的一处被应用了,那么根据调用方法时该处的实际应用类型来确定,这很容易凭着感觉推断出来,即直接根据调用方法时传递的参数类型或返回值来决定泛型参数的类型。

                  栗:swap(new String[3],3,4)

                         ——>    static <E> voidswap(E[] a, int i, int j)

      2)当某个类型变量在整个参数列表中的所有参数和返回值中的多处被应用了,如果调用方法时这多处的实际应用类型都对应同一种类型来确定,这很容易凭着感觉推断出来

                  栗:add(3,5)

                         ——>   static<T> T add(T a, T b)

      3)当某个类型变量在整个参数列表中的所有参数和返回值中的多处被应用了,如果调用方法时这多处的实际应用类型对应到了不同的类型,且没有使用返回值,这时候取多个参数中的最大交集类型,例如,下面语句实际对应的类型就是Number了,编译没问题,只是运行时出问题:

                  栗:fill(new Integer[3],3.5f)

                         ——>    static<T> void fill(T[] a, T v)


      4)当某个类型变量在整个参数列表中的所有参数和返回值中的多处被应用了,如果调用方法时这多处的实际应用类型对应到了不同的类型,并且使用返回值,这时候优先考虑返回值的类型,例如,下面语句实际对应的类型就是Integer了,编译将报告错误,将变量x的类型改为float,对比eclipse报告的错误提示,接着再将变量x类型改为Number,则没有了错误:

                  栗: intx =(3,3.5f)

                         ——>   static<T> T add(T a, T b)

      5)参数类型的类型推断具有传递性,下面第一种情况推断实际参数类型为Object,编译没有问题,而第二种情况则根据参数化的Vector类实例将类型变量直接确定为String类型,编译将出现问题:

                  栗: copy(newInteger[5],new String[5])

                         ——>   static<T> void copy(T[] a,T[]  b);

                          copy(newVector<String>(), new Integer[5])

                         ——>   static<T> void copy(Collection<T> a , T[] b);

——3.实例

import java.util.*;

//人  父类
class Person
{
	private String name;
	Person(String name)
	{
		this.name = name;
	}
	public String getName()
	{
		return name;
	}
}
//学生  继承父类
class Student extends Person
{
	Student(String name)
	{
		super(name);
	}
}

class  Demo
{
	public static void main(String[] args) 
	{
		ArrayList<Person> al = new ArrayList<Person>();
		al.add(new Person("abc1"));
		al.add(new Person("abc2"));
		al.add(new Person("abc3"));
		printColl(al);//父类对象的元素集合可以调用

		ArrayList<Student> al1 = new ArrayList<Student>();
		al1.add(new Student("abc--1"));
		al1.add(new Student("abc--2"));
		al1.add(new Student("abc--3"));
		printColl(al1);  //子类对象的元素集合也可以调用
	}

//定义一个上限的泛型方法
public static void printColl(Collection<? extends Person> al)
	{
		Iterator<? extends Person> it = al.iterator();
		while(it.hasNext())
		{
			System.out.println(it.next().getName());
		}
	}
}


——4.System类

1、字段摘要
       out:标准输出流。默认是控制台。
       in:标准输入流。默认是键盘。
2、方法
       1)获取系统的属性信息:PropertiesgetProperties();。
       2)说明:
             1.此方法返回的双列集合,即键值对;因为Properties是Hahstable的子类,也就是Map集合的一个子类对象,那么通过Map方法取出该集合中的元素。
             2.该集合存储的都是字符串,没有泛型定义。
       3)获取指定属性信息:String getProperty(Stringkey);。
       4)在系统内定义特有信息:String setProperty(Stringkey,String value);。
       5)如何在jvm启动时,加载一些属性信息:通过命令:java -D<name>=<value>可以设置特有的系统属性信息。

——5.Runtime类

1、该类中并没有提供构造函数。说明不可以new对象。那么会直接想到该类中的方法都是静态的。查阅API文档发现,该类中还有非静态方法。说明该类肯定会提供了方法获取本来对象。而且该方法是静态的,并返回值类型是本来类型。由以上特点可以看出该类使用了单例设计模式完成。
2、方法
        1)获取本类对象 :static RuntimegetRuntime();
        2)在单独的进程中执行指定字符串命令:Processexec(String command);
        3)在Process中有一个杀掉子进程的方法,可将exec方法开启的进程结束:void destroy();

——6.Date类
1.  Date类表示特定的瞬间,精确到毫秒。
     java中的默认显示格式为:Mon Jun 10 22:35:21 CST2013           
2、自定义格式
        默认的格式不一定满足每个人的需求,那么就需要自定义格式化模式。因为Date类中的大部分方法已过时,所以只能找其子类来实现。子类DateFormat中有format方法可以实现,但是DateFormat是抽象类,不能实例化。但是其下有个SimpleDateFormat子类,可以定义时间模式。
具体步骤:
        1)创建Date对象
        2)将时间模式封装到SimpleDateFormat对象中
        3)调用format方法让格式化模式指定Date对象

——7.Calendar类

方法:

1、基本获取时间
        1)获取年份:Calendar.YEAR
        2)获取月份:Calendar.MONTH
        3)获取日期:Calendar.DAY_OF_MONTH
        4)获取星期:Calendar.DAY_OF_WEEK
        5)获取小时:Calendar.HOUR_OF_DAY
        6)获取分钟:Calendar.MINUTE
        7)获取秒数:Calendar.SECOND
2、设置时间:
        1)根据日历的规则,为给定的日历字段添加或减去指定的时间量:
                void add(int field,int amount);
        2)获取指定的日历字段对应的时间值:
                int get(int field);

        3)将给定日历字段对应的时间值设置为给定值:
                void set(int field,int value);
       设置日历字段 YEAR、MONTH和DAY_OF_MONTH的值:
                void set(int year,int month,int date);
栗:
/**
 * 需求:编写程序,该程序启动后用户可以按“yyyy-MM-dd”的格式输入一个日期,程序计算这一
 * 天是星期几,并且计算出是一年中的第几天。
 */
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Scanner;
import java.util.Date;

class DateDemo
{
	public static void main(String[] args)throws Exception
	{
		//运用scanner接收键盘输入
		Scanner sc = new Scanner(System.in);
		System.out.print("请输入日期(格式为yyyy-MM-dd):");
		OutputDate(sc);
		sc.close();
	}
	
	public static void OutputDate(Scanner sc)throws Exception
	{
		//定义格式
		SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
		//scanner获取的格式进行解析存入d这个日期值中
		Date d = sdf.parse(sc.next());
		//定义一个Calendar日历用于获取需要的两个值
		Calendar cd = Calendar.getInstance();
		//将需要输出的时间值存于日历中
		cd.setTime(d);
		//获取并输出星期
		getWeek(cd);
		
		System.out.println(",是今年的第"+cd.get(Calendar.DAY_OF_YEAR)+"天。");
	}

	public static void getWeek(Calendar cd)
	{
		//获取星期的值用于查找表相对应的字符串并输出
		int num = cd.get(Calendar.DAY_OF_WEEK);
		String[] week = {"星期日","星期一","星期二","星期三","星期四","星期五","星期六"};
		System.out.print("今天是:"+week[num-1]);
	}
}

运行结果:


——8.Math类
方法:
        1、doubleceil(double d);//返回大于指定数据的最小整数
        2、doublefloor(double d);//返回小于指定数据的最大整数
        3、double pow(doublea,double b);//返回a的b次方
        4、long round(doubleb);//返回b四舍五入的值
        5、doublerandom();//返回正号的double值,是一个大于等于0.0且小于1.0的随机数

三、Random类
        这是java.util中的一个单独的类,该类对象用于获取随机数。与Math中的random方法是一样的,不过这个类有自身的方法,可以将相应的随机数强转为指定基本数据类型。







  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值