1,在词典的实现(2)-借助顺序表(数组)实现词典文章中使用了自定义的数组代替ArrayList,并实现了Map数据结构的基本功能。而借助JAVA类库ArrayList类的一些方法可以更加容易地实现Map。

2,实现思路如下

ArrayListDictionary.java 中定义了一个ArrayList的对象,该ArrayList对象用来存储Entry类的对象,而Entry类封装了(key,value)。这样,利用ArrayList类的一些方法来间接地操作(key,value),从而实现各种词典的操作。

ArrayListDictionary.java 中部分代码解释如下:

public ArrayListDictionary(){
listDictionary = new ArrayList<Entry>();
}

在构造方法中初始化ArrayList对象。这样,每生成一个词典对象,就会有一个ArrayList对象。因为,每个词典对象都需要一个ArrayList对象来存储词典中的元素。

private class Entry{
K key;
V value;

私有内部类Entry用来封装(key,value)对,key 只有get方法没有set方法。因为对于词典而言,key是不能更改的。而value既有get方法又有set方法。

private class KeyIterator<S> implements Iterator<K>{

        Iterator<Entry> it = null;
public KeyIterator(){
it = listDictionary.iterator();
} @Override
public boolean hasNext() {
return it.hasNext();
} @Override
public K next() {
return (K) it.next().getKey();
} @Override
public void remove() {
throw new UnsupportedOperationException("can not remove a entry in iterator.unsupported"); }
}

私有内部类KeyIterator用来实现词典中键的迭代器。该迭代器的实现基于遍历Entry的迭代器,由于ArrayList存储Entry对象,因此可以借助ArrayList的iterator()方法很容易地得到遍历Entry的迭代器。在KeyIterator的构造方法中实例化遍历Entry对象的迭代器后,通过实现Iterator接口的三个方法来实现遍历词典的查找键的迭代器。

@Override
public Iterator<K> getKeyIterator() {
return new KeyIterator();
}

ArrayListDictionary.java的getKeyIterator()方法返回一个遍历词典查找键的迭代器的对象。这样,ArrayListDictionary的对象就可以调用该方法来得到该迭代器了。

private int locateIndex(K key){
int index = -1;
Iterator<Entry> it = listDictionary.iterator();
while(it.hasNext()){
Entry e = it.next();//先获得每一个Map元素
if(e.getKey().equals(key)){//依次与每一个Map元素的查找键比较
index = listDictionary.indexOf(e);//获得给定的key匹配的Map元素在Arraylist中的位置
break;
}
}
return index;
}

私有方法locateIndex(K key),用来查找某个查找键在ArrayList数组中的哪个位置。由于ArrayList数组中存储的是Entry对象,因此先要通过遍历Entry的迭代器获得Entry对象,然后通过Entry类的getKey()方法来获得Entry对象的key属性。index 返回 –1 表示查找键key(所代表的Entry对象)不在ArrayList中。

public V remove(K key) {
V result = null;
int index = locateIndex(key);
if(index == -1){//key 不存在于ArrayList中
result = null;
}
else{
result = listDictionary.get(index).getValue();
listDictionary.remove(index);
}
return result;
}

remove()用来删除词典中的元素。其他的一些方法的功能可以参考DictionaryInterface.java中声明的方法解释。

整个完整的ArrayListDictionary.java实现如下

package dictionary;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.Iterator; public class ArrayListDictionary<K,V> implements DictionaryInterface<K, V>,Serializable { private static final long serialVersionUID = 1L;
ArrayList<Entry> listDictionary = null; public ArrayListDictionary(){
listDictionary = new ArrayList<Entry>();
} private class Entry{
K key;
V value; private Entry(K key, V value){
this.key = key;
this.value = value;
} public K getKey() {
return key;
} public V getValue() {
return value;
} public void setValue(V value) {
this.value = value;
}
} private class KeyIterator<S> implements Iterator<K>{ Iterator<Entry> it = null;
public KeyIterator(){
it = listDictionary.iterator();
} @Override
public boolean hasNext() {
return it.hasNext();
} @Override
public K next() {
return (K) it.next().getKey();
} @Override
public void remove() {
throw new UnsupportedOperationException("can not remove a entry in iterator.unsupported"); }
} private class ValueIterator<V> implements Iterator<V>{ Iterator<Entry> it = null;
public ValueIterator(){
it = listDictionary.iterator();
} @Override
public boolean hasNext() {
return it.hasNext();
} @Override
public V next() {
return (V) it.next().getValue();
} @Override
public void remove() {
throw new UnsupportedOperationException();
}
} //返回指定的查找键所代表的元素在arraylist中的位置
private int locateIndex(K key){
int index = -1;
Iterator<Entry> it = listDictionary.iterator();
while(it.hasNext()){
Entry e = it.next();//先获得每一个Map元素
if(e.getKey().equals(key)){//依次与每一个Map元素的查找键比较
index = listDictionary.indexOf(e);//获得给定的key匹配的Map元素在Arraylist中的位置
break;
}
}
return index;
} @Override
public V add(K key, V value) {
V result = null;
int index = locateIndex(key);
if(index == -1){//key 不在ArrayList中
listDictionary.add(new Entry(key, value));
}
else{
//这里index不会为-1,所以 get(index)不会抛出IndexOutOfBoundsException
result = listDictionary.get(index).getValue();
listDictionary.get(index).setValue(value);//调用Entry内部类中的setValue方法更新Entry的Value
}
return result;
} @Override
public V remove(K key) {
V result = null;
int index = locateIndex(key);
if(index == -1){//key 不存在于ArrayList中
result = null;
}
else{
result = listDictionary.get(index).getValue();
listDictionary.remove(index);
}
return result;
} @Override
public V getValue(K key) {
V result = null;
int index = locateIndex(key);
try{
result = listDictionary.get(index).getValue();//当index=-1时,表示key不存在
}catch(IndexOutOfBoundsException e){//index = -1时抛出异常
result = null;
}
return result;
} @Override
public boolean contains(K key) {
boolean result = false;
Iterator<Entry> it = listDictionary.iterator();
while(it.hasNext()){
Entry e = it.next();
if(e.getKey().equals(key)){
result = true;
break;
}
}
return result;
} @Override
public Iterator<K> getKeyIterator() {
return new KeyIterator();
} @Override
public Iterator<V> getValueIterator() {
return new ValueIterator();
} @Override
public boolean isEmpty() {
return listDictionary.isEmpty();
} @Override
public boolean isFull() {
return false;
} @Override
public int getSize() {
return listDictionary.size();
} @Override
public void clear() {
listDictionary.clear();
}
}

在实现完词典后,以下写了个测试程序借且词典来统计单词的个数。思路如下:

文本文件word.txt中存放待统计的单词,词典客户端程序PrintWordsFrequency.java 提供word.txt,然后查看每个单词出现的次数。

词典表FrequencyCounterDictionary.java负责统计word.txt中的单词并将结果存储到Map词典中,并实现对词典的遍历(display())。

PrintWordsFrequency.java 代码如下:

package dictionary.common;
package dictionary.client; import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.Scanner; import dictionary.common.FrequencyCounterDictionary; public class PrintWordsFrequency {
public static void main(String[] args) {
FrequencyCounterDictionary wordCounter = new FrequencyCounterDictionary();
String fileName = "word.txt";//存放单词的文本文件,统计其中的单词出现次数
try{
Scanner data = new Scanner(new File(fileName));
wordCounter.readFile(data);
}catch(FileNotFoundException e){
System.out.println("File not found: " + e.getMessage());
}catch(IOException e){
System.out.println("I/O error" + e.getMessage());
} wordCounter.display();
System.out.println("Bye");
}
}

FrequencyCounterDictionary.java代码如下:

package dictionary.common;

import java.util.Iterator;
import java.util.Scanner; import dictionary.ArrayListDictionary;
import dictionary.DictionaryInterface; public class FrequencyCounterDictionary {
private DictionaryInterface<String, Integer> wordTable; public FrequencyCounterDictionary(){
wordTable = new ArrayListDictionary<String, Integer>();
} public void readFile(Scanner data){
data.useDelimiter("\\W+");
while(data.hasNext()){
String nextWord = data.next();//从Scanner打开的输入流中读取单词
nextWord.toLowerCase();
Integer frequency = wordTable.getValue(nextWord);//判断该单词在词典中已出现的次数 if(frequency == null){//单词从未出现过
wordTable.add(nextWord, new Integer(1));//将其出现次数置 1
}
else//单词已经在词典中
{
frequency++;//出现次数增 1
wordTable.add(nextWord, frequency);
}
}//end while
data.close();
} public void display(){
Iterator<String> keyIterator = wordTable.getKeyIterator();
Iterator<Integer> valueIterator = wordTable.getValueIterator();
while(keyIterator.hasNext()){
System.out.println(keyIterator.next() + " : " + valueIterator.next());
}
}
}

运行结果

文本文件内容如下:

词典的实现(3)--使用JAVA类库ArrayList实现Map数据结构的更多相关文章

  1. Java 的 ArrayList 的底层数据结构

    1. 数据结构--ArrayList源码摘要 ublic class ArrayList<E> extends AbstractList<E> implements List& ...

  2. 介绍4款json的java类库 及 其性能测试

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式. 易于人阅读和编写.同时也易于机器解析和生成. 它基于JavaScript Programming Lan ...

  3. 使用Java类库POI生成简易的Excel报表

    使用Java类库POI生成简易的Excel报表 1.需求 1.数据库生成报表需要转义其中字段的信息.比如 1,有效 2.无效等 2.日期格式的自数据需要转义其格式. 3.标题的格式和数据的格式需要分别 ...

  4. java类库 collection与collections (转)

    http://www.cnblogs.com/dashi/p/3597937.html Java中Collection和Collections的区别 1.java.util.Collection 是一 ...

  5. 【thinking in java】ArrayList源码分析

    简介 ArrayList底层是数组实现的,可以自增扩容的数组,此外它是非线程安全的,一般多用于单线程环境下(Vector是线程安全的,所以ArrayList 性能相对Vector 会好些) Array ...

  6. 学习使用Delphi for android 调用Java类库

    http://blog.csdn.net/laorenshen/article/details/41148253 学习使用Delphi for android 调用Java类库 2014-11-15 ...

  7. 解决springmvc报No converter found for return value of type: class java.util.ArrayList问题

    一.背景 最近闲来无事,想自己搭建一套Spring+SpringMVC+Mybatis+Mysql的环境(搭建步骤会在以后博客中给出),结果运行程序时,适用@ResponseBody注解进行返回Lis ...

  8. java 遍历arrayList的四种方法

    package com.test; import java.util.ArrayList;import java.util.Iterator;import java.util.List; public ...

  9. 初涉java库--ArrayList

    我的车就差一个轮子啦,造好轮子,我就飞上天与太阳肩并肩啦,想想都激动.什么你要自己造轮子,是不是傻,商店里不都是别人造好的吗,又好又方便,只需一点money,你没有money,那你只能做个安静的美男子 ...

随机推荐

  1. pandas缺失值处理

    1.检查缺失值 为了更容易地检测缺失值(以及跨越不同的数组dtype),Pandas提供了isnull()和notnull()函数,它们也是Series和DataFrame对象的方法 - 示例1 im ...

  2. mysql学习笔记一 —— 数据的增删改查

    1.连接mysql mysql 直接回车(是以root身份,密码空,登陆的是本机localhost) [root@www mysql]# mysql -uroot -p123 -S /var/lib/ ...

  3. Delphi导出数据的多种方法

    //Dxdbgrid,则直接用SaveToexcel即可//使用 ExcelWithOdbc 控件function TDataModule1.GetDataToFile(DsData: TObject ...

  4. MySQL-如何删除hash表分区

    一个大表,之前是以hash分区表的形式存在的, MySQL> show create table history_uint; | history_uint | CREATE TABLE `his ...

  5. Lodop打印二维码内容长度不同如何大小相同

    利用Loodop打印控件打印二维码的时候,往往传入的数值是变量,有的只有一个数字,有的却一大堆数字和字母,根据内容长度不同,二维码大小也不同,这样如果批量打印二维码标签,传入的数据是不同的,会造成有的 ...

  6. codeforces1045B

    CF1045B 自己瞎鸡巴yy了一下,可知若一个数X不能被表示出来,那么X所有的表示方法都在A集合中,如a1,a2,a3······an-1,an-2中若a1+ai不能被表示出来,那么如果a1到ai是 ...

  7. docker资料---添加阿里docker加速镜像

    首先必须登录阿里云获得加速镜像地址: https://cr.console.aliyun.com/#/accelerator 基于centOS7 cp -n /lib/systemd/system/d ...

  8. 数字对——RMQ+二分答案

    题目描述 小H是个善于思考的学生,现在她又在思考一个有关序列的问题. 她的面前浮现出一个长度为n的序列{ai},她想找出一段区间[L, R](1 <= L <= R <= n). 这 ...

  9. Leading and Trailing LightOJ - 1282 (取数的前三位和后三位)

    题意: 求n的k次方的前三位 和 后三位 ...刚开始用 Java的大数写的...果然超时... 好吧  这题用快速幂取模求后三位  然后用一个技巧求前三位 ...orz... 任何一个数n均可以表示 ...

  10. 自学Aruba1.4-Aruba体系结构-产品线

    点击返回:自学Aruba之路 1. Aruba产品线 IP switches: 1500.2500.3500 Controllers:7200 .70x0 Series.7005 Meridian:基 ...