下面参考java.util.HashMap<K, V>,写了一个简单的散列表,只实现了其中的put和get方法,使用链接法"碰撞冲突"。代码最后,自定义了一个People类,并覆盖了equals,hashCode,toString方法,实现了Comparable接口。利用People类检验散列表的正确性,并利用Arrays或Collections对People进行排序。

import java.util.Arrays;
//import java.util.Collections; class ListNode<K, V> {//为解决"碰撞冲突",使用链接法的结点结构。
ListNode<K, V> next;
K key;
V val;
public ListNode(K k, V v) {
key = k; val = v;
}
}
class HashMap<K, V> {//一个简单的散列表,实现了put和get方法。
ListNode<K, V>[] table = null; //构造一个散列表:一个存放链表的数组。
static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; //
int size;
public HashMap() {
setUp(DEFAULT_INITIAL_CAPACITY);
}
public HashMap(int capacity) {
setUp(capacity);
}
@SuppressWarnings("unchecked")
public void setUp(int capacity) {
size = capacity;
table = (ListNode<K, V>[])new ListNode[size];
}
public int hash(Object k) { //散列函数
return Math.abs(k.hashCode() % table.length);
}
public V put(K key, V value) { //return the old value
if(key == null)
return putForNullKey(value);
int i = hash(key);
for(ListNode<K, V> e = table[i]; e != null; e = e.next) {
if(e.key == key || key.equals(e.key)) {
V oldVal = e.val;
e.val = value;
return oldVal;
}
}
addListNode(i, key, value);
return null;
}
public V putForNullKey(V value) {
for (ListNode<K, V> e = table[0]; e != null; e = e.next) { // 只考虑table中下标为0的位置。
if(e.key == null) {
V oldVal = e.val;
e.val = value;
return oldVal;
}
}
table[0] = new ListNode<>(null, value);
return null;
}
public V get(K key) {
if(key == null)
return getForNullKey();
int i = hash(key);
for(ListNode<K, V> e = table[i]; e != null; e = e.next) {
if(e.key == key || key.equals(e.key))
return e.val;
}
return null;
}
public V getForNullKey() {
for(ListNode<K, V> e = table[0]; e != null; e = e.next) {
if(e.key == null)
return e.val;
}
return null;
}
public void addListNode(int i, K key, V value) {
ListNode<K, V> tmp = table[i];
table[i] = new ListNode<>(key, value);
table[i].next = tmp;
}
}
class People implements Comparable<People> { // 自定义People类,并覆盖了equals,hashCode,toString方法,实现了Comparable接口。
String name;
int age;
public People(String n, int a) {
name = n; age = a;
}
@Override
public boolean equals(Object obj) {
if(this == obj)
return true;
if(!(obj instanceof People))
return false;
return this.name.equals(((People)obj).name) && this.age == ((People)obj).age;
}
@Override
public int hashCode() {
return name.hashCode()*37 + age;
}
@Override
public String toString() {
return name + "," + age;
}
@Override
public int compareTo(People o) {
return this.age - o.age;
}
}
public class TestClass { //测试类
public static void main(String[] args) {
People[] people = new People[] { new People("xue", 25),
new People("hong", 20), new People("jun", 21)};
Arrays.sort(people);
// Collections.sort(Arrays.asList(people));
for (People peo : people) {
System.out.println(peo);
}
HashMap<People, Integer> hm = new HashMap<People, Integer>();
for (int i = 0; i < people.length; i++) {
hm.put(people[i], i);
}
for (int i = 0; i < people.length; i++) {
System.out.println(hm.get(people[i]));
}
}
}

实现一个简单的散列表(HashMap)的更多相关文章

  1. 【Java】 大话数据结构(13) 查找算法(4) (散列表(哈希表))

    本文根据<大话数据结构>一书,实现了Java版的一个简单的散列表(哈希表). 基本概念 对关键字key,将其值存放在f(key)的存储位置上.由此,在查找时不需比较,只需计算出f(key) ...

  2. HashMap、lru、散列表

    HashMap HashMap的数据结构:HashMap实际上是一个数组和链表("链表散列")的数据结构.底层就是一个数组结构,数组中的每一项又是一个链表. hashCode是一个 ...

  3. 用js来实现那些数据结构12(散列表)

    上一篇写了如何实现简单的Map结构,因为东西太少了不让上首页.好吧... 这一篇文章说一下散列表hashMap的实现.那么为什么要使用hashMap?hashMap又有什么优势呢?hashMap是如何 ...

  4. 散列表和JAVA中的hash

    引文 hello,今天写的数据结构是散列表(hash表),也算是一种基础数据结构了吧.学过计算机的人大概都能说出来这是个以空间换时间的东西,那么具体怎么实现的是今天要讨论的问题. 为什么需要它?主要还 ...

  5. 哈希表(散列表),Hash表漫谈

    1.序 该篇分别讲了散列表的引出.散列函数的设计.处理冲突的方法.并给出一段简单的示例代码. 2.散列表的引出 给定一个关键字集合U={0,1......m-1},总共有不大于m个元素.如果m不是很大 ...

  6. Python与数据结构[4] -> 散列表[0] -> 散列表与散列函数的 Python 实现

    散列表 / Hash Table 散列表与散列函数 散列表是一种将关键字映射到特定数组位置的一种数据结构,而将关键字映射到0至TableSize-1过程的函数,即为散列函数. Hash Table: ...

  7. Python与数据结构[4] -> 散列表[2] -> 开放定址法与再散列的 Python 实现

     开放定址散列法和再散列 目录 开放定址法 再散列 代码实现 1 开放定址散列法 前面利用分离链接法解决了散列表插入冲突的问题,而除了分离链接法外,还可以使用开放定址法来解决散列表的冲突问题. 开放定 ...

  8. 数据结构---散列表查找(哈希表)概述和简单实现(Java)

    散列表查找定义 散列技术是在记录的存储位置和它的关键字之间建立一个确定的对应关系f,是的每个关键字key对应一个存储位置f(key).查找时,根据这个确定的对应关系找到给定值的key的对应f(key) ...

  9. Java HashMap源码分析(含散列表、红黑树、扰动函数等重点问题分析)

    写在最前面 这个项目是从20年末就立好的 flag,经过几年的学习,回过头再去看很多知识点又有新的理解.所以趁着找实习的准备,结合以前的学习储备,创建一个主要针对应届生和初学者的 Java 开源知识项 ...

随机推荐

  1. P3128 [USACO15DEC]最大流

    秒切树上查分....(最近一次集训理解的东西) 但是,我敲了半小时才切掉这道题.... 我一直迷在了“边差分”和“点差分”的区别上. 所以,先说一下此题,再说一下区别. 首先,想到差分很容易. 然后, ...

  2. 识别手写数字增强版100% - pytorch从入门到入道(一)

    手写数字识别,神经网络领域的“hello world”例子,通过pytorch一步步构建,通过训练与调整,达到“100%”准确率 1.快速开始 1.1 定义神经网络类,继承torch.nn.Modul ...

  3. .netcore之DI批量注入(支持泛型) - xms

    一旦系统内模块比较多,按DI标准方法去逐个硬敲AddScoped/AddSingleton/AddTransient缺乏灵活性且效率低下,所以批量注入提供了很大的便捷性,特别是对于泛型的服务类,下面介 ...

  4. 2019年10月13日 linux习题 wangqingchao

    1. GUN的含义是: GNU's Not UNIX . 2. Linux一般有3个主要部分:内核.命令解释层.实用工具. 3.POSIX是可携式操作系统接口的缩写,重点在规范核心与应用程序之间的接口 ...

  5. SSM整合案例--用户登录

    实现用户登录案例,并进行非法拦截 实现当用户未登录时,无法跳转到出登录页面以外的任何页面,拦截用户仍在登陆页面:当用户登录成功即可跳转到其他页面 (1)导入依赖 <!-- https://mvn ...

  6. Ansibile之playbook初识

    一.playbook简介 ansiblie的任务配置文件被称为playbook,俗称“剧本”,每一个剧本(playbook)中都包含了一系列的任务,这每个任务在ansible中又被称为“戏剧”(pla ...

  7. nyoj 25-A Famous Music Composer(字符串)

    25-A Famous Music Composer 内存限制:64MB 时间限制:1000ms Special Judge: No accepted:4 submit:9 题目描述: Mr. B i ...

  8. web自动化测试启示篇

    1.首先,对于想学自动化测试的朋友,那么你得懂一种语言,常用的比如Java或者Python.因为没有语言基础,你是写不出自动化脚本的. 我个人选择java 2.有了开发语言的铺垫,那么开始入手Sele ...

  9. 某些机root也不能访问dma-buf

    从4.3后,回顾<从surfaceflinger历史变更谈截屏>,只能通过生产消费者队列向surfaceflinger服务申请显示缓冲,这个缓冲就是dma-buf映射的共享内存. bind ...

  10. Java w3c离线手册

    提供给大家使用,懒得找: 前端后端一般都用的到  查看文档 1. JDK_API_1_6_zh_CN.CHM 2.     W3School离线手册(2018.04.01).chm 3.    jqu ...