|                                                                                                              1
 19  package bak.pcj.map;
 20
 21  import bak.pcj.IntCollection;
 22  import bak.pcj.AbstractIntCollection;
 23  import bak.pcj.IntIterator;
 24  import bak.pcj.hash.IntHashFunction;
 25  import bak.pcj.hash.DefaultIntHashFunction;
 26  import bak.pcj.util.Exceptions;
 27  import java.util.Set
  ; 28  import java.util.AbstractSet
  ; 29  import java.util.Iterator
  ; 30
 31  import java.io.Serializable
  ; 32  import java.io.IOException
  ; 33  import java.io.ObjectInputStream
  ; 34  import java.io.ObjectOutputStream
  ; 35
 36
 47  public class ObjectKeyIntOpenHashMap extends AbstractObjectKeyIntMap implements ObjectKeyIntMap, Cloneable
  , Serializable  { 48
 49
 50      private static final int    GROWTH_POLICY_RELATIVE      = 0;
 51
 52
 53      private static final int    GROWTH_POLICY_ABSOLUTE      = 1;
 54
 55
 60      private static final int    DEFAULT_GROWTH_POLICY       = GROWTH_POLICY_RELATIVE;
 61
 62
 63      public static final double DEFAULT_GROWTH_FACTOR        = 1.0;
 64
 65
 66      public static final int    DEFAULT_GROWTH_CHUNK         = 10;
 67
 68
 69      public static final int    DEFAULT_CAPACITY             = 11;
 70
 71
 72      public static final double DEFAULT_LOAD_FACTOR          = 0.75;
 73
 74
 78      private int size;
 79
 80
 85      private transient Object
  [] keys; 86
 87
 92      private transient int[] values;
 93
 94
 95      private transient byte[] states;
 96
 97      private static final byte EMPTY = 0;
 98      private static final byte OCCUPIED = 1;
 99      private static final byte REMOVED = 2;
 100
 101
 102     private transient int used;
 103
 104
 108     private int growthPolicy;
 109
 110
 115     private double growthFactor;
 116
 117
 122     private int growthChunk;
 123
 124
 128     private double loadFactor;
 129
 130
 134     private int expandAt;
 135
 136
 137     private transient Set
  ckeys; 138
 139
 140     private transient IntCollection cvalues;
 141
 142
 143     private transient boolean hasLastValue;
 144
 145
 146     private transient int lastValue;
 147
 148     private ObjectKeyIntOpenHashMap(int capacity, int growthPolicy, double growthFactor, int growthChunk, double loadFactor) {
 149         if (capacity < 0)
 150             Exceptions.negativeArgument("capacity", String.valueOf(capacity));
 151         if (growthFactor <= 0.0)
 152             Exceptions.negativeOrZeroArgument("growthFactor", String.valueOf(growthFactor));
 153         if (growthChunk <= 0)
 154             Exceptions.negativeOrZeroArgument("growthChunk", String.valueOf(growthChunk));
 155         if (loadFactor <= 0.0)
 156             Exceptions.negativeOrZeroArgument("loadFactor", String.valueOf(loadFactor));
 157         capacity = bak.pcj.hash.Primes.nextPrime(capacity);
 158         keys = new Object
  [capacity]; 159         values = new int[capacity];
 160         this.states = new byte[capacity];
 161         size = 0;
 162         expandAt = (int)Math.round(loadFactor*capacity);
 163         this.used = 0;
 164         this.growthPolicy = growthPolicy;
 165         this.growthFactor = growthFactor;
 166         this.growthChunk = growthChunk;
 167         this.loadFactor = loadFactor;
 168         hasLastValue = false;
 169     }
 170
 171
 175     public ObjectKeyIntOpenHashMap() {
 176         this(DEFAULT_CAPACITY);
 177     }
 178
 179
 188     public ObjectKeyIntOpenHashMap(ObjectKeyIntMap map) {
 189         this();
 190         putAll(map);
 191     }
 192
 193
 203     public ObjectKeyIntOpenHashMap(int capacity) {
 204         this(capacity, DEFAULT_GROWTH_POLICY, DEFAULT_GROWTH_FACTOR, DEFAULT_GROWTH_CHUNK, DEFAULT_LOAD_FACTOR);
 205     }
 206
 207
 218     public ObjectKeyIntOpenHashMap(double loadFactor) {
 219         this(DEFAULT_CAPACITY, DEFAULT_GROWTH_POLICY, DEFAULT_GROWTH_FACTOR, DEFAULT_GROWTH_CHUNK, loadFactor);
 220     }
 221
 222
 236     public ObjectKeyIntOpenHashMap(int capacity, double loadFactor) {
 237         this(capacity, DEFAULT_GROWTH_POLICY, DEFAULT_GROWTH_FACTOR, DEFAULT_GROWTH_CHUNK, loadFactor);
 238     }
 239
 240
 263     public ObjectKeyIntOpenHashMap(int capacity, double loadFactor, double growthFactor) {
 264         this(capacity, GROWTH_POLICY_RELATIVE, growthFactor, DEFAULT_GROWTH_CHUNK, loadFactor);
 265     }
 266
 267
 290     public ObjectKeyIntOpenHashMap(int capacity, double loadFactor, int growthChunk) {
 291         this(capacity, GROWTH_POLICY_ABSOLUTE, DEFAULT_GROWTH_FACTOR, growthChunk, loadFactor);
 292     }
 293
 294
 298     private static int hash(Object
  key) 299     { return key == null ? 0 : key.hashCode(); }
 300
 301     private static boolean keyeq(Object
  k1, Object  k2) 302     { return k1 == null ? k2 == null : k1.equals(k2); }
 303
 304     private void ensureCapacity(int elements) {
 305         if (elements >= expandAt) {
 306             int newcapacity;
 307             if (growthPolicy == GROWTH_POLICY_RELATIVE)
 308                 newcapacity = (int)(keys.length * (1.0 + growthFactor));
 309             else
 310                 newcapacity = keys.length + growthChunk;
 311             if (newcapacity*loadFactor < elements)
 312                 newcapacity = (int)Math.round(((double)elements/loadFactor));
 313             newcapacity = bak.pcj.hash.Primes.nextPrime(newcapacity);
 314             expandAt = (int)Math.round(loadFactor*newcapacity);
 315
 316             Object
  [] newkeys = new Object  [newcapacity]; 317             int[] newvalues = new int[newcapacity];
 318             byte[] newstates = new byte[newcapacity];
 319
 320             used = 0;
 321                         for (int i = 0; i < keys.length; i++) {
 323                 if (states[i] == OCCUPIED) {
 324                     used++;
 325                     Object
  k = keys[i]; 326                     int v = values[i];
 327                                         int h = Math.abs(hash(k));
 329                     int n = h % newcapacity;
 330                     if (newstates[n] == OCCUPIED) {
 331                                                 int c = 1 + (h % (newcapacity - 2));
 333                         for (;;) {
 334                             n -= c;
 335                             if (n < 0)
 336                                 n += newcapacity;
 337                             if (newstates[n] == EMPTY)
 338                                 break;
 339                         }
 340                     }
 341                     newstates[n] = OCCUPIED;
 342                     newvalues[n] = v;
 343                     newkeys[n] = k;
 344                 }
 345             }
 346
 347             keys = newkeys;
 348             values = newvalues;
 349             states = newstates;
 350         }
 351     }
 352
 353
 357     public Set
  keySet() { 358         if (ckeys == null)
 359             ckeys = new KeySet();
 360         return ckeys;
 361     }
 362
 363     public int lget() {
 364         if (!hasLastValue)
 365             Exceptions.noLastElement();
 366         return lastValue;
 367     }
 368
 369     public int put(Object
  key, int value) { 370         int result;
 371
 372                 int h = Math.abs(hash(key));
 374         int i = h % keys.length;
 375         if (states[i] == OCCUPIED) {
 376             if (keyeq(keys[i], key)) {
 377                 int oldValue = values[i];
 378                 values[i] = value;
 379                 return oldValue;
 380             }
 381                         int c = 1 + (h % (keys.length - 2));
 383             for (;;) {
 384                 i -= c;
 385                 if (i < 0)
 386                     i += keys.length;
 387                                 if (states[i] == EMPTY || states[i] == REMOVED)
 389                     break;
 390                 if (states[i] == OCCUPIED && keyeq(keys[i], key)) {
 391                     int oldValue = values[i];
 392                     values[i] = value;
 393                     return oldValue;
 394                 }
 395             }
 396         }
 397         if (states[i] == EMPTY)
 398             used++;
 399         states[i] = OCCUPIED;
 400         keys[i] = key;
 401         values[i] = value;
 402         size++;
 403         ensureCapacity(used);
 404         return MapDefaults.defaultInt();
 405     }
 406
 407     public IntCollection values() {
 408         if (cvalues == null)
 409             cvalues = new ValueCollection();
 410         return cvalues;
 411     }
 412
 413
 420     public Object
  clone() { 421         try {
 422             ObjectKeyIntOpenHashMap c = (ObjectKeyIntOpenHashMap)super.clone();
 423             c.keys = new Object
  [keys.length]; 424             System.arraycopy(keys, 0, c.keys, 0, keys.length);
 425             c.values = new int[values.length];
 426             System.arraycopy(values, 0, c.values, 0, values.length);
 427             c.states = new byte[states.length];
 428             System.arraycopy(states, 0, c.states, 0, states.length);
 429                         c.cvalues = null;
 431             c.ckeys = null;
 432             return c;
 433         } catch (CloneNotSupportedException
  e) { 434             Exceptions.cloning(); return null;
 435         }
 436     }
 437
 438     public ObjectKeyIntMapIterator entries() {
 439         return new ObjectKeyIntMapIterator() {
 440             int nextEntry = nextEntry(0);
 441             int lastEntry = -1;
 442
 443             int nextEntry(int index) {
 444                 while (index < keys.length && states[index] != OCCUPIED)
 445                     index++;
 446                 return index;
 447             }
 448
 449             public boolean hasNext() {
 450                 return nextEntry < keys.length;
 451             }
 452
 453             public void next() {
 454                 if (!hasNext())
 455                     Exceptions.endOfIterator();
 456                 lastEntry = nextEntry;
 457                 nextEntry = nextEntry(nextEntry+1);
 458             }
 459
 460             public void remove() {
 461                 if (lastEntry == -1)
 462                     Exceptions.noElementToRemove();
 463                 states[lastEntry] = REMOVED;
 464                 size--;
 465                 lastEntry = -1;
 466             }
 467
 468             public Object
  getKey() { 469                 if (lastEntry == -1)
 470                     Exceptions.noElementToGet();
 471                 return keys[lastEntry];
 472             }
 473
 474             public int getValue() {
 475                 if (lastEntry == -1)
 476                     Exceptions.noElementToGet();
 477                 return values[lastEntry];
 478             }
 479         };
 480     }
 481
 482     private class KeySet extends AbstractSet
  { 483
 484         public void clear()
 485         { ObjectKeyIntOpenHashMap.this.clear(); }
 486
 487         public boolean contains(Object
  v) { 488             return containsKey(v);
 489         }
 490
 491         public Iterator iterator() {
 492             return new Iterator() {
 493                 int nextEntry = nextEntry(0);
 494                 int lastEntry = -1;
 495
 496                 int nextEntry(int index) {
 497                     while (index < keys.length && states[index] != OCCUPIED)
 498                         index++;
 499                     return index;
 500                 }
 501
 502                 public boolean hasNext() {
 503                     return nextEntry < keys.length;
 504                 }
 505
 506                 public Object
  next() { 507                     if (!hasNext())
 508                         Exceptions.endOfIterator();
 509                     lastEntry = nextEntry;
 510                     nextEntry = nextEntry(nextEntry+1);
 511                     return keys[lastEntry];
 512                 }
 513
 514                 public void remove() {
 515                     if (lastEntry == -1)
 516                         Exceptions.noElementToRemove();
 517                     states[lastEntry] = REMOVED;
 518                     keys[lastEntry] = null;                     size--;
 520                     lastEntry = -1;
 521                 }
 522             };
 523         }
 524
 525         public boolean remove(Object
  v) { 526             boolean result = containsKey(v);
 527             if (result)
 528                 ObjectKeyIntOpenHashMap.this.remove(v);
 529             return result;
 530         }
 531
 532         public int size()
 533         { return size; }
 534
 535     }
 536
 537
 538     private class ValueCollection extends AbstractIntCollection {
 539
 540         public void clear()
 541         { ObjectKeyIntOpenHashMap.this.clear(); }
 542
 543         public boolean contains(int v) {
 544             return containsValue(v);
 545         }
 546
 547         public IntIterator iterator() {
 548             return new IntIterator() {
 549                 int nextEntry = nextEntry(0);
 550                 int lastEntry = -1;
 551
 552                 int nextEntry(int index) {
 553                     while (index < keys.length && states[index] != OCCUPIED)
 554                         index++;
 555                     return index;
 556                 }
 557
 558                 public boolean hasNext() {
 559                     return nextEntry < keys.length;
 560                 }
 561
 562                 public int next() {
 563                     if (!hasNext())
 564                         Exceptions.endOfIterator();
 565                     lastEntry = nextEntry;
 566                     nextEntry = nextEntry(nextEntry+1);
 567                     return values[lastEntry];
 568                 }
 569
 570                 public void remove() {
 571                     if (lastEntry == -1)
 572                         Exceptions.noElementToRemove();
 573                     states[lastEntry] = REMOVED;
 574                     size--;
 575                     lastEntry = -1;
 576                 }
 577             };
 578         }
 579
 580         public int size()
 581         { return size; }
 582
 583     }
 584
 585
 589     public void clear() {
 590         java.util.Arrays.fill(states, EMPTY);
 591         java.util.Arrays.fill(keys, null);         size = 0;
 593         used = 0;
 594     }
 595
 596     public boolean containsKey(Object
  key) { 597         int h = Math.abs(hash(key));
 598         int i = h % keys.length;
 599         if (states[i] != EMPTY) {
 600             if (states[i] == OCCUPIED && keyeq(keys[i], key)) {
 601                 hasLastValue = true;
 602                 lastValue = values[i];
 603                 return true;
 604             }
 605
 607             int c = 1 + (h % (keys.length - 2));
 608             for (;;) {
 609                 i -= c;
 610                 if (i < 0)
 611                     i += keys.length;
 612                 if (states[i] == EMPTY) {
 613                     hasLastValue = false;
 614                     return false;
 615                 }
 616                 if (states[i] == OCCUPIED && keyeq(keys[i], key)) {
 617                     hasLastValue = true;
 618                     lastValue = values[i];
 619                     return true;
 620                 }
 621             }
 622         }
 623         hasLastValue = false;
 624         return false;
 625     }
 626
 627     public boolean containsValue(int value) {
 628         for (int i = 0; i < states.length; i++)
 629             if (states[i] == OCCUPIED && values[i] == value)
 630                 return true;
 631         return false;
 632     }
 633
 634     public int get(Object
  key) { 635         int h = Math.abs(hash(key));
 636         int i = h % keys.length;
 637         if (states[i] != EMPTY) {
 638             if (states[i] == OCCUPIED && keyeq(keys[i], key))
 639                 return values[i];
 640
 642             int c = 1 + (h % (keys.length - 2));
 643             for (;;) {
 644                 i -= c;
 645                 if (i < 0)
 646                     i += keys.length;
 647                 if (states[i] == EMPTY)
 648                     return MapDefaults.defaultInt();
 649                 if (states[i] == OCCUPIED && keyeq(keys[i], key))
 650                     return values[i];
 651             }
 652         }
 653         return MapDefaults.defaultInt();
 654     }
 655
 656     public boolean isEmpty()
 657     { return size == 0; }
 658
 659     public int remove(Object
  key) { 660         int h = Math.abs(hash(key));
 661         int i = h % keys.length;
 662         if (states[i] != EMPTY) {
 663             if (states[i] == OCCUPIED && keyeq(keys[i], key)) {
 664                 int oldValue = values[i];
 665                 states[i] = REMOVED;
 666                 keys[i] = null;                 size--;
 668                 return oldValue;
 669             }
 670                         int c = 1 + (h % (keys.length - 2));
 672             for (;;) {
 673                 i -= c;
 674                 if (i < 0)
 675                     i += keys.length;
 676                 if (states[i] == EMPTY) {
 677                     return MapDefaults.defaultInt();
 678                 }
 679                 if (states[i] == OCCUPIED && keyeq(keys[i], key)) {
 680                     int oldValue = values[i];
 681                     states[i] = REMOVED;
 682                     keys[i] = null;                     size--;
 684                     return oldValue;
 685                 }
 686             }
 687         }
 688         return MapDefaults.defaultInt();
 689     }
 690
 691     public int size()
 692     { return size; }
 693
 694     public int tget(Object
  key) { 695         int h = Math.abs(hash(key));
 696         int i = h % keys.length;
 697         if (states[i] != EMPTY) {
 698             if (states[i] == OCCUPIED && keyeq(keys[i], key))
 699                 return values[i];
 700
 702             int c = 1 + (h % (keys.length - 2));
 703             for (;;) {
 704                 i -= c;
 705                 if (i < 0)
 706                     i += keys.length;
 707                 if (states[i] == EMPTY)
 708                     Exceptions.noSuchMapping(key);
 709                 if (states[i] == OCCUPIED && keyeq(keys[i], key))
 710                     return values[i];
 711             }
 712         }
 713         Exceptions.noSuchMapping(key); throw new RuntimeException
  (); 714     }
 715
 716
 720
 725     private void writeObject(ObjectOutputStream
  s) throws IOException  { 726         s.defaultWriteObject();
 727         s.writeInt(keys.length);
 728         ObjectKeyIntMapIterator i = entries();
 729         while (i.hasNext()) {
 730             i.next();
 731             s.writeObject(i.getKey());
 732             s.writeInt(i.getValue());
 733         }
 734     }
 735
 736     private void readObject(ObjectInputStream
  s) throws IOException  , ClassNotFoundException  { 737         s.defaultReadObject();
 738         keys = new Object
  [s.readInt()]; 739         states = new byte[keys.length];
 740         values = new int[keys.length];
 741         used = size;
 742
 743         for (int n = 0; n < size; n++) {
 744             Object
  key = s.readObject(); 745             int value = s.readInt();
 746
 747                         int h = Math.abs(hash(key));
 749             int i = h % keys.length;
 750             if (states[i] != EMPTY) {
 751                                 int c = 1 + (h % (keys.length - 2));
 753                 for (;;) {
 754                     i -= c;
 755                     if (i < 0)
 756                         i += keys.length;
 757                     if (states[i] == EMPTY)
 758                         break;
 759                 }
 760             }
 761             states[i] = OCCUPIED;
 762             keys[i] = key;
 763             values[i] = value;
 764         }
 765     }
 766
 767 }
                                                                                                                                                                                                             |                                                                       
 
 
 
 
 
                                                                                   Popular Tags                                                                                                                                                                                              |