美文网首页
Flutter base64Decoder

Flutter base64Decoder

作者: anliner | 来源:发表于2020-10-14 17:31 被阅读0次

    从服务端拿到的数据有+号,但是用dart自带的base64Decoder解码居然报错,说+号是非法字符???,网上冲浪一波无解,只能自己简单处理下了,不求功能全面,能解燃眉之急即可。

    class Base64Util {
      static final List<String> _base64Char = [
        "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z",
        "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z", 
        "0", "1", "2", "3", "4", "5", "6", "7", "8", "9",
        "+", "/",
      ];
    
      static List<int> base64Decoder(String str) {
        /**
         * 由于后台说sun公司的某些旧的包处理出来的base64字符串包含换行符,所以先将其删除
         */
        str = str.replaceAll("\n", "");
        List<int> list = List();
        /**
         * 由于需要将传入的字符串按4字节分组,需要先判断其长度是否为4的整数倍
         */
        if (str.length % 4 == 0) {
          /**
           * 循环取字符串的字符,每次取4个
           */
          for (int i = 0; i < str.length; i += 4) {
            String char1 = str.substring(i, i + 1);
            String char2 = str.substring(i + 1, i + 2);
            String char3 = str.substring(i + 2, i + 3);
            String char4 = str.substring(i + 3, i + 4);
    
            /**
             * 将取出的字符按照字码表进行转换成数字,当为等于号时,则不进行处理,因为等于号是填充符
             */
            int code1 = _base64Char.indexOf(char1);
            int code2 = _base64Char.indexOf(char2);
            int code3;
            if ("=" != char3) {
              code3 = _base64Char.indexOf(char3);
            }
            int code4;
            if ("=" != char4) {
              code4 = _base64Char.indexOf(char4);
            }
    
            /**
             * 将转换后的数字进行分割处理,当对应字符为等于号,则不进行分割处理
             */
            /**
             * 将第一个字节与0x3F,得到低6位,然后左移2位,腾出低2位的位置
             * 然后将第二个字节与0x30,清除高2位和低4位,再右移4位,
             * 与前一个高6位相加,得到第一个新字节
             */
            int decode1 = ((code1 & 0x3F) << 2) + ((code2 & 0x30) >> 4);
            /**
             * 将第二个字节与0x0F,得到低4位,然后左移4位,腾出低4位的位置
             * 然后将第三个字节与0x3C,清除高2位和低2位,再右移2位,
             * 与前一个高4位相加,得到第二个新字节
             */
            int decode2;
            if ("=" != char3) {
              decode2 = ((code2 & 0x0F) << 4) + ((code3 & 0x3C) >> 2);
            }
            /**
             * 将第三个字节与0x03,得到低2位,然后左移6位,腾出低6位的位置
             * 然后直接与第四个字节相加,得到第三个新字节
             */
            int decode3;
            if ("=" != char4) {
              decode3 = ((code3 & 0x03) << 6) + code4;
            }
    
            list.add(decode1);
            if ("=" != char3) {
              list.add(decode2);
            }
            if ("=" != char4) {
              list.add(decode3);
            }
          }
        }
        return list;
      }
    
      static String base64Encoder(List<int> list) {
        StringBuffer sb = StringBuffer();
        /**
         * 由于需要将传入的数组按3字节分组,所以先处理3的最大整数倍的长度的内容
         */
        int remainder = list.length % 3;
        int size = list.length - remainder;
        /**
         * 循环取数组里的字节,每次取3个
         */
        for (int i = 0; i < size; i += 3) {
          int code1 = list[i];
          int code2 = list[i + 1];
          int code3 = list[i + 2];
          /**
           * 首先将第一字节右移2位,得到左6位,与上0x3F,高2位置0,得到第一个6位
           * 然后在编码表里进行转码,得到新组的第一字节
           */
          int encode1 = (code1 >> 2) & 0x3F;
          /**
           * 接着将第一字节与0x03,即00000011,得到低2位,然后左移4位,
           * 然后将第二字节右移4位,做加法运算,得到第二个6位,转码得到新组的第二字节
           */
          int encode2 = ((code1 & 0x03) << 4) + ((code2 >> 4) & 0x0F);
          /**
           * 将第二字节与0x0F,即00001111,得到低4位,然后左移2位,
           * 然后将第三字节右移6位,得到高2位,做加法运算,得到第三个6位,转码得到新组的第三字节
           */
          int encode3 = ((code2 & 0x0F) << 2) + ((code3 >> 6) & 0x03);
          /**
           * 最后将第三字节与上0x3F,清除高2位,得到低6位,即第四个6位,转码得到新组的第四字节
           */
          int encode4 = code3 & 0x3F;
    
          String char1 = _base64Char[encode1];
          String char2 = _base64Char[encode2];
          String char3 = _base64Char[encode3];
          String char4 = _base64Char[encode4];
    
          sb.write(char1);
          sb.write(char2);
          sb.write(char3);
          sb.write(char4);
        }
        /**
         * 当原文不是3的整数倍时,则需要继续处理多出的1或2个字节
         */
        if (remainder != 0) {
          /**
           * 既然多出内容,那么至少多一个,所以第一字节直接取
           * 对第二字节需要判断
           */
          int code1 = list[size];
          int code2 = 0;
          if (remainder == 2) {
            code2 = list[size + 1];
          }
          /**
           * 新组第一二字节可直接取6位然后转码
           */
          int encode1 = (code1 >> 2) & 0x3F;
          int encode2 = ((code1 & 0x03) << 4) + ((code2 >> 4) & 0x0F);
          String char1 = _base64Char[encode1];
          String char2 = _base64Char[encode2];
          /**
           * 如果原文只多出1字节,那么新组第三字节肯定为0,那么按照规则,空字符用 '=' 代替
           * 如果多出2字节,则还继续进行取6位、转码的运算,
           */
          int encode3;
          String char3 = "=";
          if (remainder == 2) {
            encode3 = (code2 & 0x0F) << 2;
            char3 = _base64Char[encode3];
          }
          /**
           * 由于最多多出2字节,所以新组第四字节肯定没有值可以取,也就是空字符,所以直接转换为 '='
           */
          String char4 = "=";
    
          sb.write(char1);
          sb.write(char2);
          sb.write(char3);
          sb.write(char4);
        }
    
        return sb.toString();
      }
    }
    

    相关文章

      网友评论

          本文标题:Flutter base64Decoder

          本文链接:https://www.haomeiwen.com/subject/kemwpktx.html