/Ice-3.4.2/java/src/Freeze/MapInternal/SubMap.java

# · Java · 695 lines · 556 code · 89 blank · 50 comment · 77 complexity · a7bd2b45e9b25132b62ff983a9509f04 MD5 · raw file

  1. // **********************************************************************
  2. //
  3. // Copyright (c) 2003-2011 ZeroC, Inc. All rights reserved.
  4. //
  5. // This copy of Ice is licensed to you under the terms described in the
  6. // ICE_LICENSE file included in this distribution.
  7. //
  8. // **********************************************************************
  9. package Freeze.MapInternal;
  10. import Freeze.ConnectionI;
  11. import Freeze.Map;
  12. import Freeze.NavigableMap;
  13. //
  14. // Submap of a Freeze Map or of another submap
  15. //
  16. class SubMap<K, V> extends java.util.AbstractMap<K, V> implements NavigableMap<K, V>
  17. {
  18. SubMap(MapI<K, V> map, K fromKey, boolean fromInclusive, K toKey, boolean toInclusive, boolean ascending)
  19. {
  20. _map = map;
  21. if(ascending)
  22. {
  23. _view = new AscendingView(fromKey, fromInclusive, toKey, toInclusive);
  24. }
  25. else
  26. {
  27. _view = new DescendingView(fromKey, fromInclusive, toKey, toInclusive);
  28. }
  29. }
  30. private
  31. SubMap(MapI<K, V> map, View v)
  32. {
  33. _map = map;
  34. _view = v;
  35. }
  36. //
  37. // NavigableMap methods
  38. //
  39. public boolean
  40. fastRemove(K key)
  41. {
  42. if(!_view.inRange(key, true))
  43. {
  44. return false;
  45. }
  46. return _map.fastRemove(key);
  47. }
  48. public java.util.Map.Entry<K, V>
  49. firstEntry()
  50. {
  51. return _view.first();
  52. }
  53. public java.util.Map.Entry<K, V>
  54. lastEntry()
  55. {
  56. return _view.last();
  57. }
  58. public java.util.Map.Entry<K, V>
  59. ceilingEntry(K key)
  60. {
  61. return _view.ceiling(key);
  62. }
  63. public java.util.Map.Entry<K, V>
  64. floorEntry(K key)
  65. {
  66. return _view.floor(key);
  67. }
  68. public java.util.Map.Entry<K, V>
  69. higherEntry(K key)
  70. {
  71. return _view.higher(key);
  72. }
  73. public java.util.Map.Entry<K, V>
  74. lowerEntry(K key)
  75. {
  76. return _view.lower(key);
  77. }
  78. public K
  79. ceilingKey(K key)
  80. {
  81. EntryI<K, V> e = _view.ceiling(key);
  82. return e != null ? e.getKey() : null;
  83. }
  84. public K
  85. floorKey(K key)
  86. {
  87. EntryI<K, V> e = _view.floor(key);
  88. return e != null ? e.getKey() : null;
  89. }
  90. public K
  91. higherKey(K key)
  92. {
  93. EntryI<K, V> e = _view.higher(key);
  94. return e != null ? e.getKey() : null;
  95. }
  96. public K
  97. lowerKey(K key)
  98. {
  99. EntryI<K, V> e = _view.lower(key);
  100. return e != null ? e.getKey() : null;
  101. }
  102. public java.util.Set<K>
  103. descendingKeySet()
  104. {
  105. return descendingMap().keySet();
  106. }
  107. public NavigableMap<K, V>
  108. descendingMap()
  109. {
  110. if(_descendingMap == null)
  111. {
  112. View v = _view.descendingView();
  113. _descendingMap = new SubMap<K, V>(_map, v);
  114. }
  115. return _descendingMap;
  116. }
  117. public NavigableMap<K, V>
  118. headMap(K toKey, boolean inclusive)
  119. {
  120. if(toKey == null)
  121. {
  122. throw new NullPointerException();
  123. }
  124. View v = _view.subView(null, false, toKey, inclusive);
  125. return new SubMap<K, V>(_map, v);
  126. }
  127. public NavigableMap<K, V>
  128. subMap(K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  129. {
  130. if(fromKey == null || toKey == null)
  131. {
  132. throw new NullPointerException();
  133. }
  134. View v = _view.subView(fromKey, fromInclusive, toKey, toInclusive);
  135. return new SubMap<K, V>(_map, v);
  136. }
  137. public NavigableMap<K, V>
  138. tailMap(K fromKey, boolean inclusive)
  139. {
  140. if(fromKey == null)
  141. {
  142. throw new NullPointerException();
  143. }
  144. View v = _view.subView(fromKey, inclusive, null, false);
  145. return new SubMap<K, V>(_map, v);
  146. }
  147. public java.util.Map.Entry<K, V>
  148. pollFirstEntry()
  149. {
  150. EntryI<K, V> e = _view.first();
  151. if(e != null)
  152. {
  153. _map.removeImpl(e.getDbKey());
  154. }
  155. return e;
  156. }
  157. public java.util.Map.Entry<K, V>
  158. pollLastEntry()
  159. {
  160. EntryI<K, V> e = _view.last();
  161. if(e != null)
  162. {
  163. _map.removeImpl(e.getDbKey());
  164. }
  165. return e;
  166. }
  167. //
  168. // SortedMap methods
  169. //
  170. public java.util.Comparator<? super K>
  171. comparator()
  172. {
  173. return _view.comparator();
  174. }
  175. public K
  176. firstKey()
  177. {
  178. EntryI<K, V> e = _view.first();
  179. if(e == null)
  180. {
  181. throw new java.util.NoSuchElementException();
  182. }
  183. return e.getKey();
  184. }
  185. public K
  186. lastKey()
  187. {
  188. EntryI<K, V> e = _view.last();
  189. if(e == null)
  190. {
  191. throw new java.util.NoSuchElementException();
  192. }
  193. return e.getKey();
  194. }
  195. public java.util.SortedMap<K, V>
  196. headMap(K toKey)
  197. {
  198. return headMap(toKey, false);
  199. }
  200. public java.util.SortedMap<K, V>
  201. tailMap(K fromKey)
  202. {
  203. return tailMap(fromKey, true);
  204. }
  205. public java.util.SortedMap<K, V>
  206. subMap(K fromKey, K toKey)
  207. {
  208. return subMap(fromKey, true, toKey, false);
  209. }
  210. //
  211. // Map methods
  212. //
  213. public java.util.Set<java.util.Map.Entry<K, V>>
  214. entrySet()
  215. {
  216. if(_entrySet == null)
  217. {
  218. _entrySet = new java.util.AbstractSet<java.util.Map.Entry<K, V>>()
  219. {
  220. public java.util.Iterator<java.util.Map.Entry<K, V>>
  221. iterator()
  222. {
  223. return new IteratorI<K, V>(_map, _view);
  224. }
  225. public boolean
  226. contains(Object o)
  227. {
  228. //
  229. // If the main map contains this object, verify it's within the range of this submap.
  230. //
  231. if(_map.entrySet().contains(o))
  232. {
  233. @SuppressWarnings("unchecked")
  234. EntryI<K, V> entry = (EntryI<K, V>)o;
  235. return _view.inRange(entry.getKey(), true);
  236. }
  237. else
  238. {
  239. return false;
  240. }
  241. }
  242. public boolean
  243. remove(Object o)
  244. {
  245. if(o instanceof EntryI)
  246. {
  247. @SuppressWarnings("unchecked")
  248. EntryI<K, V> entry = (EntryI<K, V>)o;
  249. return _view.inRange(entry.getKey(), true) && _map.entrySet().remove(o);
  250. }
  251. else
  252. {
  253. return false;
  254. }
  255. }
  256. public int
  257. size()
  258. {
  259. throw new UnsupportedOperationException();
  260. }
  261. public boolean
  262. isEmpty()
  263. {
  264. try
  265. {
  266. firstKey();
  267. return false;
  268. }
  269. catch(java.util.NoSuchElementException e)
  270. {
  271. return true;
  272. }
  273. }
  274. };
  275. }
  276. return _entrySet;
  277. }
  278. //
  279. // Put is not implemented (you have to put in the main map view)
  280. //
  281. public boolean
  282. constainsKey(Object key)
  283. {
  284. @SuppressWarnings("unchecked")
  285. K k = (K)key;
  286. if(!_view.inRange(k, true))
  287. {
  288. return false;
  289. }
  290. return _map.containsKey(k);
  291. }
  292. public V
  293. get(Object key)
  294. {
  295. @SuppressWarnings("unchecked")
  296. K k = (K)key;
  297. if(!_view.inRange(k, true))
  298. {
  299. return null;
  300. }
  301. return _map.get(k);
  302. }
  303. public V
  304. remove(Object key)
  305. {
  306. @SuppressWarnings("unchecked")
  307. K k = (K)key;
  308. if(!_view.inRange(k, true))
  309. {
  310. return null;
  311. }
  312. return _map.remove(k);
  313. }
  314. private abstract class View implements IteratorModel<K, V>, Search.KeyValidator
  315. {
  316. protected
  317. View(java.util.Comparator<? super K> comparator, K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  318. {
  319. _comparator = comparator;
  320. _fromKey = fromKey;
  321. _fromInclusive = fromInclusive;
  322. _toKey = toKey;
  323. _toInclusive = toInclusive;
  324. //
  325. // Validate the key range.
  326. //
  327. if(_fromKey != null && _toKey != null)
  328. {
  329. int cmp = comparator.compare(_fromKey, _toKey);
  330. if(cmp > 0 || (cmp == 0 && !(_fromInclusive && _toInclusive)))
  331. {
  332. throw new IllegalArgumentException();
  333. }
  334. }
  335. }
  336. protected
  337. View(View v, java.util.Comparator<? super K> comparator, K fromKey, boolean fromInclusive, K toKey,
  338. boolean toInclusive)
  339. {
  340. this(comparator, fromKey, fromInclusive, toKey, toInclusive);
  341. //
  342. // Verify that the key range is correct with respect to the original view.
  343. //
  344. if(!v.inRange(_fromKey, _fromInclusive) || !v.inRange(_toKey, _toInclusive))
  345. {
  346. throw new IllegalArgumentException();
  347. }
  348. }
  349. abstract Search.Type mapSearchType(Search.Type type);
  350. abstract View copy(K fromKey, boolean fromInclusive, K toKey, boolean toInclusive);
  351. abstract View descendingView();
  352. final EntryI<K, V>
  353. first()
  354. {
  355. Search.Type type;
  356. byte[] key = null;
  357. if(_fromKey != null)
  358. {
  359. type = _fromInclusive ? mapSearchType(Search.Type.CEILING) : mapSearchType(Search.Type.HIGHER);
  360. key = fromKeyBytes();
  361. }
  362. else
  363. {
  364. type = mapSearchType(Search.Type.FIRST);
  365. }
  366. return _map.entrySearch(type, key, true, this);
  367. }
  368. final EntryI<K, V>
  369. last()
  370. {
  371. Search.Type type;
  372. byte[] key = null;
  373. if(_toKey != null)
  374. {
  375. type = _toInclusive ? mapSearchType(Search.Type.FLOOR) : mapSearchType(Search.Type.LOWER);
  376. key = toKeyBytes();
  377. }
  378. else
  379. {
  380. type = mapSearchType(Search.Type.LAST);
  381. }
  382. return _map.entrySearch(type, key, true, this);
  383. }
  384. final EntryI<K, V>
  385. ceiling(K key)
  386. {
  387. byte[] k = _map.encodeKey(key, _map.connection().getCommunicator());
  388. return _map.entrySearch(mapSearchType(Search.Type.CEILING), k, true, this);
  389. }
  390. final EntryI<K, V>
  391. floor(K key)
  392. {
  393. byte[] k = _map.encodeKey(key, _map.connection().getCommunicator());
  394. return _map.entrySearch(mapSearchType(Search.Type.FLOOR), k, true, this);
  395. }
  396. final EntryI<K, V>
  397. higher(K key)
  398. {
  399. byte[] k = _map.encodeKey(key, _map.connection().getCommunicator());
  400. return _map.entrySearch(mapSearchType(Search.Type.HIGHER), k, true, this);
  401. }
  402. final EntryI<K, V>
  403. lower(K key)
  404. {
  405. byte[] k = _map.encodeKey(key, _map.connection().getCommunicator());
  406. return _map.entrySearch(mapSearchType(Search.Type.LOWER), k, true, this);
  407. }
  408. final View
  409. subView(K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  410. {
  411. if(fromKey == null)
  412. {
  413. fromKey = _fromKey;
  414. fromInclusive = _fromInclusive;
  415. }
  416. if(toKey == null)
  417. {
  418. toKey = _toKey;
  419. toInclusive = _toInclusive;
  420. }
  421. return copy(fromKey, fromInclusive, toKey, toInclusive);
  422. }
  423. //
  424. // IteratorModel methods (partial)
  425. //
  426. final public String
  427. dbName()
  428. {
  429. return _map.dbName();
  430. }
  431. final public TraceLevels
  432. traceLevels()
  433. {
  434. return _map.traceLevels();
  435. }
  436. final public com.sleepycat.db.Cursor
  437. openCursor()
  438. throws com.sleepycat.db.DatabaseException
  439. {
  440. return _map.openCursor();
  441. }
  442. //
  443. // Search.KeyValidator methods
  444. //
  445. final public boolean
  446. keyInRange(byte[] key)
  447. {
  448. K k = _map.decodeKey(key, _map.connection().getCommunicator());
  449. return inRange(k, true);
  450. }
  451. final boolean
  452. inRange(K key, boolean inclusive)
  453. {
  454. return !tooLow(key, inclusive, _fromKey, _fromInclusive) &&
  455. !tooHigh(key, inclusive, _toKey, _toInclusive);
  456. }
  457. final java.util.Comparator<? super K>
  458. comparator()
  459. {
  460. return _comparator;
  461. }
  462. final protected byte[]
  463. fromKeyBytes()
  464. {
  465. if(_fromKey != null && _fromKeyBytes == null)
  466. {
  467. _fromKeyBytes = _map.encodeKey(_fromKey, _map.connection().getCommunicator());
  468. }
  469. return _fromKeyBytes;
  470. }
  471. final protected byte[]
  472. toKeyBytes()
  473. {
  474. if(_toKey != null && _toKeyBytes == null)
  475. {
  476. _toKeyBytes = _map.encodeKey(_toKey, _map.connection().getCommunicator());
  477. }
  478. return _toKeyBytes;
  479. }
  480. final protected boolean
  481. tooLow(K key, boolean inclusive, K targetKey, boolean targetInclusive)
  482. {
  483. if(key != null && targetKey != null)
  484. {
  485. int cmp = comparator().compare(key, targetKey);
  486. if(cmp < 0 || (cmp == 0 && inclusive && !targetInclusive))
  487. {
  488. return true;
  489. }
  490. }
  491. return false;
  492. }
  493. final protected boolean
  494. tooHigh(K key, boolean inclusive, K targetKey, boolean targetInclusive)
  495. {
  496. if(key != null && targetKey != null)
  497. {
  498. int cmp = comparator().compare(key, targetKey);
  499. if(cmp > 0 || (cmp == 0 && inclusive && !targetInclusive))
  500. {
  501. return true;
  502. }
  503. }
  504. return false;
  505. }
  506. final java.util.Comparator<? super K> _comparator;
  507. final K _fromKey;
  508. final boolean _fromInclusive;
  509. final K _toKey;
  510. final boolean _toInclusive;
  511. private byte[] _fromKeyBytes;
  512. private byte[] _toKeyBytes;
  513. }
  514. private class AscendingView extends View
  515. {
  516. AscendingView(K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  517. {
  518. super(_map.comparator(), fromKey, fromInclusive, toKey, toInclusive);
  519. }
  520. AscendingView(View v, K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  521. {
  522. super(v, _map.comparator(), fromKey, fromInclusive, toKey, toInclusive);
  523. }
  524. //
  525. // View methods
  526. //
  527. Search.Type
  528. mapSearchType(Search.Type type)
  529. {
  530. return type;
  531. }
  532. View
  533. copy(K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  534. {
  535. return new AscendingView(this, fromKey, fromInclusive, toKey, toInclusive);
  536. }
  537. View
  538. descendingView()
  539. {
  540. return new DescendingView(this, _toKey, _toInclusive, _fromKey, _fromInclusive);
  541. }
  542. //
  543. // IteratorModel methods
  544. //
  545. public EntryI<K, V>
  546. firstEntry(com.sleepycat.db.Cursor cursor)
  547. throws com.sleepycat.db.DatabaseException
  548. {
  549. return _map.firstEntry(cursor, _fromKey, _fromInclusive, _toKey, _toInclusive);
  550. }
  551. public EntryI<K, V>
  552. nextEntry(com.sleepycat.db.Cursor cursor)
  553. throws com.sleepycat.db.DatabaseException
  554. {
  555. return _map.nextEntry(cursor, _toKey, _toInclusive);
  556. }
  557. }
  558. private class DescendingView extends View
  559. {
  560. DescendingView(K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  561. {
  562. super(java.util.Collections.reverseOrder(_map.comparator()), fromKey, fromInclusive, toKey, toInclusive);
  563. }
  564. DescendingView(View v, K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  565. {
  566. super(v, java.util.Collections.reverseOrder(_map.comparator()), fromKey, fromInclusive, toKey, toInclusive);
  567. }
  568. //
  569. // View methods
  570. //
  571. Search.Type
  572. mapSearchType(Search.Type type)
  573. {
  574. return type.descending();
  575. }
  576. View
  577. copy(K fromKey, boolean fromInclusive, K toKey, boolean toInclusive)
  578. {
  579. return new DescendingView(this, fromKey, fromInclusive, toKey, toInclusive);
  580. }
  581. View
  582. descendingView()
  583. {
  584. return new AscendingView(this, _toKey, _toInclusive, _fromKey, _fromInclusive);
  585. }
  586. //
  587. // IteratorModel methods
  588. //
  589. public EntryI<K, V>
  590. firstEntry(com.sleepycat.db.Cursor cursor)
  591. throws com.sleepycat.db.DatabaseException
  592. {
  593. return _map.lastEntry(cursor, _fromKey, _fromInclusive, _toKey, _toInclusive);
  594. }
  595. public EntryI<K, V>
  596. nextEntry(com.sleepycat.db.Cursor cursor)
  597. throws com.sleepycat.db.DatabaseException
  598. {
  599. return _map.previousEntry(cursor, _toKey, _toInclusive);
  600. }
  601. }
  602. private final MapI<K, V> _map;
  603. private final View _view;
  604. private java.util.Set<java.util.Map.Entry<K, V>> _entrySet;
  605. private NavigableMap<K, V> _descendingMap;
  606. }