-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMapLayerFileCache.java
executable file
·192 lines (174 loc) · 5.88 KB
/
MapLayerFileCache.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
// MapLayerFileCache.java implements a class to cache file contents for
// map layer files. One MapLayerFileCache can be used for each type of
// map layer (as needed).
//---------------------------------------------------------------------
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Vector;
public class MapLayerFileCache
{
class CacheEntry
{
Vector data;
int age;
boolean loaded;
}
private Hashtable cache;
// constructor for the MapLayerFileCache
//--------------------------------------
public MapLayerFileCache()
{
cache = new Hashtable(4);
}
// method to age the cache contents
//---------------------------------
public void age()
{
// for each entry in the hash table, increase the age
Enumeration keys = cache.keys();
while (keys.hasMoreElements())
{
String filename = (String)keys.nextElement();
CacheEntry entry = (CacheEntry)cache.get(filename);
// ages don't mean much after 100, so limit the aging to 100
if (entry.age < 100)
entry.age++;
}
}
// method to add a filename to the cache. If the file is already in the
// cache, the age will be reset. Returns "true" if the file will need
// to be loaded.
//----------------------------------------------------------------------
public boolean addFile(String filename)
{
CacheEntry entry = (CacheEntry)cache.get(filename);
if (entry != null)
{
// the file is in the cache, so reset its age (don't need to
// put it back in the cache since the reference to it is already
// in the cache)
entry.age = 0;
}
else
{
// file isn't in cache at all, so add it
entry = new MapLayerFileCache.CacheEntry();
entry.data = null;
entry.age = 0;
entry.loaded = false;
cache.put(filename, entry);
}
return !entry.loaded;
}
// method to purge old cache data
//-------------------------------
public void purge()
{
while (cache.size() > 4)
{
int oldest = -1;
String oldestKey = null;
// look for the oldest cache entry to purge
Enumeration keys = cache.keys();
while (keys.hasMoreElements())
{
String filename = (String)keys.nextElement();
CacheEntry entry = (CacheEntry)cache.get(filename);
if (entry.age >= oldest)
{
oldest = entry.age;
oldestKey = filename;
}
}
// if the oldest entry is current, don't remove it since more
// than 4 files need to be cached to keep the active set available
if (oldest == 0)
break;
cache.remove(oldestKey);
}
}
// method to return the files that need to be loaded
//--------------------------------------------------
String[] getFilesToLoad()
{
String[] files = new String[cache.size()];
int i = 0;
// look for the cache entries from the current generation that
// have not been loaded yet
Enumeration keys = cache.keys();
while (keys.hasMoreElements())
{
String filename = (String)keys.nextElement();
CacheEntry entry = (CacheEntry)cache.get(filename);
if ((entry.age == 0) && !entry.loaded)
{
// needed and not yet loaded
files[i] = filename;
i++;
}
}
if (i == 0)
{
// nothing to load, so just return null
return null;
}
else
{
// files were found that need loading
String[] returnList = new String[i];
for (int j = 0; j < i; j++)
returnList[j] = files[j];
return returnList;
}
}
// set the data for a cache element
//---------------------------------
void setCacheContents(String filename, Vector data)
{
CacheEntry entry = (CacheEntry)cache.get(filename);
entry.data = data;
entry.loaded = true;
}
// method to return the data for the current cache contents
//---------------------------------------------------------
Vector getCachedData()
{
Vector data = new Vector();
// sort the keys in the cache alphabeticallys so that all the order
// of the data returned is consistent (this fixes problems with the
// displayed cities sometimes changing around based on the order of
// the data given to it)
int size = cache.size();
int count = 0;
String[] filenames = new String[size];
Enumeration keys = cache.keys();
while (keys.hasMoreElements())
{
String filename = (String)keys.nextElement();
int insert_index;
for (insert_index = 0; insert_index < count; insert_index++)
{
if (filename.compareTo(filenames[insert_index]) > 0)
break;
}
for (int i = count; i > insert_index; i--)
{
filenames[i] = filenames[i-1];
}
filenames[insert_index] = filename;
count++;
}
// look for the cache entries from the current generation
for (int i = 0; i < count; i++)
{
String filename = filenames[i];
CacheEntry entry = (CacheEntry)cache.get(filename);
if ((entry.age == 0) && entry.loaded && (entry.data != null))
{
// needed and not yet loaded
data.addElement(entry.data);
}
}
return data;
}
}