Set : 无序,不可以元素重复。
|--HashSet:数据结构是哈希表,线程是非同步的
保证元素唯一性的原理:判断元素的hashCode值是否相同,如果相同,还会继续判断元素的equals方法,是否为true。
|--TreeSet:可以对Set集合中的元素进行排序。
底层数据结构是二叉树。
保证元素唯一性的依据:
compareTo方法 return 0;
TreeSet排序的第一种方式,让元素自身具备比较性,元素需要实现Comparable接口,覆盖compareTo方法,这种方式也称为元素的自然顺序,或者叫做默认 顺序。
TreeSet排序的第二种方式,当元素自身不具备比较性时,或者具备的比较性不是所需要的。这时就需要集合自身具备比较性。在集合初始化时就有了比较方式
import java.util.*; class day15 { public static void main(String[] args) { TreeSet ts = new TreeSet(); ts.add("cba"); ts.add("abcd"); ts.add("aaa"); ts.add("bca"); Iterator it = ts.iterator(); while(it.hasNext()) { sop(it.next()); } } public static void sop(Object obj) { System.out.println(obj); } }
输出:
aaa
abcd
bca
cba
存储自定义对象
需求:在TreeSet集合中存储自定义对象Student。按照学生的年龄进行排序。
记住:排序时,当主要条件相同时,一定要判断一下次要条件。
import java.util.*; class Student implements Comparable//该接口强制让学生具备比较性 { private String name; private int age; Student(String name, int age) { this.name = name; this.age = age; } public int compareTo(Object obj) { if(!(obj instanceof Student)) throw new RuntimeException("不是学生对象"); Student s = (Student)obj; //System.out.println(this.name + "...compareto..."+s.name); if(this.age > s.age) return 1; if(this.age == s.age) { return this.name.compareTo(s.name); } return -1; } public String getName() { return name; } public int getAge() { return age; } } class day15 { public static void main(String[] args) { TreeSet ts = new TreeSet(); ts.add(new Student("lisi02", 22)); ts.add(new Student("lisi01", 24)); ts.add(new Student("lisi09", 52)); ts.add(new Student("lisi08", 52)); ts.add(new Student("lisi07", 30)); Iterator it = ts.iterator(); while(it.hasNext()) { Student stu = (Student)it.next(); sop(stu.getName()+"...."+stu.getAge()); } } public static void sop(Object obj) { System.out.println(obj); } }
输出:
lisi02....22
lisi01....24
lisi07....30
lisi08....52
lisi09....52
TreeSet内部数据结构
当元素自身不具备比较性,或者具备的比较性不是所需要的,这时需要让容器自身具备比较性。就定义一个比较器,将比较器对象作为参数传递给TreeSet集合的构造函数。
当两种排序都存在时,以比较器为主。
定义一个类,实现Comparator接口,覆盖compare方法.以return 0判断元素是否相同。
//按照学生的姓名从小到大排序,姓名相同,按照年龄从小到大排序 class MyCompare implements Comparator { public int compare(Object o1, Object o2) { Student s1 = (Student)o1; Student s2 = (Student)o2; int num = s1.getName().compareTo(s2.getName()); if(num == 0) { //return new Integer(s1.getAge()).compareTo(new Integer(s2.getAge())); return s1.getAge() - s2.getAge(); } return num; } } //主函数中:TreeSet ts = new TreeSet(new MyCompare());
练习:按照字符串长度排序。
字符串本身具备比较性,但是它的比较方式不是所需要的,这时就只能使用比较器。
import java.util.*; class StrLenComparator implements Comparator { public int compare(Object o1, Object o2) { String s1 = (String)o1; String s2 = (String)o2; int num = new Integer(s1.length()).compareTo(new Integer(s2.length())); if(num == 0) return s1.compareTo(s2); return num; } } class day15 { public static void main(String[] args) { TreeSet ts = new TreeSet(new StrLenComparator()); ts.add("cba"); ts.add("abc"); ts.add("aadfdfdfa"); ts.add("bcddfa"); Iterator it = ts.iterator(); while(it.hasNext()) { sop(it.next()); } } public static void sop(Object obj) { System.out.println(obj); } }
输出:
abc
cba
bcddfa
aadfdfdfa
泛型:
JDK 1.5 版本以后出现的新特性,用于解决安全问题,是一个类型安全机制。
好处1:将运行时期出现的问题转移到了编译时期,方便于程序员解决问题,让运行时期问题减少,安全。
好处2:避免了强制转换的麻烦。
泛型格式:通过<>来定义要操作的数据类型。
在使用java提供的对象时,什么时候写泛型呢?
通常在集合框架中很常见,只要见到<>,就要定义泛型。
其实<>就是用来接收类型的。
当使用集合时,将集合众要存储的数据类型作为参数传递到<>中即可。
import java.util.*; class day15 { public static void main(String[] args) { TreeSet<String> ts = new TreeSet<String>(); ts.add("abc01"); ts.add("abc02"); ts.add("abc034"); Iterator<String> it = ts.iterator(); while(it.hasNext()) { String s = it.next(); System.out.println(s+" : "+s.length()); } } public static void sop(Object obj) { System.out.println(obj); } } class LenComparator implements Comparator<String> { public int compare(String o1, String o2) { int num = new Integer(o1.length()).compareTo(new Integer(o2.length())); //要从大到小排序的话, 把o1 和o2换一下位置就可以了 if(num == 0) return o1.compareTo(o2); return num; } }
注意在实现Comparable 接口时重写equals方法时,参数必须为Object类型,因为重写的是Object方法,否则就不是重写了。
泛型类:
什么时候定义泛型类?
当类中要操作的引用数据类型不确定的时候,早期定义Object来完成扩展,现在定义泛型来完成扩展。
import java.util.*; class Worker { } class Student { } //泛型前做法 class Tool { private Object obj; public void setObject(Object obj) { this.obj = obj; } public Object getObject() { return obj; } } //泛型后做法,泛型类 class Utils<QQ> { private QQ q; public void setObject(QQ q) { this.q = q; } public QQ getObject() { return q; } } class day15 { public static void main(String[] args) { Utils<Worker> u = new Utils<Worker>(); u.setObject(new Worker()); Worker w = u.getObject(); } public static void sop(Object obj) { System.out.println(obj); } }
泛型方法:
泛型类定义的泛型,在整个类中有效,如果被方法使用,那么泛型类的对象明确要操作的具体类型后,所有要操作的类型就已经固定了。为了让不同方法可以操作不同类型,而且类型还不确定,那么可以将泛型定义在方法上。
import java.util.*; /* class Demo<T> { public void show(T t) { sop("show:" + t); } public void print(T t) { sop("print:" + t); } } */ class Demo { public <T> void show(T t) { System.out.println("show:" + t); } public <T> void print(T t) { System.out.println("print:" + t); } } class day15 { public static void main(String[] args) { Demo d = new Demo(); d.show("haha"); d.show(new Integer(4)); d.print("heihei"); } }
输出:
show:haha
show:4
print:heihei
特殊之处:静态方法不可以访问类上定义的泛型。如果静态方法操作的应用类型不确定,可以将泛型定义在方法上。
可以这样:
class Demo <T> { public void show(T t) { System.out.println("show:" + t); } public <Q> void print(Q q) { System.out.println("print:" + q); } public static <W> void method(W t)//泛型<W>要放在返回类型前面,修饰符后面 { System.out.println("method:"+t); } }
泛型定义在接口上。
import java.util.*; interface Inter<T> { void show(T t); } /* class InterImpl implements Inter<String> { public void show(String t) { System.out.println("show : "+t); } } */ class InterImpl <T> implements Inter<T> { public void show(T t) { System.out.println("show : "+t); } } class day15 { public static void main(String[] args) { //InterImpl i = new InterImpl(); //i.show("haha"); InterImpl<Integer> i = new InterImpl<Integer>(); i.show(4); } }
泛型限定。
?表示通配符,也可以理解为占位符。
泛型的限定:
? extends E : 可以接受E类型或者E的子类型。 上限。
? super E : 可以接收E类型或者E的父类型。 下限。
import java.util.*; class Person { private String name; Person(String name) { this.name = name; } public String getName() { return name; } } class Student extends Person { Student(String name) { super(name); } } class day15 { public static void main(String[] args) { /* ArrayList<String> al = new ArrayList<String>(); al.add("abc1"); al.add("abc2"); al.add("abc3"); ArrayList<Integer> al1 = new ArrayList<Integer>(); al1.add(4); al1.add(7); al1.add(1); printColl(al); printColl(al1); */ ArrayList<Person> al = new ArrayList<Person>(); al.add(new Person("abc1")); al.add(new Person("abc2")); al.add(new Person("abc3")); ArrayList<Student> al1 = new ArrayList<Student>(); al1.add(new Student("abc---1")); al1.add(new Student("abc---2")); al1.add(new Student("abc---3")); printColl(al1);//ArrayList<Person> al = new ArrayList<Student>(); error左右 } public static void printColl(ArrayList<? extends Person> al) //泛型限定 只能接受Person类或者其子类类型,下面有getName()方法不能 //传入任意类型,要有个限定 { Iterator<? extends Person> it = al.iterator(); while(it.hasNext()) { System.out.println(it.next().getName()); } } /* public static void printColl(ArrayList<?> al)//?是通配符,不确定的类型 { Iterator<?> it = al.iterator(); while(it.hasNext()) { System.out.println(it.next()); } }*/ /* public static <T> void printColl(ArrayList<T> al) { Iterator<T> it = al.iterator(); while(it.hasNext()) { T t = it.next();//可以操作该元素 System.out.println(it.next()); } } */ }
TreeSet中
Comparator<? super E> 比较器可以使用父类型,那么不同的子类就可以同时使用这个一个比较器
import java.util.*; class Person { private String name; Person(String name) { this.name = name; } public String getName() { return name; } } class Student extends Person { Student(String name) { super(name); } } class Worker extends Person { Worker(String name) { super(name); } } /* 这样太麻烦 class StuComp implements Comparator<Student> { public int compare(Student s1, Student s2) { return s1.getName().compareTo(s2.getName()); } } class WorkerComp implements Comparator<Worker> { public int compare(Worker s1, Worker s2) { return s1.getName().compareTo(s2.getName()); } } */ class Comp implements Comparator<Person> { public int compare(Person p1, Person p2) { return p2.getName().compareTo(p1.getName()); //这里面只能用父类的方法.getName() } } class day15 { public static void main(String[] args) { TreeSet<Student> ts = new TreeSet<Student>(new Comp()); ts.add(new Student("abc03")); ts.add(new Student("abc02")); ts.add(new Student("abc08")); ts.add(new Student("abc06")); ts.add(new Student("abc01")); Iterator<Student> it = ts.iterator(); while(it.hasNext()) { System.out.println(it.next().getName()); } TreeSet<Worker> ts1 = new TreeSet<Worker>(new Comp()); ts1.add(new Worker("abc--w--03")); ts1.add(new Worker("abc--w--02")); ts1.add(new Worker("abc--w--08")); ts1.add(new Worker("abc--w--06")); ts1.add(new Worker("abc--w--01")); Iterator<Worker> it1 = ts1.iterator(); while(it1.hasNext()) { System.out.println(it1.next().getName()); } } }
输出:
abc08
abc06
abc03
abc02
abc01
abc--w--08
abc--w--06
abc--w--03
abc--w--02
abc--w--01