极速时时彩电脑版_泛型的深入研究——面试时说出能加分

  • 时间:
  • 浏览:2
  • 来源:七月娱乐网 - 专注共享鑫迪资源网活动

    大伙儿 都可不可不可不都可以 在定义集合时设置泛型从前的约束,也都可不可不可不都可以 在定义类和最好的辦法 时加在泛型,从前能提升类和最好的辦法 的灵活性。此外大伙儿 还都可不可不可不都可以 在定义泛型时加在继承和通配符。在平时的培训中,我曾发现初学者对好多好多 繁杂的泛型(确实好多好多 繁杂,只不过是较少用)感到困惑。这里就通过好多好多 案例展示泛型在项目里的常见用法。    

1 泛型都可不可不可不都可以 作用在类和接口上

    泛型作用在类上的案例,比如在项目里,大伙儿 时要定义有一个 仓库类(WareHouse),会用有一个 列表来表示仓库里存放的东西。在定义仓库类时,大伙儿 都可不可不可不都可以 通过泛型来指定列表都可不可不可不都可以 容纳的数据类型。请看如下的GenericClass.java例子。    

1	//省略import集合包的代码
2	//请注意在定义类时,直接加在了泛型T
3	class WareHouse<T>{
4	  	private List<T> productList;//请注意这里的泛型是T,和第3行一致
5		 	public List<T> getProductList() 
6	    {return productList; }
7	  	public void setProductList(List<T> productList) 
8	    {this.productList = productList;}
9	     //构造函数
10	    	public WareHouse()
11		    {productList = new ArrayList<T>();}
12	     //加在元素的最好的辦法



,请注意参数类型是T
13			void addItem(T item)
14		    {productList.add(item);	}
15	     //打印所有的对象
16			public void printAllItems()  {
17	       //T作用到了迭代器上
18			Iterator<T> it = productList.iterator();
19			while(it.hasNext())
20			{System.out.println(it.next().toString());}
21		}	
22	}

    在第3行定义WareHourse例如类时,大伙儿 加在了泛型约束T,而在例如类的属性和最好的辦法 里,大伙儿 又多处用到了例如泛型T。比如,在第4行里通过T来创建有一个 含泛型约束的List,在第13行加在元素的最好的辦法 里,参数是T,在第16行打印所有对象的printAllItems最好的辦法 里,大伙儿 在第18行创建迭代器时,也用到了泛型T。

    大伙儿 也都可不可不可不都可以 把T修改成E等字符,但一旦定义成T,没人在使用时,就时要和例如字符”T”相匹配。    

23	class Item{
24	//货物名称 
25		private String itemName;
26	//构造函数
27		public Item(String name)
28		{this.itemName = name;}
29	//针对属性的get最好的辦法



500		public String getItemName() 
31	{return itemName;	}
32	//针对属性的set最好的辦法



33		public void setItemName(String itemName) 
34	{this.itemName = itemName;	}
35	//重写了toString最好的辦法



36		public String toString()
37		{return this.itemName; 	}
38	}

    日后大伙儿 在第23到38行定义了有一个 用于描述仓库货物的Item类,在其中的第25行,大伙儿 通过itemName例如属性来定义该货物的名字。    

39	public class GenericClass {
40		public static void main(String[] args) {
41	     //这里传入的泛型种类是String
42			WareHouse<String> wh = new WareHouse<String>();		
43			wh.addItem("Java");
44			wh.addItem("C#");
45			wh.printAllItems();//能输出Java和C#
46			//接下来大伙儿

创建有一个

Item对象
47			Item bookItem = new Item("Book");
48			Item carItem = new Item("Car");
49	     //这里的泛型种类是Item
500			WareHouse<Item> itemWh = new WareHouse<Item>();
51			itemWh.addItem(bookItem);
52			itemWh.addItem(carItem);
53			itemWh.printAllItems(); //输出是Book和Car
54		}
55	}

    大伙儿 在main函数里用到了例如带泛型的WareHouse类。在第42行里,大伙儿 实例化wh对象时,指定了该对象的泛型类型是String,也好多好多 说,在WareHouse例如类里,所有带“T”的地方都都可不可不可不都可以 用String来替代。比如private List<T> productList;都可不可不可不都可以 被替代成private List<String> productList。日后在第43和44行里,大伙儿 调用了addItem最好的辦法 加在对象,并在第45行通过了printAllItems最好的辦法 输出了存储在wh里的所有商品。

    在第500行里,大伙儿 指定了泛型类型是自定义的Item;如是,在第51和52行调用addItem最好的辦法 时,传入的参数就需好多好多 Item类型了。

    在例如例子中,大伙儿 把泛型作用到类上。没人,大伙儿 就都可不可不可不都可以 用比较灵活的最好的辦法 来定义类里的数据类型,从而例如类也有比较高的通用性。

    泛型也都可不可不可不都可以 作用到接口上,例如语法点和作用到类上的很例如,就不示例了。

    此外,在上述代码里,大伙儿 也见到了泛型作用到最好的辦法 上的基本用法,比如大伙儿 让泛型作用到类的返回类型上。代码如下:

    public List<T> getProductList()

    也都可不可不可不都可以 让泛型作用到最好的辦法 的参数类型上,代码如下:

    void addItem(T item)

2 泛型的继承和通配符

    在定义泛型时,大伙儿 都可不可不可不都可以 通过extends来限定泛型类型的上限,也都可不可不可不都可以 通过super来限定下限,例如个 限定字一般会和?等关键字搭配使用。

    比如有从前的代码List<? super Father> dest,这里,super带有 “高于”的意思,? Super Father就表示dest存放的对象应当“以Father为子类”;换句话说,在dest里,都可不可不可不都可以 存放任何子类是Father类的对象。

    再来看个extends的用法。比如有从前的代码,List<? extends Father> src,extends用来表示继承,这里的src都可不可不可不都可以 存放以”Father”为父类的对象;也好多好多 说,src都可不可不可不都可以 存放任何Father对象的子类。

    在实际的项目里,大伙儿 一般从List<? extends Father> src例如的集合里读元素,而从List<? super Father> dest从前的集合里写元素。通过下面的GenericExtends.java例子,再来了解extends,super和?的用法。    

1	import java.util.ArrayList;
2	import java.util.List;
3	//定义有一个

空的父类和空的子类
4	class Father{ }
5	class Son extends Father{}
6	//这是个带有

main最好的辦法



的主类
7	public class GenericExtends {
8	   //例如最好的辦法



里,将把src里的对象基因重组到dest里   
9	   static void copy(List<? super Father> dest, 
10	                    List<? extends Father> src) {  
11	        for (int i=0; i<src.size(); i++)
12	        {  dest.add(src.get(i));    }
13	    }

    在第9行copy最好的辦法 的有一个 参数里,大伙儿 看到了有一个 带有 extends和super泛型的参数。在最好的辦法 体的for循环里,大伙儿 的做法符合刚才讲到的原则:从带extends泛型的集合里读,往带super泛型的集合里写。    

14   public static void main(String[] args) {
15	     Father f = new Father();
16			Son s = new Son();
17	     //创建了有一个

带Father泛型的集合,并向其贴到

了有一个

元素
18			List<Father> srcFatherList = new ArrayList<Father>();
19			srcFatherList.add(f);		
20			List<Father> destFatherList = new ArrayList<Father>();
21	     //通过copy最好的辦法



,把元素基因重组进了destFatherList里
22			copy(destFatherList,srcFatherList);
23	     //这里的输出是1,说明copy最好的辦法



成功地往destFatherList里写了元素
24			System.out.println(destFatherList.size());		
25		}
26	}

    在定义最好的辦法 的参数时,大伙儿 都可不可不可不都可以 用带extends和super的泛型来确保输入参数类型的准确性。除此之外,例如种泛型的用处不大,比如在main函数的第22行里,调用copy最好的辦法 时,大伙儿 传入的参数也有List<Father>类型。

    下面大伙儿 来展示些错误的用法:

    错误用法一:用带问号的类型实例化集合对象。

1   List<?> list = new ArrayList<String>(); //正确

2   //List<?> list = new ArrayList<?>(); //错误

    第1行里,其确实等号的左边大伙儿 用到了问号,但在右边,大伙儿 确立了泛型类型是String,例如是正确的。与之相比,在等号的左边和右边大伙儿 都用了问号,这是错误的,好多好多 编译器我不在乎 list集合该采用哪种泛型类型。

    错误用法二:向带有 <? extends Father>泛型的集合里写。

1    List<? extends Father> list = new ArrayList<Father>();

2    //list.add(f); //error

    第2行会报语法错,愿因 是编译器我不在乎 例如基于Father的子类型究竟是哪几个;好多好多 没人选用,为了保证类型安全,好多好多 就不允许往里边加数据。”

    错误用法三:从带有 <? super Father>泛型的集合里读。

    1             List<? super Father> list1 = new ArrayList<Father>();

    2             list.add(f); //正确

    3             //list.get(0);//错误

    第3行会报语法错,愿因 是编译器我不在乎 该用哪种Father的父类来接收get的返回值;于是,同样为了保证类型安全,好多好多 就不允许读。

    从上述的第二和第例如错误的用法里,大伙儿 能感受到,extends和super例如种定义泛型的用法除了在定义最好的辦法 参数之外,还真没好多好多 共要的用途。