how2j.cn

本视频是解读性视频,所以希望您已经看过了本知识点的内容,并且编写了相应的代码之后,带着疑问来观看,这样收获才多。 不建议一开始就观看视频



11分34秒
本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)



示例 1 : ? extends   
示例 2 : ? super   
示例 3 : 泛型通配符?   
示例 4 : 总结   
示例 5 : 练习- extends   
示例 6 : 答案- extends   
示例 7 : 练习-二叉树   
示例 8 : 答案-二叉树   

ArrayList heroList<? extends Hero> 表示这是一个Hero泛型或者其子类泛型
heroList 的泛型可能是Hero
heroList 的泛型可能是APHero
heroList 的泛型可能是ADHero
所以 可以确凿的是,从heroList取出来的对象,一定是可以转型成Hero的

但是,不能往里面放东西,因为
放APHero就不满足<ADHero>
放ADHero又不满足<APHero>
? extends
package generic; import java.util.ArrayList; import charactor.ADHero; import charactor.APHero; import charactor.Hero; public class TestGeneric { public static void main(String[] args) { ArrayList<APHero> apHeroList = new ArrayList<APHero>(); apHeroList.add(new APHero()); ArrayList<? extends Hero> heroList = apHeroList; //? extends Hero 表示这是一个Hero泛型的子类泛型 //heroList 的泛型可以是Hero //heroList 的泛型可以使APHero //heroList 的泛型可以使ADHero //可以确凿的是,从heroList取出来的对象,一定是可以转型成Hero的 Hero h= heroList.get(0); //但是,不能往里面放东西 heroList.add(new ADHero()); //编译错误,因为heroList的泛型 有可能是APHero } }
ArrayList heroList<? super Hero> 表示这是一个Hero泛型或者其父类泛型
heroList的泛型可能是Hero
heroList的泛型可能是Object

可以往里面插入Hero以及Hero的子类
但是取出来有风险,因为不确定取出来是Hero还是Object
? super
package generic; import java.util.ArrayList; import charactor.ADHero; import charactor.APHero; import charactor.Hero; public class TestGeneric { public static void main(String[] args) { ArrayList<? super Hero> heroList = new ArrayList<Object>(); //? super Hero 表示 heroList的泛型是Hero或者其父类泛型 //heroList 的泛型可以是Hero //heroList 的泛型可以是Object //所以就可以插入Hero heroList.add(new Hero()); //也可以插入Hero的子类 heroList.add(new APHero()); heroList.add(new ADHero()); //但是,不能从里面取数据出来,因为其泛型可能是Object,而Object是强转Hero会失败 Hero h= heroList.get(0); } }
泛型通配符? 代表任意泛型
既然?代表任意泛型,那么换句话说,这个容器什么泛型都有可能

所以只能以Object的形式取出来
并且不能往里面放对象,因为不知道到底是一个什么泛型的容器
泛型通配符?
package generic; import java.util.ArrayList; import property.Item; import charactor.APHero; import charactor.Hero; public class TestGeneric { public static void main(String[] args) { ArrayList<APHero> apHeroList = new ArrayList<APHero>(); //?泛型通配符,表示任意泛型 ArrayList<?> generalList = apHeroList; //?的缺陷1: 既然?代表任意泛型,那么换句话说,你就不知道这个容器里面是什么类型 //所以只能以Object的形式取出来 Object o = generalList.get(0); //?的缺陷2: 既然?代表任意泛型,那么既有可能是Hero,也有可能是Item //所以,放哪种对象进去,都有风险,结果就什么什么类型的对象,都不能放进去 generalList.add(new Item()); //编译错误 因为?代表任意泛型,很有可能不是Item generalList.add(new Hero()); //编译错误 因为?代表任意泛型,很有可能不是Hero generalList.add(new APHero()); //编译错误 因为?代表任意泛型,很有可能不是APHero } }
如果希望只取出,不插入,就使用? extends Hero
如果希望只插入,不取出,就使用? super Hero
如果希望,又能插入,又能取出,就不要用通配符?
示例 5 :

练习- extends

edit Or  姿势不对,事倍功半! 点击查看做练习的正确姿势
如代码所示,为了遍历不同泛型的3种集合,需要设计3个方法

借助? extends, 把代码减肥到只是用一种方法
package generic; import java.util.ArrayList; import charactor.ADHero; import charactor.APHero; import charactor.Hero; public class TestGeneric { public static void iterate(ArrayList<Hero> list) { for (Hero hero : list) { System.out.println(hero.name); } } public static void iterateAP(ArrayList<APHero> list) { for (Hero hero : list) { System.out.println(hero.name); } } public static void iterateAD(ArrayList<ADHero> list) { for (Hero hero : list) { System.out.println(hero.name); } } public static void main(String[] args) { ArrayList<Hero> hs = new ArrayList<>(); ArrayList<APHero> aphs = new ArrayList<>(); ArrayList<ADHero> adhs = new ArrayList<>(); iterate(hs); iterateAP(aphs); iterateAD(adhs); } }
在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
查看本答案会花费4个积分,您目前总共有点积分。查看相同答案不会花费额外积分。 积分增加办法 或者一次性购买JAVA 中级总计0个答案 (总共需要0积分)
查看本答案会花费4个积分,您目前总共有点积分。查看相同答案不会花费额外积分。 积分增加办法 或者一次性购买JAVA 中级总计0个答案 (总共需要0积分)
账号未激活 账号未激活,功能受限。 请点击激活
本视频是解读性视频,所以希望您已经看过了本答案的内容,带着疑问来观看,这样收获才多。 不建议一开始就观看视频

3分3秒 本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)


使用一个方法:

public static void iterate(ArrayList<? extends Hero> list) {

就表示传递进去的集合的泛型可以是Hero的子类
package generic; import java.util.ArrayList; import charactor.ADHero; import charactor.APHero; import charactor.Hero; public class TestGeneric { public static void iterate(ArrayList<? extends Hero> list) { for (Hero hero : list) { System.out.println(hero.name); } } public static void main(String[] args) { ArrayList<Hero> hs = new ArrayList<>(); ArrayList<APHero> aphs = new ArrayList<>(); ArrayList<ADHero> adhs = new ArrayList<>(); iterate(hs); iterate(aphs); iterate(adhs); } }
示例 7 :

练习-二叉树

edit Or  姿势不对,事倍功半! 点击查看做练习的正确姿势
练习-支持泛型的二叉树改造成 支持泛型 <T extends Comparable>,并在比较的时候使用compare方法
在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
查看本答案会花费4个积分,您目前总共有点积分。查看相同答案不会花费额外积分。 积分增加办法 或者一次性购买JAVA 中级总计0个答案 (总共需要0积分)
查看本答案会花费4个积分,您目前总共有点积分。查看相同答案不会花费额外积分。 积分增加办法 或者一次性购买JAVA 中级总计0个答案 (总共需要0积分)
账号未激活 账号未激活,功能受限。 请点击激活
本视频是解读性视频,所以希望您已经看过了本答案的内容,带着疑问来观看,这样收获才多。 不建议一开始就观看视频

4分57秒 本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)


把Node设计为

public class Node<T extends Comparable<T>> {

就表示只能使用那些实现了Comparable接口的泛型,比如Integer,Hero

Node<Integer> roots = new Node<>();
Node<Hero> heros = new Node<>();

但是GiantDragon就不行,因为它没有实现Comparable接口

Node<GiantDragon> dragons = new Node<>();
package collection; import java.util.ArrayList; import java.util.List; import charactor.GiantDragon; import charactor.Hero; public class Node<T extends Comparable<T>> { public Node<T> leftNode; public Node<T> rightNode; public T value; public void add(T t) { if (null == value) value = t; else { //t和value都是T 类型,而T类型extends Comparable,所以必然提供compare接口 if (t.compareTo(value) <= 0) { if (null == leftNode) leftNode = new Node<T>(); leftNode.add(t); } else { if (null == rightNode) rightNode = new Node<T>(); rightNode.add(t); } } } public List<T> values() { List<T> values = new ArrayList<>(); if (null != leftNode) values.addAll(leftNode.values()); values.add(value); if (null != rightNode) values.addAll(rightNode.values()); return values; } public static void main(String[] args) { int randoms[] = new int[] { 67, 7, 30, 73, 10, 0, 78, 81, 10, 74 }; Node<Integer> roots = new Node<>(); for (int number : randoms) { roots.add(number); } System.out.println(roots.values()); //Hero实现了 Comparable接口,所以可以作为Node的泛型 Node<Hero> heros = new Node<>(); //GiantDragon 没有实现 Comparable接口,所以不能作为Node的泛型 Node<GiantDragon> dragons = new Node<>(); } }


HOW2J公众号,关注后实时获知布最新的教程和优惠活动,谢谢。


问答区域    
2019-05-15 作业一
NoColor



提交
	public static void iii(ArrayList<? extends Hero> list) {
		for(Hero hero : list) {
			System.out.println(hero.name);
		}
	}
	

							





回答已经提交成功,正在审核。 请于 我的回答 处查看回答记录,谢谢
答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到





2019-04-22 交作业(一),欢迎暂时没思路的朋友来交流参考
simple_曹泽



随时欢迎交流、指正,感谢! /* * 需求:构建一个<T extends Comparable>的二叉树 * 注意:printf打印浮点数时,指定小数位数的同时必须得指定所有位数7.2f */ 个人学习博客:https://blog.csdn.net/hui1setouxiang 欢迎交流指正,内容是根据版主推荐的《Core Java》写的一些关键章节(内部类、lambda表达式、String类、集合、IO流等)的总结。 祝您生活顺心,学习愉快,共勉~
package generic;

import java.util.ArrayList;
import java.util.List;
import java.util.Random;
/*
 * 	需求:构建一个<T extends Comparable>的二叉树
 * 	注意:printf打印浮点数时,指定小数位数的同时必须得指定所有位数7.2f
 */
public class GenericBinaryTreeDemo {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Node<Employee> emps = new Node<>(); 
		for (int i = 1000 ; i < 1030 ; i++) {
			String name = i + "号" + "雇员" ;
			double salary = Math.random() * 30000;
			
			emps.addNode(new Employee(name, salary));
		}
		
		List<Employee> sortEmps = emps.values();
		for(Employee emp : sortEmps)
			System.out.printf("%s的薪水为%7.2f%n",emp.getName(),emp.getSalary());
	}

}

class Employee implements Comparable{
	private String name;
	private double salary;
	
	public Employee(String name, double salary) {
		super();
		this.name = name;
		this.salary = salary;
	}

	@Override
	public int compareTo(Object o) {
		// TODO Auto-generated method stub
		Employee e = (Employee) o;
		return Double.compare(salary, e.getSalary());
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public double getSalary() {
		return salary;
	}

	public void setSalary(double salary) {
		this.salary = salary;
	}
	
}
class Node<T extends Comparable> {
	private T value;
	private Node<T> leftNode;
	private Node<T> rightNode;	
	
	public List<T> values() {
		List<T> lists = new ArrayList<>();
		
		if (null != leftNode)
			lists.addAll(leftNode.values());
		
		lists.add(value);
		
		if (null != rightNode)
			lists.addAll(rightNode.values());
		
		return lists;
	}
	
	public void addNode(T addValue) {
		if (null == value)
			value = addValue;
		else {
			if (addValue.compareTo(value) < 0) {
				if (null == leftNode)
					leftNode = new Node<T>();
				leftNode.addNode(addValue);
			} else {
				if (null == rightNode)
					rightNode = new Node<T>();
				rightNode.addNode(addValue);
			}
		}
	}
}

							





回答已经提交成功,正在审核。 请于 我的回答 处查看回答记录,谢谢
答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到





2019-04-16 <Hero>也能使用Hero类和子类那和<? extends Hero>有什么区别呢?
2019-03-26 交作业
2019-01-25 为什么<? super Hero>取出来可能是Object?
2019-01-18 这个为什么会报错
2019-01-09 交作业!
2019-01-01 二叉树Hero放入2个以上就出错
2018-09-17 泛型通配符存在的意义
2018-04-19 实例6,会不会答案那种方式不对?还是我哪里有问题?新人渴求解答?
2018-04-13 请问在示例 1 : 如果我只定义了APHero 和Hero 并且APHero extends Hero
2017-11-30 那泛型的通配符?跟T有什么区别么?
2017-11-19 关于public class Node<E extends Comparable<? super E>>的<? super E>疑惑
2017-08-27 使用? extend之后的遍历为什么都是父类Hero的name
2017-08-08 注释不懂
2017-06-14 关于 <? super E>
2017-04-19 为什么不行啊,版主帮忙看看,郁闷了




提问之前请登陆
提问已经提交成功,正在审核。 请于 我的提问 处查看提问记录,谢谢
关于 JAVA 中级-泛型-通配符 的提问

尽量提供截图代码异常信息,有助于分析和解决问题。 也可进本站QQ群交流: 1001964108
提问尽量提供完整的代码,环境描述,越是有利于问题的重现,您的问题越能更快得到解答。
对教程中代码有疑问,请提供是哪个步骤,哪一行有疑问,这样便于快速定位问题,提高问题得到解答的速度
在已经存在的几千个提问里,有相当大的比例,是因为使用了和站长不同版本的开发环境导致的,比如 jdk, eclpise, idea, mysql,tomcat 等等软件的版本不一致。
请使用和站长一样的版本,可以节约自己大量的学习时间。 站长把教学中用的软件版本整理了,都统一放在了这里, 方便大家下载: http://how2j.cn/k/helloworld/helloworld-version/1718.html

上传截图