泛型:
把明确数据类型的工作,提前到了编译时期,在创建集合的时候明确存储元素的数据类型。
这样的做法有点向把数据类型当作参数一样传递,所以泛型还有一个叫法:参数化类型
泛型的语句定义格式:
<引用数据类型>
注意:尖括号中的数据类型只能是引用数据类型
泛型的好处:
1、将我们运行时期出现的问题,提前到编译时期
2、不需要做强制类型转换
3、优化了代码。消除了不必要的黄色警告线,使代码看起来更简洁
package com.shujia.wyh.day18;
import java.util.ArrayList;
import java.util.Iterator;
public class GenericDemo {
public static void main(String[] args) {
//创建一个List集合对象
//在JDK1.7之后,泛型会进行自动类型推断
ArrayList<String> list = new ArrayList<>();
//向集合中添加元素
list.add("hello");
list.add("hello");
list.add("hello");
list.add("hello");
list.add("hello");
//遍历
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()){
String next = iterator.next();
System.out.println(next);
}
}
}
泛型通配符<?>
任意类型,如果没有明确,那么就是Object以及任意的Java类了
? extends E
向下限定,E及其子类
? super E
向上限定,E及其父类
package com.shujia.wyh.day19;
import java.util.ArrayList;
class Animal {
}
class Cat extends Animal{
}
class Dog extends Animal{
}
public class GenericDemo1 {
public static void main(String[] args) {
ArrayList<Animal> list1 = new ArrayList<Animal>();
ArrayList<Dog> list2 = new ArrayList<Dog>();
ArrayList<Object> list3 = new ArrayList<Object>();
//泛型通配符<?>
//任意类型,如果没有明确,那么就是Object以及任意的Java类了
ArrayList<?> list4 = new ArrayList<Animal>();
ArrayList<?> list5 = new ArrayList<Dog>();
ArrayList<?> list6 = new ArrayList<Object>();
//? extends E 向下限定,E及其子类
ArrayList<? extends Animal> list7 = new ArrayList<Animal>();
ArrayList<? extends Animal> list8 = new ArrayList<Dog>();
ArrayList<? extends Animal> list9 = new ArrayList<Cat>();
//? super E 向上限定,E及其父类
ArrayList<? super Animal> list11 = new ArrayList<Animal>();
ArrayList<? super Animal> list13 = new ArrayList<Object>();
}
}
泛型类
把泛型定义在类上
格式:public class 类名<泛型类型1,…>
注意:泛型类型必须是引用类型
package com.shujia.wyh.day19;
class GenericTool1<T> {
private T obj;
public T getObj(){
return obj;
}
public void setObj(T obj){
this.obj = obj;
}
}
public class GenericTest1 {
public static void main(String[] args) {
GenericTool1<String> gt2 = new GenericTool1<>();
gt2.setObj("hello");
String obj = gt2.getObj();
}
}
泛型方法
把泛型定义在方法上
格式:public <泛型类型> 返回类型 方法名(泛型类型 .)
package com.shujia.wyh.day19;
public class GenericTool2 {
//用泛型方法,将来不确定要传入什么类型的数据
public <F> void show(F f){
System.out.println(f);
}
}
public class GenericTest2 {
public static void main(String[] args) {
GenericTool2 genericTool2 = new GenericTool2();
genericTool2.show("hello");
genericTool2.show(10);
genericTool2.show(12.34);
}
}
泛型接口
把泛型定义在接口上
格式:public interface 接口名<泛型类型1…>
package com.shujia.wyh.day19;
interface GenericTool3<W> {
public void show(W w);
}
class GenericTool3Impl<W> implements GenericTool3<W>{
@Override
public void show(W w) {
System.out.println(w);
}
}
public class GenericTest3 {
public static void main(String[] args) {
GenericTool3Impl<String> stringGenericTool3 = new GenericTool3Impl<>();
stringGenericTool3.show("hello");
stringGenericTool3.show(20);
}
}
增强for循环概述:简化数组和Collection集合的遍历
语句定义格式:
for(元素数据类型 变量名(自定义) : 数组或者Collection集合) {
使用变量即可,该变量就是元素
}
好处:简化遍历
package com.shujia.wyh.day19;
import java.util.ArrayList;
public class ForDemo {
public static void main(String[] args) {
ArrayList<String> strings = new ArrayList<>();
strings.add("hello");
strings.add("world");
strings.add("java");
strings.add("bigdata");
strings.add("hadoop");
for(String string : strings){
System.out.println(string);
}
}
}
静态导入:
语句定义格式:import static 包名...类名.方法名;
可以直接导入到方法级别
注意事项:
方法必须是静态的
可变参数概述
定义方法的时候不知道该定义多少个参数
格式
修饰符 返回值类型 方法名(数据类型… 变量名){}
注意:
这里的变量其实是一个数组
如果一个方法有可变参数,并且有多个参数,那么,可变参数肯定是最后一个
Arrays工具类中的一个方法
public static <T> List<T> asList(T... a)
package com.shujia.wyh.day19;
import java.util.Arrays;
import java.util.List;
public class ArgsDemo {
public static void main(String[] args) {
//使用方法求两个数之和
int a = 10;
int b = 20;
sum(a, b);
//使用方法求三个数之和
int c = 30;
sum(a, b, c);
//使用方法求四个数之和
int d = 40;
sum(a, b, c, d);
sum("小明", 90, 92, 93);
}
//当使用可变参数定义方法时候,有其他数据类型参与的时候,将可变参数的定义放在最后
public static void sum(String name, int... ints) {
int sum = 0;
for (int i : ints) {
sum += i;
}
System.out.println(name + "学生的总成绩为:" + sum);
}
//使用可变参数的形式定义加法的方法
//注意可变参数的变量最终是一个数组的变量名
//这里将来传入多个参数的时候,内部会自动将这多个参数形成一个数组,数组名就是我们定义的ints
public static void sum(int... ints) {
int sum = 0;
for (int i : ints) {
sum += i;
}
System.out.println(ints.length + "个数之和为:" + sum);
}
}
Set集合:元素是唯一的,并且元素的顺序是无序的集合
HashSet: 元素有序唯一 无序
对象类需要重写equals(),hashCode()方法
package com.shujia.wyh.day19;
import java.util.HashSet;
import java.util.Set;
import java.util.Objects;
class Student2 {
private String name;
private int age;
public Student2() {
}
public Student2(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Student2 student2 = (Student2) o;
return age == student2.age && Objects.equals(name, student2.name);
}
@Override
public int hashCode() {
return Objects.hash(name, age);
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
public class SetDemo2 {
public static void main(String[] args) {
//创建集合对象
Set<Student2> student2s = new HashSet<>();
//创建学生对象
Student2 s1 = new Student2("小明", 18);
Student2 s2 = new Student2("小王", 17);
Student2 s3 = new Student2("小周", 19);
Student2 s4 = new Student2("小明", 18);
//将学生对象添加到集合中
student2s.add(s1);
student2s.add(s2);
student2s.add(s3);
student2s.add(s4);
//遍历
for (Student2 student2 : student2s) {
System.out.println(student2);
}
}
}
LinkedHashSet(父类HashSet):
1、底层数据结构是哈希表和链表
2、哈希表保证元素的唯一
3、链表保证了元素的有序(存储和取出的顺序一致)
4、线程不安全,效率高
TreeSet:元素唯一,元素可以按照某种规则进行排序
两种排序方式:
自然排序 对象类需要实现Comparable接口并重写compareTo()方法 (无参构造方法)
比较器排序 对象类需要实现Comparable接口并重写compare()方法(有参构造方法所以可匿名内部类实现)
自然排序
package com.shujia.wyh.day19;
import java.util.TreeSet;
import java.util.Objects;
class Student3 implements Comparable<Student3>{
private String name;
private int age;
public Student3() {
}
public Student3(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
@Override
public int compareTo(Student3 o) {
//年龄一样,姓名不一定一样
int i = this.age - o.age;
//判断姓名是否一样(三目运算符)
int i2 = i == 0 ? this.name.compareTo(o.name) : i;
return i2;
}
}
class Student4 implements Comparable<Student4> {
private String name;
private int age;
public Student4() {
}
public Student4(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
@Override
public int compareTo(Student4 o) {
//主要条件:姓名的长度排序
int i = this.name.length() - o.name.length();
//长度一样,姓名内容不一定一样
int i2 = i == 0 ? this.name.compareTo(o.name) : i;
//姓名内容一样,年龄的大小不一定一样
int i3 = i2 == 0 ? this.age - o.age : i2;
return i3;
}
}
public class TreeSetDemo2 {
public static void main(String[] args) {
//创建TreeSet集合
TreeSet<Student3> set = new TreeSet<>();
//创建学生对象
Student3 s1 = new Student3("theshy", 20);
Student3 s2 = new Student3("xiaohu", 24);
Student3 s3 = new Student3("uzi", 25);
Student3 s4 = new Student3("卡萨", 22);
Student3 s5 = new Student3("rookie", 23);
Student3 s6 = new Student3("姿态", 21);
Student3 s7 = new Student3("faker", 25);
Student3 s8 = new Student3("xiaohu", 24);
//将学生对象添加到集合中
set.add(s1);
set.add(s2);
set.add(s3);
set.add(s4);
set.add(s5);
set.add(s6);
set.add(s7);
set.add(s8);
for(Student3 student3 : set){
System.out.println(student3);
}
for(Student4 student4 : set){
System.out.println(student4);
}
}
}
比较器排序
利用TreeSet创建对象时的带参数的构造方法来进行比较器排序
TreeSet(Comparator<? super E> comparator)
构造一个新的,空的树集,根据指定的比较器进行排序
package com.shujia.wyh.day19;
import java.util.Comparator;
import java.util.TreeSet;
class Student5 {
private String name;
private int age;
public Student5() {
}
public Student5(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
public class TreeSetDemo4 {
public static void main(String[] args) {
//匿名内部类实现Comparator接口
TreeSet<Student5> set = new TreeSet<>(new Comparator<Student5>() {
@Override
public int compare(Student5 o1, Student5 o2) {
//主要条件:姓名的长度排序
int i = o1.getName().length() - o2.getName().length();
//长度一样,姓名内容不一定一样
int i2 = i == 0 ? o1.getName().compareTo(o2.getName()) : i;
//姓名内容一样,年龄的大小不一定一样
int i3 = i2 == 0 ? o1.getAge() - o2.getAge() : i2;
return i3;
}
});
//创建学生对象
Student5 s1 = new Student5("mingwang", 18);
Student5 s2 = new Student5("wangyu", 19);
Student5 s3 = new Student5("zhoujiaxiang", 17);
Student5 s4 = new Student5("zhangbaogui", 18);
Student5 s5 = new Student5("liuzhicheng", 18);
Student5 s6 = new Student5("wangyu", 20);
//将学生对象添加到集合中
set.add(s1);
set.add(s2);
set.add(s3);
set.add(s4);
set.add(s5);
set.add(s6);
for(Student5 Student5 : set){
System.out.println(Student5);
}
}
}