Base64是网络上最常用的传输8bit字节数据的编码方式之一,是一种简单的加密方式,实际中使用的加密比这复杂的多,也可以用在复杂数据的存储上,比如我们要把类对象和图片等复杂数据进行存储,就需要将这些对象和图片的字节数据进行Base64编码,然后再讲编码后的String存储到XML文件中。
我们先来简单的了解Base64的原理:3*8=4*6
例子:s13
首先将其转换成ASCII后对应的是:115 49 51 ,这3个数字的二进制是01110011 00110001 00110011,
然后6个为一组(共4组):011100 110011 000100 110011,再将每组高位补全为8个得到:00011100 00110011 00000100 00110011,得到这4组二进制数对应的 十进制:28 51 4 51 ,最后根据这4个数字到表(这个表格百度可以找到)中其所对应的值:c z E z,那么就得到了经过Base64编码后的数据。
关于Base64的代码实现在后面我会给出,现在我们来看一下Android中Base64的使用方法。
在网上找到关于Base64的文章很少,所以就自己看了Android源码做出下面的总结,如有错误地方请指出。我们来看一下Andorid中提供给我们可以使用的Base64函数的编码。
1. public class Base64{
2. public static final int DEFAULT=0; //下面6个静态全局变量是编码和解码时的标示,我们一般使用第一个
3. public static final int NO_PADDING=1;
4. public static final int NO_WRAP=2;
5. public static final int CRLF=4;
6. public static final int URL_SAFE=8;
7. public static final int NO_CLOSE=16;
8.
9. ...
10.
11. //下面是我们在编码时常调用的函数,均为静态函数
12.
13. /**
14. * 编码核心函数
15. * 参数input:要进行编码的字符串的字节数组
16. * 参数offset:开始编码位置的偏移值,一般为0
17. * 参数len: 要编码的字节数组的长度
18. * 参数flags: 编码的标示,一般为DEFAULT
19. * 返回值为编码后得到的字节数组
20. */
21. public static byte[] encode(byte[] input,int offset,int len,int flags){
22. ...
23. }
24.
25. public static byte[] encode(byte[] input,int flags){
26. return encode(input,0,input.length,flags);//调用第一个函数
27. }
28.
29. //调用第二个函数,并将其结果封装成字符串形式
30. public static String encodeToString(byte[] input,int flags){
31. try{
32. return new String(encode(input,flags),"US-ASCII");
33. catch(UnsupportEncodingException e){
34. throw new AssertionError(e);
35. }
36. }
37.
38. //对第二个函数进行了封装,将原来返回的字节数组封装成字符串
39. public static String encodeToString(byte[] input,int flags){
40. try{
41. return new String(encode(input,flags),"US-ASCII");
42. catch(UnsupportedEncodingException e){
43. throw new AssertionError(e);
44. }
45. }
46. ......
47. }
Base64类提供了四个编码函数,其他三个最终都会调用encode(byte[] input,int offset,int len,int flags)。因为我们在使用时有两种需求:返回字节数组被编码后得到的字符串和返回字节数组被编码后得到字节数组。encode函数返回字节数组被编码后得到的字节数组,encodeToString内部调用了encode, 并将返回的字节数组转换成字符串。
下面来看看关于解码的函数:
1. public static byte[] decode(String str,int flags){
2. return decode(str.getBytes(),flags);//调用下面那个函数
3. }
4.
5. public static byte[] decode(byte[] input,int flags){
6. return decode(input,0,input.length,flags);//调用下面那个函数
7. }
8.
9. /**
10. * 在编码时最终都调用该函数
11. * 参数input:要编码字符串的字节数组
12. * 参数offset:开始编码位置的偏移值
13. * 参数len:编码数组的长度
14. * 参数flag:6个标示之一,一般为DEFAULT
15. */
16. public static byte[] decode(byte[] input,int offset,int len,int flags{
17. ...
18. }
19.
20.
解码函数的类型和使用与编码函数类似,我就不再赘述了。
对于在本文开头所举的例子:s13进行编码后得到czEz的代码实现如下:
1. String s="s13";
2. String str=Base64.encodeToString(s.getBytes(), 0);
得到的str值为czEz。
那么对czEz进行解码操作后就能得到s13了。
1. String s="s13";
2. String str_encode=Base64.encodeToString(s.getBytes(), 0);
3. byte[] byteDecode=Base64.decode(str_encode, 0);
4. String str_decode=new String(byteDecode);
得到的str_decode的值就为s13,在断点调试时就会发现字节数组byteDecode中值为[ 115,49,51],刚好就是s13的ASCII值。我们只需用String转换一下即可。
上面分析了Android内置的Base64编码和解码,但是其只提供了简单的字符串或字节数组的编解码,我们在项目开发中就会发现时我们的需求是多种多样的,如果使用内置的Base64会导致项目中代码过多,所以我们一般自己实现Base64加解码,自定义很多方法。我会单独写一篇博客来讲解实现自定义的Base64。
下面我们在项目中来使用Base64来实现复杂数据的存取,这里是使用SharedPreferences来存取的,SharedPreferences是最简单的存取方式,只能用于存放key-value对,一般我们只用于存取int,String等基本类型,这里我们通过将复杂数据类型,比如类对象和图片,进行编码后再以键值对的形式存储。
先看一下实现的效果: 1. 将两个编辑框中的数据赋值给类对象然后进行编码后存储起来,再解码后显示出来;
2.将图上的红点图像编码保存,再解码显示到底部的安卓头像上。
我们的要存储的类,需要序列化:
1. import java.io.Serializable;
2.
3. public class Produce implements Serializable{
4. private String id;
5. private String name;
6.
7. public void setId(String id){
8. this.id=id;
9. }
10. public void setName(String name){
11. this.name=name;
12. }
13.
14. public String getId(){
15. return id;
16. }
17.
18. public String getName(){
19. return name;
20. }
21.
22. }
点击“编码后存储”按钮时:
1. String str1=edit1.getText().toString();
2. String str2=edit2.getText().toString();
3. if(!str1.equals(null)&&!str2.equals(null)){
4. new Produce();
5. pro.setId(str1);
6. pro.setName(str2);
7. //对Produce对象进行Base64编码
8. new ByteArrayOutputStream();
9. try {
10. //对类对象pro编码后存储
11. new ObjectOutputStream(baos);
12. //将对象写到字节数组流中的
13. 0);//编码
14. "base64",Activity.MODE_PRIVATE);
15. SharedPreferences.Editor editor=mySharedPreferences.edit();
16. "produce", encodeStr);
17. editor.commit();
18. catch (IOException e) {
19. e.printStackTrace();
20. }
21. }
当点击“解码显示”按钮时:
1. String produceBase64=mySharedPreferences.getString("produce", "");
2. if(!produceBase64.equals("")){
3. //解码显示
4. byte[] strDecode=Base64.decode(produceBase64, 0);
5. new ByteArrayInputStream(strDecode);
6. try {
7. new ObjectInputStream(bais);
8. //从ObjectInputStream 中读取Produce对象
9. try {
10. Produce pro=(Produce)ois.readObject();
11. "Produce id:"+pro.getId()+"name:"+pro.getName());
12. catch (ClassNotFoundException e) {
13. e.printStackTrace();
14. }
15. catch (StreamCorruptedException e) {
16. e.printStackTrace();
17. catch (IOException e) {
18. e.printStackTrace();
19. }
1. }
对图片的编码,压缩,存储过程如下:
1. ByteArrayOutputStream baos1=new ByteArrayOutputStream();
2. ((BitmapDrawable)image1.getDrawable()).getBitmap().compress(CompressFormat.JPEG, 50, baos1);
3. String encodeBitmap=Base64.encodeToString(baos1.toByteArray(), 0);
4. editor.putString("Bitmap", encodeBitmap);
5. editor.commit();
解码并显示出来:
1. ByteArrayInputStream array=new ByteArrayInputStream(Base64.decode(decodeBitmap, 0));
2. image2.setImageDrawable(Drawable.createFromStream(array, "image_name"));
由于时间问题,完成比较仓促。