### 概述
泛型:泛泛的类型.就是不确定的类型.
集合中是可以存放任意对象的,只要把对象存储集合后,那么这时它们都会被提升为object类型. 当我们在取出每一个对象,并且进行相应的操作, 这时必须采用类型转换 . 由于集合什么类型的元素都可以存储 .导致取出时 ,如果出现强转就会引发运行时classCastException .怎么来解决这个问题呢 ? 使用集合时,必须明确集合中元素的类型. 这种方式称为"泛型" .
为了避免安全隐患, 1.5出现新的安全机制,保证程序的安全性 , 这就是泛型 .
~~~
ArrayList<String> arr = new ArrayList<String>(); //中括号的内容就是指定泛型
~~~
## 泛型中的好处
1. 将运行时期的ClassCastException,转移到了编译时期的编译失败.
2. 避免了类型强制转换的麻烦.
~~~
public static void main(String[] args)
{
dome1();
dome2();
}
//不使用泛型
public static void dome1()
{
ArrayList names = new ArrayList();
names.add("钢铁侠");
names.add(123);
names.add(true);
for (Object o : names) {
System.out.println(((String) o).length()); //下面没法操作了
}
}
//使用泛型
public static void dome2()
{
ArrayList<String> names = new ArrayList();
names.add("超人"); //如果传入不是String类型就报错了
names.add("钢铁侠");
names.add("蝙蝠侠");
for (String s : names) {
System.out.println(s.length());
}
}
~~~
## 泛型中E的含义
就是储存类型的变量,等待接收一种数据类型.
![](https://box.kancloud.cn/5ca24ed57e6a094aa354a1f3b9fc5ab3_842x405.png)
## 泛型类
什么时候确定使用的类型呢?当创建该类对象的时候就可以确定了.
### 伪泛型
java中的泛型是伪泛型 , 是一种编译手段 . 如下面这段代码, 如果add的是String类型则编译成功,如果不是String类型则编译失败 . 编译成功后在class文件中是不存在泛型的 . 也就是说泛型只是在编译时有泛型 , 编译成功后是不存在 .
~~~
ArrayList<String> arr = new ArrayList<String>();
arr.add("")
~~~
### 使用泛型的好处
1. 将运行时期的ClassCastException转移到了编译期,变成了编译失败 .
2. 避免了类型强转的麻烦 .
~~~
ArrayList<String> arr = new ArrayList<String>();
arr.add("123"); //当集合明确类型之后,存放不一致的类型就会编译失败
arr.add("456"); //集合已经明确存放的元素类型,那么在使用迭代器的时候,迭代器也同样会知道具体遍历元素类型
arr.add("789");
Iterator<String> it = arr.iterator(); //当使用Iterator<String>控制元素类型后,就不需要强转了.获取到的元素直接就是String类型
while(it.hasNext())
{
System.out.println(it.next());
}
~~~
### 泛型方法
~~~
public class GenericityDome<E>
{
E name;
public E getName()
{
return name;
}
public void setName(E n) //使用了类的泛型方法
{
this.name = n;
}
public <T> void show(T t) //泛型方法
{
System.out.println(t);
}
}
~~~
调用
~~~
GenericityDome<String> g = new GenericityDome<>();
g.show(1); //调用方法传递参数的时候才能确定类型.这个就是泛型方法
~~~
### 带有泛型接口
泛型接口上的泛型到底什么时候确定?
1. 当子类实现接口的时候确定接口上的泛型.
2. 当子类实现接口的时候,还是不确定泛型,把接口的泛型继续实现下来.
~~~
public interface GenericityDome<E>
{
public abstract void show(E e);
}
//子类实现接口的时候确定
class MyClass1 implements GenericityDome<String>
{
public void show(String s)
{
}
}
//当实现类创建对象的时候确定泛型
class MyClass2<Q> implements GenericityDome<Q>
{
public void show(Q s)
{
}
}
~~~
### 泛型通配符
? 代表任意类型
~~~
?
~~~
优点 : 各种类型的泛型都可以迭代 .
缺点 : 类型已经不确定了 .
所以这个方法只适合做遍历 .
~~~
public class Test {
public static void main(String[] args)
{
ArrayList<Integer> arr = new ArrayList<Integer>();
arr.add(123);
arr.add(456);
HashSet<String> set = new HashSet<String>();
set.add("123");
set.add("456");
it(set);
}
public static void it(Collection<?> collection)
{
Iterator<?> coll = collection.iterator();
while(coll.hasNext()){
System.out.println(coll.next());
}
}
}
~~~
## 泛型通配符注意
泛型通配符是没有意义单独的,一般是组合使用.
~~~
<? extends Animal> //泛型必须是Animal或者Animal的子类
~~~
~~~
<? super Animal> //泛型必须是Animal或者Animal的父类
~~~
~~~
public static void main(String[] args)
{
ArrayList<String> names1 = new ArrayList<>();
names1.add("111");
names1.add("222");
names1.add("333");
ArrayList<String> names2 = new ArrayList<>();
names2.add("aaa");
names2.add("bbb");
names2.add("ccc");
names1.addAll(names2);
ArrayList<Integer> names3 = new ArrayList<>();
names3.add(111);
names3.add(222);
names3.add(333);
names1.addAll(names3); //这里是加不进去的,可以查看addAll()的源码,它使用泛型通配符控制住了添加进去的类型.
}
~~~
- 环境
- Java基础
- 数据类型
- 基本类型
- 引用类型
- Array
- Class
- ArrayList
- 流程控制
- ASCII码表
- 面向对象
- 概述
- 变量
- this
- 修饰符
- 封装
- 继承
- final
- static
- 接口
- 接口
- 抽象类
- overload
- 接口和抽象类区别
- 案例
- 多态
- 构造方法
- super
- 匿名对象
- 内部类
- 静态导入
- 包的声明和访问
- 代码块
- 代理设计模式
- 静态代码块
- Object类
- lambda表达式
- 枚举
- 定义枚举
- 常用方法
- 枚举常量
- 异常
- 可变参数
- Properties
- 概述
- Properties和持久化有关的方法
- 反射
- 类加载器
- 概述
- Class类
- 获取构造方法并运行
- 获取成员变量并改值
- 获取成员方法并运行
- 反射泛型擦除
- 反射通过配置文件运行
- 常用 API
- Date类
- DateFormat类
- Calendar类
- Math类
- String类
- object类
- System类
- 正则表达式
- BigInteger类
- BigDecimal类
- Arrays工具类
- StringBuffer类
- 基本类型包装类
- 自动装箱和自动拆箱
- 框架集合
- collection接口
- 概述
- 接口方法
- 迭代器
- 迭代器并发异常
- 增强for循环
- 泛型
- hashCode和equals
- List接口
- 概述
- ArrayList
- LinkedList
- vector
- Set接口
- 概述
- 对象的哈希值
- HashSet
- LinkedHashSet
- 保证自定义元素的唯一性(重写)
- ArrayList和HashSet的contains原理
- Map接口
- 概述
- Map集合常用方法
- 迭代Map
- Map存储自定义类型
- HashMap
- LinkedHashMap
- Hashtable
- 嵌套Map遍历
- 方法的可变参数
- collections工具类
- 集合嵌套
- 集合特点总结
- Collection集合工具类
- Arrays工具类
- File类
- 概述
- 获取方法
- 创建和删除
- 判断方法
- 其他方法
- 文件过滤器
- IO流
- 字节流
- 概述
- 字节流
- 字节流复制文件
- 字节缓冲流
- 复制单级文件夹
- 使用字节流读取中文的乱码问题
- 字符编码表
- 方便程序员的IO流
- 转换流
- 概述
- OutputStreamWriter类
- InputStreamWriter类
- 转换流和其他子类的区别
- 序列化流
- 概述
- 序列化接口以及版本号问题
- transient
- 打印流
- 多线程
- 概述
- main线程介绍
- 开启线程方式一:Thread类
- 开启线程方式二:Runnable类
- 线程的执行原理
- 使用哪种方式好
- 使用匿名内部类创建线程
- 线程安全
- ThreadLocal
- 网络编程
- XML
- 概述
- 语法
- 元素标签
- 属性
- 转义字符
- CDATA区
- XML约束
- DTD约束
- 三种引入DTD的方式
- DTD语法
- Schema约束
- JDBC
- 概述
- JDBC原理
- JDBC查询操作
- JDBCUtils
- 预处理对象
- JDBC连接池
- 概述
- DBCP
- C3P0
- DBUtils操作
- ArrayHandler
- ArrayListHandler
- BeanHandler
- BeanListHandler
- ColumnListHandler
- MapHandler
- MapListHandler
- ScalarHandler
- 事务
- 事务概述
- 原生JDBC完成转账
- DBUtils完成转账
- 事务特性
- 三层思想
- 概述
- JavaWEB
- tomcat
- servlet
- 概述
- 映射servlet
- Servlet接口
- HttpServlet类
- load-on-startup
- servletContext
- ServletRequest
- ServletResponse
- Cooike&Session
- servlet三大域对象
- JSP
- 概述
- jsp内置对象
- 指令
- page
- include
- taglib
- JSP中的四个域对象
- jsp的一个动作标签
- EL表达式
- 概述
- 获取域中数据
- 执行运算
- el表达式中的内置对象
- jsti标签
- 概念
- jstl分类
- Filter过滤器
- 概述
- 生命周期
- url-pattern配置
- filter的chain过滤链
- filter-mapping子标签
- 全局统一错误页面
- filterConfig
- 自动登录实例
- 全局统一编码
- Listener监听器
- 概念
- 监听器分类
- 监听三个对象的创建和销毁
- 监听三个对象属性的变化
- 监听javabean在session中的状态变化
- 定时器
- 概念
- calendar日历
- 邮件
- 概述
- JSON
- 数据结构
- 堆栈结构
- 队列结构
- 数组结构
- 链表结构
- 基础加强
- 面试题:增强一个对象的方法有几种方式?
- 装饰者增强
- 动态代理
- 动态代理
- 动态代理细节
- 使用动态代理解决字符集编码
- 注解
- 概述
- 扩展的自定义注解
- 类加载器
- 全盘负责托管机制
- 开发模式
- 问题
- lambda
- 函数式编程思想概述
- String字符串