// Copyright (c) 2006 Dustin Sallings <dustin@spy.net>
package net.spy.memcached.transcoders;
/**
* Utility class for transcoding Java types.
*/
public final class TranscoderUtils {
private final boolean packZeros;
/**
* Get an instance of TranscoderUtils.
*
* @param pack if true, remove all zero bytes from the MSB of the packed num
*/
public TranscoderUtils(boolean pack) {
super();
packZeros=pack;
}
public byte[] encodeNum(long l, int maxBytes) {
byte[] rv=new byte[maxBytes];
for(int i=0; i<rv.length; i++) {
int pos=rv.length-i-1;
rv[pos]=(byte) ((l >> (8 * i)) & 0xff);
}
if(packZeros) {
int firstNon0=0;
for(;firstNon0<rv.length && rv[firstNon0]==0; firstNon0++) {
// Just looking for what we can reduce
}
if(firstNon0 > 0) {
byte[] tmp=new byte[rv.length - firstNon0];
System.arraycopy(rv, firstNon0, tmp, 0, rv.length-firstNon0);
rv=tmp;
}
}
return rv;
}
public byte[] encodeLong(long l) {
return encodeNum(l, 8);
}
public long decodeLong(byte[] b) {
long rv=0;
for(byte i : b) {
rv = (rv << 8) | (i<0?256+i:i);
}
return rv;
}
public byte[] encodeInt(int in) {
return encodeNum(in, 4);
}
public int decodeInt(byte[] in) {
assert in.length <= 4
: "Too long to be an int (" + in.length + ") bytes";
return (int)decodeLong(in);
}
public byte[] encodeByte(byte in) {
return new byte[]{in};
}
public byte decodeByte(byte[] in) {
assert in.length <= 1 : "Too long for a byte";
byte rv=0;
if(in.length == 1) {
rv=in[0];
}
return rv;
}
public byte[] encodeBoolean(boolean b) {
byte[] rv=new byte[1];
rv[0]=(byte)(b?'1':'0');
return rv;
}
public boolean decodeBoolean(byte[] in) {
assert in.length == 1 : "Wrong length for a boolean";
return in[0] == '1';
}
}