mirror of
https://github.com/zxing/zxing.git
synced 2025-02-02 05:41:08 -08:00
Wrote a reasonable implementation of BitVector which now compiles.
git-svn-id: https://zxing.googlecode.com/svn/trunk@698 59b500cc-1b3d-0410-9834-0bbf25fbcc57
This commit is contained in:
parent
6f4898c68f
commit
5dba28fc86
|
@ -17,47 +17,57 @@
|
||||||
package com.google.zxing.qrcode.encoder;
|
package com.google.zxing.qrcode.encoder;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* JAVAPORT: This should be combined with BitArray in the future, although that class is not yet
|
||||||
|
* dynamically resizeable. This implementation is reasonable but there is a lot of function calling
|
||||||
|
* in loops I'd like to get rid of.
|
||||||
|
*
|
||||||
* @author satorux@google.com (Satoru Takabayashi) - creator
|
* @author satorux@google.com (Satoru Takabayashi) - creator
|
||||||
* @author dswitkin@google.com (Daniel Switkin) - ported from C++
|
* @author dswitkin@google.com (Daniel Switkin) - ported from C++
|
||||||
*/
|
*/
|
||||||
public final class BitVector {
|
public final class BitVector {
|
||||||
|
|
||||||
private int size_;
|
private int sizeInBits;
|
||||||
private String bytes_;
|
private int bytePosition;
|
||||||
|
private byte[] array;
|
||||||
|
|
||||||
|
// For efficiency, start out with some room to work.
|
||||||
|
private static final int DEFAULT_SIZE_IN_BITS = 32 * 8;
|
||||||
|
|
||||||
public BitVector() {
|
public BitVector() {
|
||||||
size_ = 0;
|
sizeInBits = DEFAULT_SIZE_IN_BITS;
|
||||||
|
bytePosition = 0;
|
||||||
|
array = new byte[DEFAULT_SIZE_IN_BITS / 8];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return the bit value at "index".
|
// Return the bit value at "index".
|
||||||
public int at(final int index) {
|
public int at(final int index) {
|
||||||
Debug.DCHECK_LE(0, index);
|
Debug.DCHECK_LE(0, index);
|
||||||
Debug.DCHECK_LT(index, size_);
|
Debug.DCHECK_LT(index, sizeInBits);
|
||||||
final uint8 byte = bytes_.at(index / 8);
|
final int value = array[index / 8];
|
||||||
return (byte >> (7 - (index % 8))) & 1;
|
return (value >> (7 - (index % 8))) & 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return the number of bits in the bit vector.
|
// Return the number of bits in the bit vector.
|
||||||
public int size() {
|
public int size() {
|
||||||
return size_;
|
return sizeInBits;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return the number of bytes in the bit vector.
|
// Return the number of bytes in the bit vector.
|
||||||
public int num_bytes() {
|
public int num_bytes() {
|
||||||
return size_ / 8;
|
return sizeInBits / 8;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Append one bit to the bit vector.
|
// Append one bit to the bit vector.
|
||||||
public void AppendBit(final int bit) {
|
public void AppendBit(final int bit) {
|
||||||
Debug.DCHECK(bit == 0 || bit == 1);
|
Debug.DCHECK(bit == 0 || bit == 1);
|
||||||
final int num_bits_in_last_byte = size_ % 8;
|
final int num_bits_in_last_byte = sizeInBits % 8;
|
||||||
// We'll expand bytes_ if we don't have bits in the last byte.
|
// We'll expand array if we don't have bits in the last byte.
|
||||||
if (num_bits_in_last_byte == 0) {
|
if (num_bits_in_last_byte == 0) {
|
||||||
bytes_.push_back(0);
|
appendByte(0);
|
||||||
}
|
}
|
||||||
// Modify the last byte.
|
// Modify the last byte.
|
||||||
bytes_[bytes_.size() - 1] |= (bit << (7 - num_bits_in_last_byte));
|
array[array.length - 1] |= (bit << (7 - num_bits_in_last_byte));
|
||||||
++size_;
|
++sizeInBits;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Append "num_bits" bits in "value" to the bit vector.
|
// Append "num_bits" bits in "value" to the bit vector.
|
||||||
|
@ -67,15 +77,15 @@ public final class BitVector {
|
||||||
// - AppendBits(0x00, 1) adds 0.
|
// - AppendBits(0x00, 1) adds 0.
|
||||||
// - AppendBits(0x00, 4) adds 0000.
|
// - AppendBits(0x00, 4) adds 0000.
|
||||||
// - AppendBits(0xff, 8) adds 11111111.
|
// - AppendBits(0xff, 8) adds 11111111.
|
||||||
public void AppendBits(final uint32 value, final int num_bits) {
|
public void AppendBits(final int value, final int num_bits) {
|
||||||
Debug.DCHECK(num_bits >= 0 && num_bits <= 32);
|
Debug.DCHECK(num_bits >= 0 && num_bits <= 32);
|
||||||
int num_bits_left = num_bits;
|
int num_bits_left = num_bits;
|
||||||
while (num_bits_left > 0) {
|
while (num_bits_left > 0) {
|
||||||
// Optimization for byte-oriented appending.
|
// Optimization for byte-oriented appending.
|
||||||
if (size_ % 8 == 0 && num_bits_left >= 8) {
|
if (sizeInBits % 8 == 0 && num_bits_left >= 8) {
|
||||||
final uint8 byte = (value >> (num_bits_left - 8)) & 0xff;
|
final int newByte = (value >> (num_bits_left - 8)) & 0xff;
|
||||||
bytes_.push_back(byte);
|
appendByte(newByte);
|
||||||
size_ += 8;
|
sizeInBits += 8;
|
||||||
num_bits_left -= 8;
|
num_bits_left -= 8;
|
||||||
} else {
|
} else {
|
||||||
final int bit = (value >> (num_bits_left - 1)) & 1;
|
final int bit = (value >> (num_bits_left - 1)) & 1;
|
||||||
|
@ -86,39 +96,36 @@ public final class BitVector {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Append "bytes".
|
// Append "bytes".
|
||||||
public void AppendBytes(final StringPiece &bytes) {
|
//
|
||||||
for (int i = 0; i < bytes.size(); ++i) {
|
// JAVAPORT: Uncomment and implement when a substitute for StringPiece is chosen.
|
||||||
AppendBits(bytes[i], 8);
|
// public void AppendBytes(final StringPiece stringPiece) {
|
||||||
}
|
// for (int i = 0; i < stringPiece.size(); ++i) {
|
||||||
}
|
// AppendBits(stringPiece[i], 8);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
// Append "bits".
|
// Append "bits".
|
||||||
public void AppendBitVector(final BitVector &bits) {
|
public void AppendBitVector(final BitVector bits) {
|
||||||
for (int i = 0; i < bits.size(); ++i) {
|
int size = bits.size();
|
||||||
|
for (int i = 0; i < size; ++i) {
|
||||||
AppendBit(bits.at(i));
|
AppendBit(bits.at(i));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Modify the bit vector by XOR'ing with "other"
|
// Modify the bit vector by XOR'ing with "other"
|
||||||
public void XOR(final BitVector &other) {
|
public void XOR(final BitVector other) {
|
||||||
Debug.DCHECK_EQ(size_, other.size());
|
Debug.DCHECK_EQ(sizeInBits, other.size());
|
||||||
for (int i = 0; i < bytes_.size(); ++i) {
|
for (int i = 0; i < array.length; ++i) {
|
||||||
// The last byte could be incomplete (i.e. not have 8 bits in
|
// The last byte could be incomplete (i.e. not have 8 bits in
|
||||||
// it) but there is no problem since 0 XOR 0 == 0.
|
// it) but there is no problem since 0 XOR 0 == 0.
|
||||||
bytes_[i] ^= other.ToString()[i];
|
array[i] ^= other.array[i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return the content of the bit vector as String.
|
|
||||||
public final String &ToString() {
|
|
||||||
return bytes_;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return String like "01110111" for debugging.
|
// Return String like "01110111" for debugging.
|
||||||
public String ToASCII() {
|
public String toString() {
|
||||||
String result;
|
StringBuffer result = new StringBuffer(sizeInBits);
|
||||||
result.reserve(size_);
|
for (int i = 0; i < sizeInBits; ++i) {
|
||||||
for (int i = 0; i < size_; ++i) {
|
|
||||||
if (at(i) == 0) {
|
if (at(i) == 0) {
|
||||||
result.append("0");
|
result.append("0");
|
||||||
} else if (at(i) == 1) {
|
} else if (at(i) == 1) {
|
||||||
|
@ -127,7 +134,19 @@ public final class BitVector {
|
||||||
Debug.DCHECK(false);
|
Debug.DCHECK(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
return result.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add a new byte to the end, possibly reallocating and doubling the size of the array if we've
|
||||||
|
// run out of room.
|
||||||
|
private void appendByte(int value) {
|
||||||
|
if (bytePosition >= array.length) {
|
||||||
|
byte[] newArray = new byte[array.length * 2];
|
||||||
|
System.arraycopy(array, 0, newArray, 0, array.length);
|
||||||
|
array = newArray;
|
||||||
|
}
|
||||||
|
array[bytePosition] = (byte) value;
|
||||||
|
bytePosition++;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue