16.集合与泛型

  1. 除了ArrayList外还有这些集合
  • Treeset 以有序状态保持并可防止重复
  • HashMap 可用成对的name/value来保存与取出
  • LinkedList 针对经常插入或删除中间元素所设计的高效率集合
  • HashSet 防止重复的集合,可快速地寻找相符的元素
  • LinkedHashMap 类似HashMap,但可记住元素插入的顺序,也可以设定成依照元素上次存取的先后来排序
  1. 可以用TreeSet或Collections.sort( )方法对数组进行按字母排序
  2. toString( )
  • 因为toString( )是定义在Object类中的,所以Java中的每个类都会继承toString( ),又因为对象被System.out.println(anObject)列出来时会被调用toString( ),所以当你把list列出时,每个对象的toString( )都会被调用一次。
public String toString(){ 
return title;
}
  1. 为什么要用泛型?(带有<>的就是泛型)
  • 运用泛型可以创建类型安全更好的集合,让问题尽可能在编译器就能抓到,而不会等到执行期才冒出来
  • 如果没有泛型,编译器会很愉快地接受你把绵羊对象送到老虎集合中
  1. 泛型使用须知
    (1)创建被泛型化类(如ArrayList)的实例new ArrayList<Song>();
    (2)声明与指定泛型类型的变量List<Song> songList = new ArrayList<Song>();
    (3)声明(与调用)取用泛型类型的方法。void foo(List<Song> list);
    6.ArrayList的说明文件
public class ArrayList<E> extends AbstractList<E> implements List<E>...{
  E部分会用你所声明与创建的真正类型来替代
public boolean add(E o);//E用来指示可以加入ArrayList的元素类型
}E代表用来创建与初始ArrayList的类型,当你看到ArrayList文件上的E时,就可以把它换成实际上的类型
  1. 使用泛型的方法
  • 使用定义在类声明的类型参数
public class ArrayList<E> extends AbstractList<E>...{
        public boolean add(E o);只能在此使用E,因为它已经被定义成类的一部分
  • 使用未定义在类声明的类型参数
public <T extends Animal> void takeThing(ArrayList<T> list)
可以传入Animal或任何Animal的子型(如Cat或Dog),可以使用ArrayList<Dog>来调用该方法
  1. sort()方法只能接受Comparable对象的list。
public static <T extends Comparable<?  super T>> void sort(List<T> list)
             这表示它必须是Comparable           仅能传入继承Comparable的参数化类型的list

因此,只要让要排序的类(比如Song)实现Comparable就行了

class Song implements Comparable<Song>{
        public int comparaTo(Song s){
                return title.compareTo(s.getTitle());
用compareTo()方法将执行方法的Song与传入的Song比较并输出
  1. int compareTo(T o)
    比较此对象与指定对象的顺序。如果该对象小于、等于或大于指定对象,则分别返回负整数、零或正整数。
  2. Comparator
  • 使用compareTo()方法时,list中的元素只能有一种将自己与同类的另一个元素做比较的方法
  • 但Comparator是独立与所比较的元素类型之外的——它是独立的类
  1. compareTo()与Caomparator比较
  • 调用单一参数的sort(List o)方法代表由List元素上的compareTo()方法来决定顺序。因此元素必须要实现Comparable这个接口
  • 调用两个参数的sort(List o,Compararot c)方法代表不会调用list元素的compareTo()方法,而会使用Comparator的compare()方法。这意味着list元素不需要实现Comparable
class ArtistCompare implements Comparator<Song>{
        public int Compare(Song one,Song two){
        return one.getArtist().compareTo(two.getArtist());
    }              这会返回String ,以String来比较
}
public void go(){
...
        ArtistCompare artistCompare = new AritstCompare();
        Collections.sort(songList,artistCompare);
}

2

  1. 如何解决数据重复——用Set集合
  2. 集合中的三个主要接口——List Set Map
  • List——对付顺序的好帮手(是一种知道索引位置的集合,可重复)
    List知道某物在系列集合中的位置,可以由多个元素引用相同的对象
  • SET——注重独一无二的性质(不允许重复的集合,不会重复)
    它知道某物是否已存在与集合中,不会有多个元素引用相同的对象
  • MAP——用Key来搜索的专家(值可以重复,但key不行)
    两个key可以引用相同的对象,但key不能重复,典型的key会是String,但也可以是任何对象
  1. 对象怎样才算相等?
  • 引用相等性——堆上同一个对象的两个引用
    可以用==来比较两个引用是否相等
  • 对象相等性——堆上的两个不同对象在意义上是相同的
    如果想要把两个不同的对象视为相等的,就必须覆盖过从Object继承下来的hashCode()方法与equals方法
    4.HashSet如何检查重复:hashCode()equals()
  • 当你把对象加入HashSet时,它会使用对象的hashCode值来判断对象加入的位置。但同时也会与其他已经加入的对象的hashCode作比对,如果没有相符的hashCode,HashSet就会假设新对象没有重复出现。
    也就是说,如果hashCode不同,则HashSet会假设对象不可能是相同的(但有相同hashCode的对象也不一定相等)
  1. 覆盖过hashCode()equals()的Song类
public boolean equals(Object aSong){  //aSong是要被比较的对象
        Song s = (Song) aSong;
        return getTitle().equals(s.getTitle));
}//因为歌名是String,且String本来就覆盖过的equals(),所以我们可以调用它
public int hashCode(){
        return title.hashCode();
}//String也有覆盖过的hashCode(),注意到hashCode()与equals()使用相同的实例变量
  1. hashCodeequals()的相关规定
  • 如果两个对象相等,则hashcode必须也是相等的
  • 如果两个对象相等,对其中一个对象调用equals()必须返回true,也就是说,若a.equas(b),则b.equals(a)
  • 如果两个对象有相同的hashCode值,它们也不一定是相等的。但若两个对象相等,则hashCode值一定是相等的
  • 因此若equals()被覆盖过,则hashCode()也必须被覆盖
  • hashCode()的默认行为是对在heap上的对象产生独特的值。如果你没有override过hashCode(),则该class的两个对象怎样都不会被认为是相同的
  • equals()的默认行为是执行==的比较。也就是说会去测试两个引用是否是否对上heap上的同一个对象。如果equals()没有被覆盖过,两个对象永远都不会被视为相同的,因为不同的对象有不同的字节组合。
    a.equals(b)必须与a.hashCode() == b.hashCode()等值
    a.hashCode() == b.hashCode()不一定要与a.equals()等值
  1. TreeSet
    TreeSet在防止重复上面与HashSe一样。但它还会一直保持集合处于有序状态
public void go(){  //调用没有参数的构造函数来用TreeSet
...                //取代HashSet意味着以对象的compareTo()方法来进行排序
TreeSet<Song> songSet = new TreeSet<Song>();
songSet.addAll(songList);
}
  1. TreeSet的元素必须是Comparable
    要使用TreeSet,下列其中一项必须为真
  • 集合中的元素必须是有实现Comparable的类型
class Book implements Comparable{
        String title;
        public Book(String t){
                title = t;
        }
        public int compareTo(Object b){
                Book book = (Book) b;
                return (title.compareTo(book.title));
        }
}
  • 使用重载、取用Comparator参数的构造函数来创建TreeSet
public class BookCompare implements Comparator<Book>{
      public int compare(Book one,Book two){
          return (one.title.compareTo(two.title));
      }
}
class Test{
        public void go(){
                BookCompare bCompare = new BookCompare();
                TreeSet<Book> tree = new TreeSet<Book>(bCompare);
        }
  1. Map
    Map中的元素实际上是两个对象:关键字和值,值可以重复,但是key不行
public static void main(String[] args){//hashMap需要两个参数,关键字和值
        hashMap<String,Integer> scores = new hashMap<String,Integer>();
        scores.put("kathy",42);
        scores.put("Bert",343);//使用put()取代add(),它需要两个参数
        System.out.println(scores);
        System.out.println(scores.get("Bert"));//get()取用关键字参数,返回它的值
}
  1. 数组的类型是在运行期间检查的,但集合的类型检查只会发生在编译期间
  2. 万用字符<?>
public void takeAnimals(ArrayList<? extends Animals> animals) {
...}       //此处的extends同时代表继承和实现
  • 在方法中使用万用字符时,编译器会阻止任何可能破坏引用参数所指集合的行为
  • 你能够调用List中任何元素的方法,但不能加入元素
  • 也就是说,你可以操作集合元素,但不能新增集合元素。如此才能保证执行期间的安全性,因为编译器会阻止执行期的恐怖行动。
  1. 相同功能的另一种语法
public <T extends Animal> void takeThing(ArrayList<T> list)
  • 如果都一样,为什么要用有?那个
    这要看你是否会用到T来决定,如果方法有两个参数,则可以
public <T extends Animals> void takeThing(ArrayList<T> one,ArrayList<T> two)

而不必这样

public void takeThing(ArrayList<? extends Animal> one,ArrayList<? extends Animals> two)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容

  • 上一篇文章介绍了Set集合的通用知识。Set集合中包含了三个比较重要的实现类:HashSet、TreeSet和En...
    Ruheng阅读 15,635评论 3 57
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,612评论 18 399
  • java笔记第一天 == 和 equals ==比较的比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量...
    jmychou阅读 1,495评论 0 3
  • 3.3 集合 一方面, 面向对象语言对事物的体现都是以对象的形式,为了方便对多个对象的操作,就要对对象进行存储。另...
    闫子扬阅读 726评论 0 1
  • 阳春三月,窗外跳跃着细细碎碎的阳光,我随意打开QQ好友,发现了以前的一位同事,发过去一个简短的问候:“你好!”对方...
    董灿阅读 732评论 0 3