All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.scudata.dm.SerialBytes Maven / Gradle / Ivy

Go to download

SPL(Structured Process Language) A programming language specially for structured data computing.

The newest version!
package com.scudata.dm;

import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.math.BigInteger;

import com.scudata.common.MessageManager;
import com.scudata.common.RQException;
import com.scudata.resources.EngineMessage;
import com.scudata.util.HashUtil;
import com.scudata.util.Variant;

/**
 * ?źţ????8???ֽڹ??ɵ??????????ڱ?ʾ??ֵ???ɱȽϣ?????????
 * ??һ???ֽڱ?????value1??????ֽ???
 * value1					value2
 * b1 b2 b3 b4 b5 b6 b7 b8	b9 b10 b11 b12?ڿڿڿ?
 *
 * @author WangXiaoJun
 */
public class SerialBytes implements Externalizable, Comparable {
	private static final long serialVersionUID = 0x02613003;
	private static final long LONGSIGN = 0xFFFFFFFFFFFFFFFFL;
	private static final String []ZEROSTRS;
	static {
		ZEROSTRS = new String[16];
		String str = "";
		for (int i = 0; i < 16; ++i) {
			ZEROSTRS[i] = str;
			str += '0';
		}
	}
	
	private long value1;
	private long value2;
	//private int len;
	
	// ???????л?
	public SerialBytes() {
	}
	
	public SerialBytes(byte []bytes, int len) {
		int index = 0;
		for (byte b : bytes) {
			++index;
			if (index <= 8) {
				value1 |= (0xFFL & b) << (8 - index) * 8;
			} else {
				value2 |= (0xFFL & b) << (16 - index) * 8;
			}
		}
	}
	
	public SerialBytes(long value1, long value2) {
		this.value1 = value1;
		this.value2 = value2;
	}
	
	/**
	 * ?????źŶ???
	 * @param num ??
	 * @param len ?ֽ???????Χ??[1,16]
	 */
	public SerialBytes(Number num, int len) {
		if (len > 16) {
			MessageManager mm = EngineMessage.get();
			throw new RQException(mm.getMessage("serialbytes.outOfLimit"));
		}
		
		if (len <= 8) {
			value1 = num.longValue() << (8 - len) * 8;
		} else {
			BigInteger bi = Variant.toBigInteger(num);
			byte []bytes = bi.toByteArray();
			int blen = bytes.length;
			if (blen > len) {
				MessageManager mm = EngineMessage.get();
				throw new RQException(mm.getMessage("serialbytes.biLenMismatch"));
			}
			
			// ???ʵ???ֽ??????ڸ??????ֽ???????0????
			int index = len - blen;
			for (byte b : bytes) {
				++index;
				if (index <= 8) {
					value1 |= (0xFFL & b) << (8 - index) * 8;
				} else {
					value2 |= (0xFFL & b) << (16 - index) * 8;
				}
			}
		}
	}
	
	/**
	 * ?????źŶ???
	 * @param vals ??????
	 * @param lens ÿ???????ֽ??????飬???ֽ??????ܳ???16
	 */
	public SerialBytes(Number []vals, int []lens) {
		int len = 0;
		for (int i = 0; i < vals.length; ++i) {
			int curLen = lens[i];
			if (curLen <= 8) {
				// ???????С??8??Ѹ?λ??0???ɵ?????ת?????????????Ǹ?ֵ
				long curVal = vals[i].longValue() & (LONGSIGN >>> 8 - curLen);
				
				if (len < 8) {
					len += curLen;
					if (len <= 8) {
						value1 |= curVal << (8 - len) * 8;
					} else {
						value1 |= curVal >>> (len - 8) * 8;
						value2 = curVal << (16 - len) * 8;
					}
				} else {
					len += curLen;
					if (len > 16) {
						MessageManager mm = EngineMessage.get();
						throw new RQException(mm.getMessage("serialbytes.outOfLimit"));
					}
					
					value2 |= curVal << (16 - len) * 8;
				}
			} else {
				BigInteger bi = Variant.toBigInteger(vals[i]);
				byte []bytes = bi.toByteArray();
				int blen = bytes.length;
				if (blen > curLen) {
					MessageManager mm = EngineMessage.get();
					throw new RQException(mm.getMessage("serialbytes.biLenMismatch"));
				}
				
				// ???ʵ???ֽ??????ڸ??????ֽ???????0????
				len += curLen - blen;
				
				for (byte b : bytes) {
					++len;
					if (len <= 8) {
						value1 |= (0xFFL & b) << (8 - len) * 8;
					} else {
						value2 |= (0xFFL & b) << (16 - len) * 8;
					}
				}
				
				if (len > 16) {
					MessageManager mm = EngineMessage.get();
					throw new RQException(mm.getMessage("serialbytes.outOfLimit"));
				}
			}
		}
	}
	
	/**
	 * ?????źŵij???
	 * @return int
	 */
	public int length() {
		return 16;
	}
	
	/**
	 * ?????źŵĹ?ϣֵ
	 * @return int
	 */
    public int hashCode() {
    	return HashUtil.hashCode(value1 + value2);
    }
	
    /**
     * ת???ַ???????????ʾ
	 * @return String
     */
	public String toString() {
		String str1 = Long.toHexString(value1);
		int strLen = str1.length();
		
		if (strLen < 16) {
			str1 = ZEROSTRS[16 - strLen] + str1;
		}
		
		String str2 = Long.toHexString(value2);
		strLen = str2.length();
		if (strLen < 16) {
			return str1 + ZEROSTRS[16 - strLen] + str2;
		} else {
			return str1 + str2;
		}
	}
	
	/**
	 * ???ź?ת???ֽ????飬???ڴ洢
	 * @return ?ֽ?????
	 */
	public byte[] toByteArray() {
		byte []bytes = new byte[16];
		int i = 16;
		for (; i > 8; --i) {
			bytes[i - 1] = (byte)(value2 >>> (16 - i) * 8);
		}
		
		for (; i > 0; --i) {
			bytes[i - 1] = (byte)(value1 >>> (8 - i) * 8);
		}
		
		return bytes;
	}
	
	/**
	 * ȡ?ź?ָ???ֽڵ?ֵ
	 * @param q ?ֽںţ???1??ʼ????
	 * @return long
	 */
	public long getByte(int q) {
		if (q < 1 || q > 16) {
			MessageManager mm = EngineMessage.get();
			throw new RQException(q + mm.getMessage("engine.indexOutofBound"));
		} else if (q <= 8) {
			return (value1 >>> (8 - q) * 8) & 0xFFL;
		} else {
			return (value2 >>> (16 - q) * 8) & 0xFFL;
		}
	}
	
	/**
	 * ȡ?ź?ָ???ֽ??????ֵ
	 * @param start ??ʼ?ֽںţ???1??ʼ????????????
	 * @param end ?????ֽںţ???1??ʼ????????????
	 * @return long
	 */
	public long getBytes(int start, int end) {
		if (start < 1 || end < start || end > 16) {
			MessageManager mm = EngineMessage.get();
			throw new RQException(start + "," + end + mm.getMessage("engine.indexOutofBound"));
		}
		
		if (start <= 8) {
			if (end <= 8) {
				return (value1 << (start - 1) * 8) >>> (7 + start - end) * 8;
			} else {
				long result = (value1 << (start - 1) * 8) >>> (start - 1) * 8;
				return (result << (end - 8) * 8) | (value2 >>> (16 - end) * 8);
			}
		} else {
			return (value2 << (start - 9) * 8) >>> (7 + start - end) * 8;
		}
	}
	
	/**
	 * ?Ƚ??????źŵĴ?С
	 * @param value1 ?ź?1??ֵ1
	 * @param value2 ?ź?1??ֵ2
	 * @param otherValue1 ??һ???źŵ?ֵ1
	 * @param otherValue2 ??һ???źŵ?ֵ2
	 * @return 1???ź?1??0??һ???ģ?-1???ź?2??
	 */
	public static int compare(long value1, long value2, long otherValue1, long otherValue2) {
		if (value1 == otherValue1) {
			if (value2 == otherValue2) {
				return 0;
			} else if (value2 < 0) {
				if (otherValue2 >= 0) {
					return 1;
				} else {
					return value2 > otherValue2 ? 1 : -1;
				}
			} else if (otherValue2 < 0) {
				return -1;
			} else {
				return value2 > otherValue2 ? 1 : -1;
			}
		} else {
			if (value1 < 0) {
				if (otherValue1 >= 0) {
					return 1;
				} else {
					return value1 > otherValue1 ? 1 : -1;
				}
			} else if (otherValue1 < 0) {
				return -1;
			} else {
				return value1 > otherValue1 ? 1 : -1;
			}
		}
	}
	
	/**
	 * ?Ƚ??????źŵĴ?С??????????
	 * @param o
	 * @return int
	 */
	public int compareTo(SerialBytes o) {
		return compare(value1, value2, o.value1, o.value2);
	}
	
	public boolean equals(Object obj) {
		if (obj instanceof SerialBytes) {
			return equals((SerialBytes)obj);
		} else {
			return false;
		}
	}
	
	public long getValue1() {
		return value1;
	}

	public long getValue2() {
		return value2;
	}

	/**
	 * ?ж??????ź??Ƿ????
	 * @param other
	 * @return
	 */
	public boolean equals(SerialBytes other) {
		return value1 == other.value1 && value2 == other.value2;
	}
	
	public void writeExternal(ObjectOutput out) throws IOException {
		out.writeLong(value1);
		out.writeLong(value2);
	}
	
	public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
		value1 = in.readLong();
		value2 = in.readLong();
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy