# JAVA之旅(二十)—HashSet,自定義存儲對象,TreeSet,二叉樹,實現Comparator方式排序,TreeSet小練習
* * *
> 我們繼續說一下集合框架
* Set:元素是無序(存入和取出的順序不一定一致),元素不可以重復
> Set集合的功能和Collection是一致的

> 我們重點關注的是子類對象

> 我們來聊聊
## 一.HashSet
> HashSet底層結構是哈希表
>
> 什么是HashSet?
~~~
package com.lgl.hellojava;
//公共的 類 類名
public class HelloJJAVA {
public static void main(String[] args) {
Demo d1 = new Demo();
Demo d2 = new Demo();
sop(d1);
sop(d2);
}
// 輸出
public static void sop(Object obj) {
System.out.println(obj);
}
}
class Demo {
}
~~~
> 我們這樣輸出的結果就是哈希值

> 當然,我們是來介紹HashSet的,我們演示一下
~~~
package com.lgl.hellojava;
import java.util.HashSet;
import java.util.Iterator;
//公共的 類 類名
public class HelloJJAVA {
public static void main(String[] args) {
HashSet h = new HashSet();
h.add("hello 01");
h.add("hello 02");
h.add("hello 03");
h.add("hello 04");
// set取出只有一種辦法,迭代器
Iterator iterator = h.iterator();
while (iterator.hasNext()) {
sop(iterator.next());
}
}
// 輸出
public static void sop(Object obj) {
System.out.println(obj);
}
}
~~~
> 是不是很類似,但是輸出,你們仔細看了

> 輸出是無序的,我們還有一個現象,就是直接輸出
~~~
sop(h.add("lgl"));
sop(h.add("lgl"));
~~~
> 相同的

> 因為他不能重復
## 二.自定義存儲對象
> 我們可以存數據,那肯定可以自定義存儲數據咯?
~~~
package com.lgl.hellojava;
import java.util.HashSet;
import java.util.Iterator;
//公共的 類 類名
public class HelloJJAVA {
public static void main(String[] args) {
HashSet h = new HashSet();
h.add(new Person("lgl1", 18));
h.add(new Person("lgl2", 19));
h.add(new Person("lgl3", 20));
h.add(new Person("lgl4", 21));
// set取出只有一種辦法,迭代器
Iterator iterator = h.iterator();
while (iterator.hasNext()) {
Person p = (Person) iterator.next();
sop(p.getName() + ":" + p.getAge());
}
}
// 輸出
public static void sop(Object obj) {
System.out.println(obj);
}
}
/**
* 存儲對象
*
* @author LGL
*
*/
class Person {
private String name;
private int age;
public Person(String name, int age) {
this.setName(name);
this.setAge(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;
}
}
~~~
> 這樣就可以定下來了
* HashSet是如何保證元素的唯一性呢??
* 是通過元素的兩個方法,hasCode和equals來完成的
* 如果元素的hasCode相同。才會去判斷equals是否為true
* 如果元素的hasCode不同。不會調用equals
> 這里要注意一點的就是,對于判斷元素是否存在的話,以及刪除的操作,依賴的方法就是元素的hasCode和equals
## 三.TreeSet
> hashSet說完,我們再來看一下TreeSet,我們用小例子來說明
~~~
package com.lgl.hellojava;
import java.util.Iterator;
import java.util.TreeSet;
import org.omg.PortableInterceptor.Interceptor;
//公共的 類 類名
public class HelloJJAVA {
public static void main(String[] args) {
TreeSet s = new TreeSet();
s.add("abc");
s.add("acd");
s.add("age");
s.add("abf");
Iterator iterator = s.iterator();
while (iterator.hasNext()) {
sop(iterator.next());
}
}
// 輸出
public static void sop(Object obj) {
System.out.println(obj);
}
}
~~~
> 我們仔細看他的輸出

> 他會排序,那我們就知道TreeSet的特性了
* 可以對Set集合中的元素進行排序
> 如果你用自定義對象去村粗的話,你會發現他可以存一個對象,但是不能存儲多個對象,為什么?因為他會強制進行排序,如果是對象的話,他沒法排序,是不行的
>
> 對了我們沒有講TreeSet的數據結構呢,他的數據結構是二叉樹,這是一個比較難的概念了
## 四.二叉樹
> 二叉樹其實通俗一點,就是樹形圖數據,比如

> 就是比較,一直分支,很大的節約了計算方式,我們比較,大的話,開一個分支,小的話,再開一個分支,就這樣一直比較!
>
> 那TreeSet保證元素唯一性的是compareTo方法return 0;
* TreeSet排序的第一種方式,讓元素自身具備比較性,元素需要實現Comparable 接口,覆蓋compareTo方法,這種也稱為元素的自然順序!
## 五.實現Comparator方式排序
> 當元素不具備比較性時,或者具備的元素的比較性不是所需要的,這時就需要讓集合自身具備比較性,那就是在集合一初始化時就有了比較方式.這么說有點繞啊,我們還是用代碼來說明吧,原理都是二叉樹
~~~
package com.lgl.hellojava;
import java.util.Comparator;
import java.util.Iterator;
import java.util.TreeSet;
//公共的 類 類名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 當元素自身不具備比較性或者具備的比較性不是所需要的,這時需要讓容器自生具備比較性,定義一個比較器,
* 將比較器對象作為參數傳遞給TreeSet集合的構造函數
*/
TreeSet s = new TreeSet(new MyCompare());
s.add(new Student("lgl1", 22));
s.add(new Student("lgl2", 26));
s.add(new Student("lgl3", 10));
s.add(new Student("lgl4", 19));
Iterator iterator = s.iterator();
while (iterator.hasNext()) {
Student student = (Student) iterator.next();
sop(student.getName() + ":" + student.getAge());
}
}
// 輸出
public static void sop(Object obj) {
System.out.println(obj);
}
}
class Student {
private String name;
private int age;
public 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;
if (this.age > s.age) {
return 1;
} else if (this.age == s.age) {
return this.name.compareTo(s.name);
}
return -1;
}
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;
}
}
// 定義比較器
class MyCompare implements Comparator {
public int compare(Object o1, Object o2) {
Student s1 = (Student) o1;
Student s2 = (Student) o2;
return s1.getName().compareTo(s2.getName());
}
}
~~~
## 六.TreeSet小練習
> 我們到這里,就用一個小練習來結束吧,畢竟在后面就需要講泛型了,我們的需求就是按照字符串長度排序
~~~
package com.lgl.hellojava;
import java.util.Comparator;
import java.util.Iterator;
import java.util.TreeSet;
//公共的 類 類名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 按照字符串長度排序
*/
TreeSet s = new TreeSet(new StringLengthComparator());
s.add("ffffffff");
s.add("fffff");
s.add("ff");
s.add("ffffff");
Iterator iterator = s.iterator();
while (iterator.hasNext()) {
sop(iterator.next());
}
}
// 輸出
public static void sop(Object obj) {
System.out.println(obj);
}
}
// 定義比較性
class StringLengthComparator implements Comparator {
@Override
public int compare(Object o1, Object o2) {
String s1 = (String) o1;
String s2 = (String) o2;
if (s1.length() > s2.length())
return 1;
if (s1.length() == s2.length())
return 0;
return -1;
}
}
~~~
> 這樣就OK了,輸出的結果

> 這樣就O了,好的,但是我們重復元素也會被干掉的,這時候我們就要處理了
~~~
@Override
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;
}
~~~
> 到這里,就基本上都搞定了,我們的博文到這里也結束了,如果有機會
## 可以加群討論:555974449
版權聲明:本文為博主原創文章,博客地址:http://blog.csdn.net/qq_26787115,未經博主允許不得轉載。
- 0-發現
- AndroidInterview-Q-A
- Android能讓你少走彎路的干貨整理
- LearningNotes
- temp
- temp11
- 部分地址
- 0-待辦任務
- 待補充列表
- 0-未分類
- AndroidView事件分發與滑動沖突處理
- Spannable
- 事件分發機制詳解
- 1-Java
- 1-Java-01基礎
- 未歸檔
- 你應該知道的JDK知識
- 集合框架
- 1-Java-04合集
- Java之旅0
- Java之旅
- JAVA之旅01
- JAVA之旅02
- JAVA之旅03
- JAVA之旅04
- JAVA之旅05
- JAVA之旅06
- JAVA之旅07
- JAVA之旅08
- JAVA之旅09
- java之旅1
- JAVA之旅10
- JAVA之旅11
- JAVA之旅12
- JAVA之旅13
- JAVA之旅14
- JAVA之旅15
- JAVA之旅16
- JAVA之旅17
- JAVA之旅18
- JAVA之旅19
- java之旅2
- JAVA之旅20
- JAVA之旅21
- JAVA之旅22
- JAVA之旅23
- JAVA之旅24
- JAVA之旅25
- JAVA之旅26
- JAVA之旅27
- JAVA之旅28
- JAVA之旅29
- java之旅3
- JAVA之旅30
- JAVA之旅31
- JAVA之旅32
- JAVA之旅33
- JAVA之旅34
- JAVA之旅35
- 1-Java-05辨析
- HashMapArrayMap
- Java8新特性
- Java8接口默認方法
- 圖解HashMap(1)
- 圖解HashMap(2)
- 2-Android
- 2-Android-1-基礎
- View繪制流程
- 事件分發
- AndroidView的事件分發機制和滑動沖突解決
- 自定義View基礎
- 1-安卓自定義View基礎-坐標系
- 2-安卓自定義View基礎-角度弧度
- 3-安卓自定義View基礎-顏色
- 自定義View進階
- 1-安卓自定義View進階-分類和流程
- 10-安卓自定義View進階-Matrix詳解
- 11-安卓自定義View進階-MatrixCamera
- 12-安卓自定義View進階-事件分發機制原理
- 13-安卓自定義View進階-事件分發機制詳解
- 14-安卓自定義View進階-MotionEvent詳解
- 15-安卓自定義View進階-特殊形狀控件事件處理方案
- 16-安卓自定義View進階-多點觸控詳解
- 17-安卓自定義View進階-手勢檢測GestureDetector
- 2-安卓自定義View進階-繪制基本圖形
- 3-安卓自定義View進階-畫布操作
- 4-安卓自定義View進階-圖片文字
- 5-安卓自定義View進階-Path基本操作
- 6-安卓自定義View進階-貝塞爾曲線
- 7-安卓自定義View進階-Path完結篇偽
- 8-安卓自定義View進階-Path玩出花樣PathMeasure
- 9-安卓自定義View進階-Matrix原理
- 通用類介紹
- Application
- 2-Android-2-使用
- 2-Android-02控件
- ViewGroup
- ConstraintLayout
- CoordinatorLayout
- 2-Android-03三方使用
- Dagger2
- Dagger2圖文完全教程
- Dagger2最清晰的使用教程
- Dagger2讓你愛不釋手-終結篇
- Dagger2讓你愛不釋手-重點概念講解、融合篇
- dagger2讓你愛不釋手:基礎依賴注入框架篇
- 閱讀筆記
- Glide
- Google推薦的圖片加載庫Glide:最新版使用指南(含新特性)
- rxjava
- 這可能是最好的RxJava2.x入門教程完結版
- 這可能是最好的RxJava2.x入門教程(一)
- 這可能是最好的RxJava2.x入門教程(三)
- 這可能是最好的RxJava2.x入門教程(二)
- 這可能是最好的RxJava2.x入門教程(五)
- 這可能是最好的RxJava2.x入門教程(四)
- 2-Android-3-優化
- 優化概況
- 各種優化
- Android端秒開優化
- apk大小優化
- 內存分析
- 混淆
- 2-Android-4-工具
- adb命令
- 一鍵分析Android的BugReport
- 版本控制
- git
- git章節簡述
- 2-Android-5-源碼
- HandlerThread 源碼分析
- IntentService的使用和源碼分析
- 2-Android-9-辨析
- LRU算法
- 什么是Bitmap
- 常見圖片壓縮方式
- 3-Kotlin
- Kotlin使用筆記1-草稿
- Kotlin使用筆記2
- kotlin特性草稿
- Kotlin草稿-Delegation
- Kotlin草稿-Field
- Kotlin草稿-object
- 4-JavaScript
- 5-Python
- 6-Other
- Git
- Gradle
- Android中ProGuard配置和總結
- gradle使用筆記
- Nexus私服搭建
- 編譯提速最佳實踐
- 7-設計模式與架構
- 組件化
- 組件化探索(OKR)
- 1-參考列表
- 2-1-組件化概述
- 2-2-gradle配置
- 2-3-代碼編寫
- 2-4-常見問題
- 2-9-值得一讀
- 8-數據結構與算法
- 0臨時文件
- 漢諾塔
- 8-數據-1數據結構
- HashMap
- HashMap、Hashtable、HashSet 和 ConcurrentHashMap 的比較
- 遲到一年HashMap解讀
- 8-數據-2算法
- 1個就夠了
- Java常用排序算法(必須掌握的8大排序算法)
- 常用排序算法總結(性能+代碼)
- 必須知道的八大種排序算法(java實現)
- 9-職業
- 閱讀
- 書單
- 面試
- 面試-01-java
- Java面試題全集駱昊(上)
- Java面試題全集駱昊(下)
- Java面試題全集駱昊(中)
- 面試-02-android
- 40道Android面試題
- 面試-03-開源源碼
- Android圖片加載框架最全解析(二),從源碼的角度理解Glide的執行流程
- 面試-07-設計模式
- 面試-08-算法
- 面試-09-其他
- SUMMARY
- 版權說明
- temp111