Различная кодировка base64 для одного и того же изображения - PullRequest
0 голосов
/ 07 января 2019

Мне нужно конвертировать base64 to opencv::mat файл. Итак, я взял набор из 10 кадров из видео и добавил base64(A) (используя base64.b64encode в python).

После этого я использовал код из здесь для преобразования этого base64 в Mat. Изображения выглядели хорошо.

Но они имеют больший размер файла, чем исходные изображения, плюс, когда я закодировал эти окончательные изображения в base64(B) (используя base64.b64encode в python), кодированный base64 отличается от исходного base64(A). Не могу понять почему? Это также влияет на вывод моего приложения, использующего вывод cv::mat.

Для base64 to Mat Я использую код от здесь (как указано выше).

Отредактировано : ниже приведен мой скрипт на python для преобразования набора jpeg to base64 (.txt)

 def img2txt(file_directory):    
        imageFiles = glob.glob(file_directory+"/*.jpg")
        imageFiles.sort()
        fileWrite='base64encoding.txt'
        #print fileWrite
        for i in range(0,len(imageFiles)):
            image = open(imageFiles[i],'rb')
            image_read = image.read()
            image_64_encode = base64.b64encode(image_read)
            with open (fileWrite, 'a') as f: 
                f.writelines(image_64_encode+'\n')

функция base64decode : с здесь

static const std::string base64_chars =
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"abcdefghijklmnopqrstuvwxyz"
"0123456789+/";


static inline bool is_base64(unsigned char c) {
    return (isalnum(c) || (c == '+') || (c == '/'));
}

std::string base64_decode(std::string const& encoded_string) {
    int in_len = encoded_string.size();
    int i = 0;
    int j = 0;
    int in_ = 0;
    unsigned char char_array_4[4], char_array_3[3];
    std::string ret;

    while (in_len-- && (encoded_string[in_] != '=') && is_base64(encoded_string[in_])) {
        char_array_4[i++] = encoded_string[in_]; in_++;
        if (i == 4) {
            for (i = 0; i < 4; i++)
                char_array_4[i] = base64_chars.find(char_array_4[i]);

            char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
            char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
            char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];

            for (i = 0; (i < 3); i++)
                ret += char_array_3[i];
            i = 0;
        }
    }

    if (i) {
        /*for (j = i; j < 4; j++)
            char_array_4[j] = 0;*/

        for (j = 0; j < i; j++)
            char_array_4[j] = base64_chars.find(char_array_4[j]);

        char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
        char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
        //char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];

        for (j = 0; (j < i - 1); j++) 
            ret += char_array_3[j];
    }

    return ret;
}

C ++ Основная функция :

int main()
{  
  ifstream in("TestBase64.txt");
  if(!in) {
    cout << "Cannot open input file.\n";
    return 1;
    }
    int i=0;
    string encoded_string;
    while (getline(in, encoded_string)) 
    {
        string decoded_string = base64_decode(encoded_string);
        vector<uchar> data(decoded_string.begin(), decoded_string.end());
        cv::imwrite("/Frames_from_B_to_Mat/Frames_from_B_to_Mat"+b+".jpg");
        i++; 
    }

    return 0;
}

1 Ответ

0 голосов
/ 07 января 2019

На линии:

vector<uchar> data(decoded_string.begin(), decoded_string.end());

вы, вероятно, держите в кодировке JPEG представление одного изображения в data. Таким образом, вы можете просто записать это в двоичный файл, а не использовать cv::imwrite(), который предназначен для записи Mat в файл.

Если по какой-то необъяснимой причине вы хотите использовать cv::imwrite(), вам нужно передать его Mat. Таким образом, вы в конечном итоге декодируете представление JPEG в Mat, а затем кодируете в JPEG и пишете - что выглядит глупо:

cv::Mat img = cv::imdecode(data, cv::IMREAD_COLOR);
cv::imwrite('result.jpg',img);

TLDR;

Я говорю, что ваши данные уже закодированы в формате JPEG, вы читаете их из файла JPEG.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...