brouter/brouter-util/src/main/java/btools/util/DenseLongMap.java

218 lines
5.4 KiB
Java

package btools.util;
import java.util.ArrayList;
/**
* Special Memory efficient Map to map a long-key to
* a "small" value (some bits only) where it is expected
* that the keys are dense, so that we can use more or less
* a simple array as the best-fit data model (except for
* the 32-bit limit of arrays!)
*
* Target application are osm-node ids which are in the
* range 0...3 billion and basically dense (=only few
* nodes deleted)
*
* @author ab
*/
public class DenseLongMap
{
private ArrayList<byte[]> blocklist = new ArrayList<byte[]>(4096);
private int blocksize; // bytes per bitplane in one block
private int blocksizeBits;
private long blocksizeBitsMask;
private int maxvalue = 254; // fixed due to 8 bit lookup table
private int[] bitplaneCount = new int[8];
private long putCount = 0L;
private long getCount = 0L;
/**
* Creates a DenseLongMap for the default block size
* ( 512 bytes per bitplane, covering a key range of 4096 keys )
* Note that one value range is limited to 0..254
*
* @param valuebits number of bits to use per value
*/
public DenseLongMap()
{
this(512);
}
/**
* Creates a DenseLongMap for the given block size
*
* @param blocksize bytes per bit-plane
*/
public DenseLongMap( int blocksize )
{
int bits = 4;
while( bits < 28 && (1 << bits) != blocksize )
{
bits++;
}
if ( bits == 28 )
{
throw new RuntimeException( "not a valid blocksize: " + blocksize + " ( expected 1 << bits with bits in (4..27) )");
}
blocksizeBits = bits + 3;
blocksizeBitsMask = (1L << blocksizeBits ) -1;
this.blocksize = blocksize;
}
public void put( long key, int value )
{
putCount++;
if ( value < 0 || value > maxvalue )
{
throw new IllegalArgumentException( "value out of range (0.." + maxvalue + "): " + value );
}
int blockn = (int)(key >> blocksizeBits);
int offset = (int)(key & blocksizeBitsMask);
byte[] block = blockn < blocklist.size() ? blocklist.get( blockn ) : null;
int valuebits = 1;
if ( block == null )
{
block = new byte[sizeForBits(valuebits)];
bitplaneCount[0] ++;
while (blocklist.size() < blockn+1 )
{
blocklist.add(null);
}
blocklist.set( blockn, block );
}
else
{
// check how many bitplanes we have from the arraysize
while( sizeForBits( valuebits) < block.length )
{
valuebits++;
}
}
int headersize = 1 << valuebits;
byte v = (byte)(value + 1); // 0 is reserved (=unset)
// find the index in the lookup table or the first entry
int idx = 1;
while( idx < headersize )
{
if ( block[idx] == 0 )
{
block[idx] = v; // create new entry
}
if ( block[idx] == v )
{
break;
}
idx++;
}
if ( idx == headersize )
{
block = expandBlock( block, valuebits );
block[idx] = v; // create new entry
blocklist.set( blockn, block );
valuebits++;
headersize = 1 << valuebits;
}
int bitmask = 1 << (offset & 0x7);
int invmask = bitmask ^ 0xff;
int probebit = 1;
int blockidx = (offset >> 3) + headersize;
for( int i=0; i < valuebits; i++ )
{
if ( ( idx & probebit ) != 0 )
{
block[blockidx] |= bitmask;
}
else
{
block[blockidx] &= invmask;
}
probebit <<= 1;
blockidx += blocksize;
}
}
private int sizeForBits( int bits )
{
// size is lookup table + datablocks
return ( 1 << bits ) + blocksize * bits;
}
private byte[] expandBlock( byte[] block, int valuebits )
{
bitplaneCount[valuebits] ++;
byte[] newblock = new byte[sizeForBits(valuebits+1)];
int headersize = 1 << valuebits;
System.arraycopy(block, 0, newblock, 0, headersize ); // copy header
System.arraycopy(block, headersize, newblock, 2*headersize, block.length - headersize ); // copy data
return newblock;
}
public int getInt( long key )
{
// bit-stats on first get
if ( getCount++ == 0L )
{
System.out.println( "**** DenseLongMap stats ****" );
System.out.println( "putCount=" + putCount );
for( int i=0; i<8; i++ )
{
System.out.println( i + "-bitplanes=" +bitplaneCount[i] );
}
System.out.println( "****************************" );
}
if ( key < 0 )
{
return -1;
}
int blockn = (int)(key >> blocksizeBits);
int offset = (int)(key & blocksizeBitsMask);
byte[] block = blockn < blocklist.size() ? blocklist.get( blockn ) : null;
if ( block == null )
{
return -1;
}
// check how many bitplanes we have from the arrayzize
int valuebits = 1;
while( sizeForBits( valuebits) < block.length )
{
valuebits++;
}
int headersize = 1 << valuebits;
int bitmask = 1 << (offset & 7);
int probebit = 1;
int blockidx = (offset >> 3) + headersize;
int idx = 0; // 0 is reserved (=unset)
for( int i=0; i < valuebits; i++ )
{
if ( ( block[blockidx] & bitmask ) != 0 )
{
idx |= probebit;
}
probebit <<= 1;
blockidx += blocksize;
}
// lookup that value in the lookup header
return ((256 + block[idx]) & 0xff ) -1;
}
}