提示:以下是本篇文章正文内容,下面案例可供参考
Map是一种专门用来进行搜索的容器或者数据结构,其搜索的效率与其具体的实例化子类有关。以前常见的
搜索方式有:
一般把搜索的数据称为关键字(Key),和关键字对应的称为值(Value),将其称之为Key-value的键值对,所以模型会有两种。
1.字典中找一个字
2.通讯录找名字
1.统计文件中每个单词出现的次数,统计结果是每个单词都有与其对应的次数:<单词,单词出现的次数>。
2.梁山好汉的绰号
而Map中存储的就是key-value的键值对,Set中只存储了Key。
Map是一个接口类,该类没有继承自Collection,该类中存储的是
Map.Entry
三大方法:
注意:Map.Entry
注意:
public static void TestMap(){
Map<String, String> m = new TreeMap<>();
// put(key, value):插入key-value的键值对
// 如果key不存在,会将key-value的键值对插入到map中,返回null
m.put("林冲", "豹子头");
m.put("鲁智深", "花和尚");
m.put("武松", "行者");
m.put("宋江", "及时雨");
String str = m.put("李逵", "黑旋风");
System.out.println(m.size());
System.out.println(m);
// put(key,value): 注意key不能为空,但是value可以为空
// key如果为空,会抛出空指针异常
//m.put(null, "花名");
str = m.put("无名", null);
System.out.println(m.size());
// put(key, value):
// 如果key存在,会使用value替换原来key所对应的value,返回旧value
str = m.put("李逵", "铁牛");
// get(key): 返回key所对应的value
// 如果key存在,返回key所对应的value
// 如果key不存在,返回null
System.out.println(m.get("鲁智深"));
System.out.println(m.get("史进"));
//GetOrDefault(): 如果key存在,返回与key所对应的value,如果key不存在,返回一个默认值
System.out.println(m.getOrDefault("李逵", "铁牛"));
System.out.println(m.getOrDefault("史进", "九纹龙"));
System.out.println(m.size());
//containKey(key):检测key是否包含在Map中,时间复杂度:O(logN)
// 按照红黑树的性质来进行查找
// 找到返回true,否则返回false
System.out.println(m.containsKey("林冲"));
System.out.println(m.containsKey("史进"));
// containValue(value): 检测value是否包含在Map中,时间复杂度: O(N)
// 找到返回true,否则返回false
System.out.println(m.containsValue("豹子头"));
System.out.println(m.containsValue("九纹龙"));
// 打印所有的key
// keySet是将map中的key防止在Set中返回的
for(String s : m.keySet()){
System.out.print(s + " ");
} S
ystem.out.println();
// 打印所有的value
// values()是将map中的value放在collect的一个集合中返回的
for(String s : m.values()){
System.out.print(s + " ");
} S
ystem.out.println();
// 打印所有的键值对
// entrySet(): 将Map中的键值对放在Set中返回了
for(Map.Entry<String, String> entry : m.entrySet()){
System.out.println(entry.getKey() + "--->" + entry.getValue());
} S
ystem.out.println();
}
哈希函数方法:hash(key) = key%capacity
public class HashBuck {
static class Node{
public int key;
public int val;
public Node next;
public Node(int key,int val) {
this.key = key;
this.val = val;
}
}
public Node []arr;
public int usedsize;
public HashBuck(){
arr = new Node[10];
}
public void put(int key,int val){
int index = key% arr.length;
Node cur = arr[index];
while(cur!=null){
if (cur.key == key){
cur.val = val;
return;
}
cur = cur.next;
}
Node node = new Node(key, val);
node.next = arr[index];
arr[index] = node;
usedsize++;
}
public void resize(){
Node []newarr = new Node[2* arr.length];
for (int i = 0; i < arr.length; i++) {
Node cur = arr[i];
}
}
public double calculateLoadFactor(){
return usedsize*1.0/arr.length;
}
public int get(int key){
int index = key% arr.length;
Node cur = arr[index];
while (cur!=null){
if (cur.key == key){
return cur.val;
}
cur = cur.next;
}
return -1;
}
测试方法
HashBuck hashBuck = new HashBuck();
hashBuck.put(10, 20);
hashBuck.put(30, 5);
hashBuck.put(43, 6);
hashBuck.put(50, 7);
Integer val = hashBuck.get(10);
System.out.println(val);
对于两个数据元素的关键字 和 (i != j),有 != ,但有:Hash( ) == Hash( ),即:不同关键字通过相同哈
希哈数计算出相同的哈希地址,该种现象称为哈希冲突或哈希碰撞
把具有不同关键码而具有相同哈希地址的数据元素称为“同义词”。
引起哈希冲突的一个原因可能是:哈希函数设计不够合理。 哈希函数设计原则:
1.哈希函数的定义域必须包括需要存储的全部关键码,而如果散列表允许有m个地址时,其值域必须在0到m-1之间
2.哈希函数计算出来的地址能均匀分布在整个空间中
3.哈希函数应该比较简单
常见哈希函数
1.闭散列:也叫开放定址法,当发生哈希冲突时,如果哈希表未被装满,说明在哈希表中必然还有空位置,那么可以把key存放到冲突位置中的“下一个” 空位置中去。
1.1. 线性探测
比如上面的场景,现在需要插入元素44,先通过哈希函数计算哈希地址,下标为4,因此44理论上应该插在该位置,但是该位置已经放了值为4的元素,即发生哈希冲突。
线性探测:从发生冲突的位置开始,依次向后探测,直到寻找到下一个空位置为止。
插入:
通过哈希函数获取待插入元素在哈希表中的位置
如果该位置中没有元素则直接插入新元素,如果该位置中有元素发生哈希冲突,使用线性探测找到
下一个空位置,插入新元素。
2.开散列/哈系桶:
开散列法又叫链地址法(开链法),首先对关键码集合用散列函数计算散列地址,具有相同地址的关键码归于同一子集合,每一个子集合称为一个桶,各个桶中的元素通过一个单链表链接起来,各链表的头结点存储在哈希表中。
思路:
每一次哈希后指向数组下标的值,然后桶中的元素通过一个单链表链接起来,各链表的头结点存储在哈希表中。
开散列,可以认为是把一个在大集合中的搜索问题转化为在小集合中做搜索了。
public class HashBuck {
static class Node{
public int key;
public int val;
public Node next;
public Node(int key,int val) {
this.key = key;
this.val = val;
}
}
public Node []arr;
public int usedsize;
public HashBuck(){
arr = new Node[10];
}
public void put(int key,int val){
int index = key% arr.length;
Node cur = arr[index];
while(cur!=null){
if (cur.key == key){
cur.val = val;
return;
}
cur = cur.next;
}
Node node = new Node(key, val);
node.next = arr[index];
arr[index] = node;
usedsize++;
}
public void resize(){
Node []newarr = new Node[2* arr.length];
for (int i = 0; i < arr.length; i++) {
Node cur = arr[i];
}
}
public double calculateLoadFactor(){
return usedsize*1.0/arr.length;
}
public int get(int key){
int index = key% arr.length;
Node cur = arr[index];
while (cur!=null){
if (cur.key == key){
return cur.val;
}
cur = cur.next;
}
return -1;
}
好了,今天的博客到这里就结束了,欢迎各位在评论区讨论,求个一键三连。