基础加强三 注解 泛型

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


注解:
   1,注解就相当于一个你的源程序中要调用的一个类,要在源程序中应用某个注解,得先准备好了这个注解类。
   就像你要调用某个类,得先有开发好这个类。
   2,注解相当于一种标记,在程序中加了注解就等于为程序打上了某种标记,没加,则等于没有某种标记,
   以后,javac编译器,开发工具和其他程序可以用反射来了解你的类及各种元素上有无何种标记,看你有什么
   标记,就去干相应的事。标记可以加在包,类,字段,方法,方法的参数以及局部变量上。
   3,在java.lang包中提供了常见的三种注解方式:
      1)Deprecated:提示成员等已经过时,不再推荐使用。
      2)Override:重写超类方法的注解。可对判断自己类中的方法是否覆盖了父类的方法。
      3)SuppressWarnings:压制警告注解,也就是取消警告


注释的应用--->注解类:
    1、 格式: @interface 名称{statement}
    2、元注解(注解的注解)
      一个注解有其生命周期(Retetion)和存放的位置(Target),这就可以通过元注解说明。
    1)Retetion:用于说明注解保留在哪个时期,加在定义的注解之上。
       ①一个注解的声明生命周期包含:
          java源程序-->class文件-->内存中的字节码
    第一、当在源程序上加了注解,javac将java源程序编译为class文件,可能会把源程序中的一些注解去掉,
    进行相应的处理操作,当我们拿到源程序的时候,就看不到这些注解了。
    第二、假设javac把这些注解留在了源程序中(或者说留在了class文件中),当运行此class文件的时候,
    用类加载器将class文件调入内存中,此时有转换的过程,即把class文件中的注解是否保留下来也不一定。
    注意:class文件中不是字节码,只有把class文件中的内部加载进内存,用类加载器加载处理后(进行完整的检查
    等处理),最终得到的二进制内容才是字节码。
       ②Reteton(枚举类)取值:
           Retetion.Policy.SOURSE:保留到java源文件时期,如 @Overried和 @SuppressWarning
           Retetion.Policy.CLASS: 保留到class文件时期(默认阶段)
           Retetion.Policy.RUNTIME:保留到运行时期,如 @Deprecated
    2)Taget:用于说明注解存放在哪些成分上,默认值是任何元素
    其值可设置为枚举类ElementType类中的任何一个:
        ANNOTATION_TYPE : 注释类型声明 
CONSTRUCTOR :构造方法声明 
FIELD :字段声明(包括枚举常量) 
LOCAL_VARIABLE :局部变量声明 
METHOD :方法声明 
PACKAGE :包声明 
PARAMETER :参数声明 
TYPE :类、接口(包括注释类型)或枚举声明   
    注意:其中代表类的值是TYPE。因为class、enum、interface和 @interface等都是属于Type的。不可用CLASS表示。


    3,为注解增加基本属性
       1)属性:
 一个注解相当于一个胸牌,如果你胸前贴了胸牌,就是传智播客的学生,否则,就不是。如果还想区
 分出是传智播客哪个班的学生,这时候可以为胸牌在增加一个属性来进行区分。加了属性的标记效果
 为: @MyAnnotation(color="red")


       2)添加属性:String color();
       添加属性时可以设定缺省值,也就是默认值,如:String value() default ”ignal”;
       如果添加的属性只有一个时,在使用时可以直接写入属性值,不需要把属性名也写入。
       如果属性有多个,但是除了其中一个其它的都设定了缺省值,那么写入的时候也可以直接写入属性值
  
   4,为注解增加高级属性
      1)注解的返回值类型有:基本类型,String,Class,枚举,注解,数组。
      2)数组类型的属性:
    定义:int[]arrayArr() default {1,2,3};     -->可不定义默认值
    应用: @MyAnnotation(arrayArr={2,3,4})  --> 可重新赋值
    注:若数组属性中只有一个元素(或重新赋值为一个元素),这时属性值部分可省略大括号。
      3)枚举类型的属性:
    假设定义了一个枚举类TraffLamp,它是EnumTest的内部类,其值是交通灯的三色。
    定义:EnumTest.TrafficLamplamp();
    应用: @MyAnnotation(lamp=EnumTestTrafficLamp.GREEN)
      4)注解类型的属性:
    假定有个注解类:MetaAnnotation,其中定义了一个属性:String value()
    定义:MetaAnnotationannotation() default @MetaAnnotation(”xxx”);
    应用: @MyAnnotation(annotation= @MetaAnnotation(”yyy”))  --> 可重新赋值
    可认为上面的 @MetaAnnotation是MyAnnotation类的一个实例对象,
    同样可以认为上面的 @MetaAnnotation是MetaAnnotation类的一个实例对象,调用:
    MetaAnnotation ma =MyAnnotation.annotation();
    System.out.println(ma.value());

代码演示:

@ItcastAnnotation(annotationAttr=@MetaAnnotation("myy"),color="red",value="abc",arrayAttr={1,2})//通过方法设置属性值。
public class AnnotationTest {

	/**
	 * @param args
	 */
	@SuppressWarnings("deprecation")
	@ItcastAnnotation("acb") //如果方法中只有value的话,这个value可以省略不写
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		System.runFinalizersOnExit(true);
		
		if(AnnotationTest.class.isAnnotationPresent(ItcastAnnotation.class)){//判断ItcastAnnotation是否存在于AnnotationTest上
			ItcastAnnotation annotation = 
					AnnotationTest.class.getAnnotation(ItcastAnnotation.class);
//			System.out.println(annotation);
			System.out.println(annotation.color());//取出的时候一样通过方法获取,不能直接用属性值获取。
			System.out.println(annotation.value());
			System.out.println(annotation.arrayAttr().length);
			
			System.out.println(annotation.lamp().nextLamp());
			System.out.println(annotation.annotationAttr().value());
		}
	}
}
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import cn.itcast.day01.EnumTest;

@Retention(RetentionPolicy.RUNTIME) //注解保持到运行时期
@Target({ElementType.METHOD,ElementType.TYPE})//表示注解可以放在方法上也可以放在类上
public @interface ItcastAnnotation {
//	String color();
	String color() default "blue" ;//为color设置缺省值,也就是默认值
	String value();
	//数组类型的属性
	int[] arrayAttr() default {3,4,5} ;
	//枚举类型的属性
	EnumTest.TrafficLamp lamp() default EnumTest.TrafficLamp.RED ;
	//注解类型的属性
	MetaAnnotation annotationAttr() default @MetaAnnotation("wbl");
	
}
public @interface MetaAnnotation {
	String value() ;
}


泛型:JDK1.5出现的新特性。

1,泛型的好处:
       1)因为使用泛型集合可以将一个集合中的元素限定为一个特定类型,集合中只能存储同一个类型的对象。如果
       在集合中存入其它类型的元素编译器就会报错,这样就可以将运行时期的ClassCastException异常转移到了
       编译事情,进行检查,并以编译失败来体现。这样有利于程序员尽早解决问题。
       2)当从集合中获取一个对象时,编译器也可知道这个对象的类型,就不需要对对象进行向下转型,
       避免了向下转型(强转)的麻烦

    2,去类型化:
       1)泛型是提供给javac编译器使用的,可以限定集合中的输入类型,让编译器挡住源程序中的非法输入,编译
    器编译带类型说明的集合时会去除掉“类型”信息,使程序运行效率不受影响,对于参数化的泛型类型,
    getClass()方法的返回值和原始类型完全一样。
       2)由于编译生成的字节码会去掉泛型的类型信息,只要能跳过编译器,就可以往某个泛型集合中加入其它
       类型的数据,例如,用反射得到集合,再调用其add方法即可。
    3,泛型定义中的术语
         如:ArrayList<E>类和ArrayList<Integer>
       1)ArrayList<E>整个称为泛型类型
       2)ArrayList<E>中的E称为类型变量或类型参数
       3)整个ArrayList<Integer>称为参数化类型
       4)ArrayList<Integer>中的Integer称为类型参数的实例或实际类型参数
       5)ArrayList<Integer>中的<>称为typeof
       6)ArrayList称为原始类型
       参数化:parametered,已经将参数变为实际类型的状态。

     4,参数化类型与原始类型的兼容性:
 1)参数化类型可以引用一个原始类型的对象,编译报告警告。
     例如, Collection<String> c = new Vector();//可不可以通过,编译器说的算。
 2)原始类型可以引用一个参数化类型的对象,编译报告警告。
     例如, Collection c = new Vector<String>();//原来的方法接受一个集合参数,新的类型也要能传进去。
 3)参数化类型不考虑类型参数的继承关系:
     Vector<String> v = new Vector<Object>(); //错误! 不写<Object>没错。
     Vector<Object> v = new Vector<String>(); //也错误!
 4)编译器不允许创建泛型变量的数组。即在创建数组实例时,数组的元素不能使用参数化的类型。
    例如,Vector<Integer> vectorList[] = new Vector<Integer>[10];也错误

     5,泛型通配符:?
 1)当操作的不同容器中的类型都不确定时,而且使用的都是元素从Object类中继承的方法。
 这时就要用通配符 ? 来表示即可。 
 2)使用?通配符可以引用其他各种参数化的类型,?通配符定义的变量主要用作引用,可以调用与参数化
 无关的方法,不能调用与参数化有关的方法。
 3)泛型限定:明确具体类型代表哪一个类型,明确 ?代表哪一个类型。
 将操作的类型限制在一个范围之内:? extends E :接收E类型或者E的子类型。这就是上限。
 下限:? super E: 接收E类型或者E的父类型。
 例如,上限:Vector<? extends Number> x = new Vector<Integer>();
       下限:Vector<? super Integer> x = new Vector<Number>();
 注意:限定通配符不论是上限还是下限总是包括自己。
       ?只能用作引用,不能用它去给其他变量赋值

 6,自定义泛型:
     1)泛型类型只能是对象的引用参数类型,不可以是基本数据类型。

     2)用于放置泛型的类型参数的尖括号应出现在方法的其他所有修饰符之后和在方法的返回类型之前,
     也就是紧邻返回值之前。按照惯例,类型参数通常用单个大写字母表示。
  例如,交换数组中的两个元素的位置的泛型方法语法定义如下:

public static <E> void swap(E[] a, int i, int j) {
   E temp = a[i];
   a[i] = a[j];
   a[j] = temp;
}

    3)除了在应用泛型时可以使用extends限定符,在定义泛型时也可以使用extends限定符,
    例如,Class.getAnnotation()方法的定义。并且可以用&来指定多个边界,
    如<V extends Serializable & cloneable> void method(){},
    普通方法、构造方法和静态方法中都可以使用泛型。
    也可以用类型变量表示异常,称为参数化的异常,可以用于方法的throws列表中,但是不能用于catch子句中。
    4)在泛型中可以同时有多个类型参数,在定义它们的尖括号中用逗号分,例如:
  public static <K,V> V getValue(K key) { return map.get(key);}

 

 通配符 ? 和自定义泛型<T>的区别:
     1)T限定了类型,传入什么类型即为什么类型,可以定义变量,接收赋值的内容。
     2)?为通配符,也可以接收任意类型但是不可以定义变量。
     它虽然提高了扩展性,可还是有一个局限性,就是不能使用其他类对象的特有方法。比如集合中的add方法。
     3)总结:通配符方案要比泛型方法更有效,当一个类型变量用来表达两个参数之间或参数和返回值之间
     的关系时,即同一个类型变量在方法签名的两处被使用,或者类型变量在方法体代码中也被使用,
            而不是仅在签名的时候使用,才需要使用泛型方法。

 7,类型参数的类型推断:
     1)编译器判断范型方法的实际类型参数的过程称为类型推断,类型推断是相对于知觉推断的,其实现
     方法是一种非常复杂的过程。
     2)根据调用泛型方法时实际传递的参数类型或返回值的类型来推断,具体规则如下:
  1,当某个类型变量只在整个参数列表中的所有参数和返回值中的一处被应用了,那么根据调用方法时
  该处的实际应用类型来确定,这很容易凭着感觉推断出来,即直接根据调用方法时传递的参数类型
  或返回值来决定泛型参数的类型,例如:
   swap(new String[3],3,4)——>static <E> void swap(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,则没有了错误:
    int x =(3,3.5f)———>static <T> T add(T a, T b)
  5,参数类型的类型推断具有传递性,下面第一种情况推断实际参数类型为Object,编译没有问题,
  而第二种情况则根据参数化的Vector类实例将类型变量直接确定为String类型,编译将出现问题:
   copy(new Integer[5],new String[5])——>static <T> void copy(T[] a,T[]  b);
  copy(new Vector<String>(), new Integer[5])——>static <T> void copy(Collection<T> a , T[] b);


通过反射获取泛型类型的实际参数类型:
  因为泛型是提供给javac编译器使用的,在编译结束之后泛型类型就被去除了,所以通过正常的字节码是无法获得泛型
  类型的,在这里我们可以定义一个方法,将泛型类型作为参数传给该方法,然后通过方法的method方法获取泛型类型。

 

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.Vector;

import cn.itcast.day01.ReflectPoint2;

public class GenericTest {

	/**
	 * @param args
	 * @throws Exception 
	 * @throws SecurityException 
	 */
	public static void main(String[] args) throws Exception {
		// TODO Auto-generated method stub
		
		genericMethod_1();
		genericMethod_2();
		
		swap(new String[]{"java","csdn","itcast"},1,2);
		
		Object obj = "abc";
		String str = autoConvert(obj);
		
		copy1(new Vector<String>(),new String[10]);
		copy2(new Date[5],new String[5]);
		//copy1(new Vector<Date>(),new String[10]);//会报错,编译时类型推断,都把它们当作Object 。
		
		
		genericMethod_4();
	}
	
	//通过反射的方式获得泛型的实际参数类型。
	public static void genericMethod_4() throws Exception{
		Method applyMethod = GenericTest.class.getMethod("applyVector", Vector.class);
		Type[] types = applyMethod.getGenericParameterTypes();
		ParameterizedType pType = (ParameterizedType)types[0];
		System.out.println(pType.getRawType());//获取原始类型
		System.out.println(pType.getActualTypeArguments()[0]);//获取实际的参数类型
	}
	public static void applyVector(Vector<Date> vector){
		
	}
	
	public static void genericMethod_3(){
		GenericDao<ReflectPoint2> dao = new GenericDao<ReflectPoint2>();
		dao.add(new ReflectPoint2(3, 5));
	}
	
	//把任意参数类型的集合中的数据安全地复制到相应类型的数组中。
	public static <T> void copy1(Collection<T> desk,T[] src){
		
	}
	
	//把任意参数类型的一个数组中的数据安全地复制到相应类型的另一个数组中。
	public static <T> void copy2(T[] desk,T[] src){
		
	}
	
	//打印出任意参数化类型的集合中的所有内容。
	private static <T> void printCollection(Collection<T> coll){
		System.out.println(coll.size());
		for(Object obj : coll){
			System.out.println(obj);
		}
	}
	
	//自动将Object类型的对象转换成其他类型。
	private static <T> T autoConvert(Object obj){
		return (T)obj;
		
	}
	
	private static <T> void swap(T[] a,int i,int j){
		T temp = a[i] ;
		a[i] = a[j] ;
		a[j] = temp ;
	}
	
	public static void genericMethod_2() {
		HashMap<String, Integer> maps = new HashMap<String, Integer>();
		maps.put("wbl", 24);
		maps.put("myy", 22);
		maps.put("ly", 23);
		Set<Map.Entry<String,Integer>> entrySet = maps.entrySet();
		for(Map.Entry<String,Integer> entry : entrySet){
			System.out.println(entry.getKey()+":"+entry.getValue());
		}
		
	}

	public static void genericMethod_1() throws Exception{
		Constructor<String> constructor = 
				String.class.getConstructor(StringBuffer.class);
		String str = 
				constructor.newInstance(new StringBuffer("abc"));
		System.out.println(str.charAt(2));
		
		ArrayList<Integer> coll1 = new ArrayList<Integer>(); 
		ArrayList<String> coll2 = new ArrayList<String>(); 
		
		coll1.add(3);
		coll2.add("abc");
		System.out.println(coll2.get(0));
		
		System.out.println(coll1.getClass()==coll2.getClass());//打印结果表明它们是同一份字节码,这说明,泛型编译器在编译带类型说明
																//的集合时会去除掉“类型”信息
		//通过反射的方式往coll1中存储String类型的对象。
		coll1.getClass().getMethod("add", Object.class).invoke(coll1, "abc");
		coll2.getClass().getMethod("add", Object.class).invoke(coll2, 2);
		System.out.println(coll1);
		System.out.println(coll1.get(1));
		System.out.println(coll2);
		System.out.println(coll2.getClass().getMethod("get", int.class).invoke(coll2,1));

	}

}


定义泛型类型:

    1)如果类的实例对象中的多处都要用到同一个泛型参数,即这些地方引用的泛型类型要保持同一个实际类型时,这时候就要采用泛型类型的方式进行定义,也就是类级别的泛型,语法格式如下:
 public class GenericDao<T> {
      private T field1;
      public void save(T obj){}
      public T getById(int id){}
 }

    2)类级别的泛型是根据引用该类名时指定的类型信息来参数化类型变量的,例如,如下两种方式都可以:
                             GenericDao<String> dao = null;
                              new genericDao<String>();

注意:1)在对泛型类型进行参数化时,类型参数的实例必须是引用类型,不能是基本类型。
            2)当一个变量被声明为泛型时,只能被实例变量、方法和内部类调用,而不能被静态变量和静态方法调用。因为静态成员是被所有参数化的类所共享的,所以静态成员不应该有类级别的类型参数。



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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值