1 /* 2 * Copyright 2001-2004 The Apache Software Foundation. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package org.apache.commons.codec.binary; 18 19 import org.apache.commons.codec.BinaryDecoder; 20 import org.apache.commons.codec.BinaryEncoder; 21 import org.apache.commons.codec.DecoderException; 22 import org.apache.commons.codec.EncoderException; 23 24 /** 25 * Translates between byte arrays and strings of "0"s and "1"s. 26 * 27 * <b>TODO:</b> may want to add more bit vector functions like and/or/xor/nand. 28 * <B>TODO:</b> also might be good to generate boolean[] 29 * from byte[] et. cetera. 30 * 31 * @author Apache Software Foundation 32 * @since 1.3 33 * @version $Id $ 34 * 35 * @deprecated Please use {@link java.net.URL#openConnection} instead. 36 * Please visit <a href="http://android-developers.blogspot.com/2011/09/androids-http-clients.html">this webpage</a> 37 * for further details. 38 */ 39 @Deprecated 40 public class BinaryCodec implements BinaryDecoder, BinaryEncoder { 41 /* 42 * tried to avoid using ArrayUtils to minimize dependencies while using these empty arrays - dep is just not worth 43 * it. 44 */ 45 /** Empty char array. */ 46 private static final char[] EMPTY_CHAR_ARRAY = new char[0]; 47 48 /** Empty byte array. */ 49 private static final byte[] EMPTY_BYTE_ARRAY = new byte[0]; 50 51 /** Mask for bit 0 of a byte. */ 52 private static final int BIT_0 = 1; 53 54 /** Mask for bit 1 of a byte. */ 55 private static final int BIT_1 = 0x02; 56 57 /** Mask for bit 2 of a byte. */ 58 private static final int BIT_2 = 0x04; 59 60 /** Mask for bit 3 of a byte. */ 61 private static final int BIT_3 = 0x08; 62 63 /** Mask for bit 4 of a byte. */ 64 private static final int BIT_4 = 0x10; 65 66 /** Mask for bit 5 of a byte. */ 67 private static final int BIT_5 = 0x20; 68 69 /** Mask for bit 6 of a byte. */ 70 private static final int BIT_6 = 0x40; 71 72 /** Mask for bit 7 of a byte. */ 73 private static final int BIT_7 = 0x80; 74 75 private static final int[] BITS = {BIT_0, BIT_1, BIT_2, BIT_3, BIT_4, BIT_5, BIT_6, BIT_7}; 76 77 /** 78 * Converts an array of raw binary data into an array of ascii 0 and 1 characters. 79 * 80 * @param raw 81 * the raw binary data to convert 82 * @return 0 and 1 ascii character bytes one for each bit of the argument 83 * @see org.apache.commons.codec.BinaryEncoder#encode(byte[]) 84 */ 85 public byte[] encode(byte[] raw) { 86 return toAsciiBytes(raw); 87 } 88 89 /** 90 * Converts an array of raw binary data into an array of ascii 0 and 1 chars. 91 * 92 * @param raw 93 * the raw binary data to convert 94 * @return 0 and 1 ascii character chars one for each bit of the argument 95 * @throws EncoderException 96 * if the argument is not a byte[] 97 * @see org.apache.commons.codec.Encoder#encode(java.lang.Object) 98 */ 99 public Object encode(Object raw) throws EncoderException { 100 if (!(raw instanceof byte[])) { 101 throw new EncoderException("argument not a byte array"); 102 } 103 return toAsciiChars((byte[]) raw); 104 } 105 106 /** 107 * Decodes a byte array where each byte represents an ascii '0' or '1'. 108 * 109 * @param ascii 110 * each byte represents an ascii '0' or '1' 111 * @return the raw encoded binary where each bit corresponds to a byte in the byte array argument 112 * @throws DecoderException 113 * if argument is not a byte[], char[] or String 114 * @see org.apache.commons.codec.Decoder#decode(java.lang.Object) 115 */ 116 public Object decode(Object ascii) throws DecoderException { 117 if (ascii == null) { 118 return EMPTY_BYTE_ARRAY; 119 } 120 if (ascii instanceof byte[]) { 121 return fromAscii((byte[]) ascii); 122 } 123 if (ascii instanceof char[]) { 124 return fromAscii((char[]) ascii); 125 } 126 if (ascii instanceof String) { 127 return fromAscii(((String) ascii).toCharArray()); 128 } 129 throw new DecoderException("argument not a byte array"); 130 } 131 132 /** 133 * Decodes a byte array where each byte represents an ascii '0' or '1'. 134 * 135 * @param ascii 136 * each byte represents an ascii '0' or '1' 137 * @return the raw encoded binary where each bit corresponds to a byte in the byte array argument 138 * @see org.apache.commons.codec.Decoder#decode(Object) 139 */ 140 public byte[] decode(byte[] ascii) { 141 return fromAscii(ascii); 142 } 143 144 /** 145 * Decodes a String where each char of the String represents an ascii '0' or '1'. 146 * 147 * @param ascii 148 * String of '0' and '1' characters 149 * @return the raw encoded binary where each bit corresponds to a byte in the byte array argument 150 * @see org.apache.commons.codec.Decoder#decode(Object) 151 */ 152 public byte[] toByteArray(String ascii) { 153 if (ascii == null) { 154 return EMPTY_BYTE_ARRAY; 155 } 156 return fromAscii(ascii.toCharArray()); 157 } 158 159 // ------------------------------------------------------------------------ 160 // 161 // static codec operations 162 // 163 // ------------------------------------------------------------------------ 164 /** 165 * Decodes a byte array where each char represents an ascii '0' or '1'. 166 * 167 * @param ascii 168 * each char represents an ascii '0' or '1' 169 * @return the raw encoded binary where each bit corresponds to a char in the char array argument 170 */ 171 public static byte[] fromAscii(char[] ascii) { 172 if (ascii == null || ascii.length == 0) { 173 return EMPTY_BYTE_ARRAY; 174 } 175 // get length/8 times bytes with 3 bit shifts to the right of the length 176 byte[] l_raw = new byte[ascii.length >> 3]; 177 /* 178 * We decr index jj by 8 as we go along to not recompute indices using multiplication every time inside the 179 * loop. 180 */ 181 for (int ii = 0, jj = ascii.length - 1; ii < l_raw.length; ii++, jj -= 8) { 182 for (int bits = 0; bits < BITS.length; ++bits) { 183 if (ascii[jj - bits] == '1') { 184 l_raw[ii] |= BITS[bits]; 185 } 186 } 187 } 188 return l_raw; 189 } 190 191 /** 192 * Decodes a byte array where each byte represents an ascii '0' or '1'. 193 * 194 * @param ascii 195 * each byte represents an ascii '0' or '1' 196 * @return the raw encoded binary where each bit corresponds to a byte in the byte array argument 197 */ 198 public static byte[] fromAscii(byte[] ascii) { 199 if (ascii == null || ascii.length == 0) { 200 return EMPTY_BYTE_ARRAY; 201 } 202 // get length/8 times bytes with 3 bit shifts to the right of the length 203 byte[] l_raw = new byte[ascii.length >> 3]; 204 /* 205 * We decr index jj by 8 as we go along to not recompute indices using multiplication every time inside the 206 * loop. 207 */ 208 for (int ii = 0, jj = ascii.length - 1; ii < l_raw.length; ii++, jj -= 8) { 209 for (int bits = 0; bits < BITS.length; ++bits) { 210 if (ascii[jj - bits] == '1') { 211 l_raw[ii] |= BITS[bits]; 212 } 213 } 214 } 215 return l_raw; 216 } 217 218 /** 219 * Converts an array of raw binary data into an array of ascii 0 and 1 character bytes - each byte is a truncated 220 * char. 221 * 222 * @param raw 223 * the raw binary data to convert 224 * @return an array of 0 and 1 character bytes for each bit of the argument 225 * @see org.apache.commons.codec.BinaryEncoder#encode(byte[]) 226 */ 227 public static byte[] toAsciiBytes(byte[] raw) { 228 if (raw == null || raw.length == 0) { 229 return EMPTY_BYTE_ARRAY; 230 } 231 // get 8 times the bytes with 3 bit shifts to the left of the length 232 byte[] l_ascii = new byte[raw.length << 3]; 233 /* 234 * We decr index jj by 8 as we go along to not recompute indices using multiplication every time inside the 235 * loop. 236 */ 237 for (int ii = 0, jj = l_ascii.length - 1; ii < raw.length; ii++, jj -= 8) { 238 for (int bits = 0; bits < BITS.length; ++bits) { 239 if ((raw[ii] & BITS[bits]) == 0) { 240 l_ascii[jj - bits] = '0'; 241 } else { 242 l_ascii[jj - bits] = '1'; 243 } 244 } 245 } 246 return l_ascii; 247 } 248 249 /** 250 * Converts an array of raw binary data into an array of ascii 0 and 1 characters. 251 * 252 * @param raw 253 * the raw binary data to convert 254 * @return an array of 0 and 1 characters for each bit of the argument 255 * @see org.apache.commons.codec.BinaryEncoder#encode(byte[]) 256 */ 257 public static char[] toAsciiChars(byte[] raw) { 258 if (raw == null || raw.length == 0) { 259 return EMPTY_CHAR_ARRAY; 260 } 261 // get 8 times the bytes with 3 bit shifts to the left of the length 262 char[] l_ascii = new char[raw.length << 3]; 263 /* 264 * We decr index jj by 8 as we go along to not recompute indices using multiplication every time inside the 265 * loop. 266 */ 267 for (int ii = 0, jj = l_ascii.length - 1; ii < raw.length; ii++, jj -= 8) { 268 for (int bits = 0; bits < BITS.length; ++bits) { 269 if ((raw[ii] & BITS[bits]) == 0) { 270 l_ascii[jj - bits] = '0'; 271 } else { 272 l_ascii[jj - bits] = '1'; 273 } 274 } 275 } 276 return l_ascii; 277 } 278 279 /** 280 * Converts an array of raw binary data into a String of ascii 0 and 1 characters. 281 * 282 * @param raw 283 * the raw binary data to convert 284 * @return a String of 0 and 1 characters representing the binary data 285 * @see org.apache.commons.codec.BinaryEncoder#encode(byte[]) 286 */ 287 public static String toAsciiString(byte[] raw) { 288 return new String(toAsciiChars(raw)); 289 } 290 } 291