C++ 从二进制数据还原MP3文件

C++ 从二进制数据还原MP3文件,c++,audio,binary,mp3,encode,C++,Audio,Binary,Mp3,Encode,我的任务是还原一个mp3文件,它是PNG文件中每一位的编码。我从矢量中的PNG RGB数据(每像素)中获得了正确的位。我用的是C++。 我必须遍历png文件并读取像素的RGB数据:然后我有3个十进制值。从十进制值的二进制表示中,我需要最小的局部值。11个像素在33位上显示mp3的长度。然后我解码像素中的所有二进制数据,并放入一个向量中 #include <unistd.h> #include <stdlib.h> #include <stdio.h> #inc

我的任务是还原一个mp3文件,它是PNG文件中每一位的编码。我从矢量中的PNG RGB数据(每像素)中获得了正确的位。我用的是C++。 我必须遍历png文件并读取像素的RGB数据:然后我有3个十进制值。从十进制值的二进制表示中,我需要最小的局部值。11个像素在33位上显示mp3的长度。然后我解码像素中的所有二进制数据,并放入一个向量中

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <stdarg.h>
#include <vector>
#include <math.h>
#include <iostream>
#include <fstream>

#define PNG_DEBUG 3
#include <png.h>

void abort_(const char * s, ...)
{
        va_list args;
        va_start(args, s);
        vfprintf(stderr, s, args);
        fprintf(stderr, "\n");
        va_end(args);
        abort();
}

void itob(short n, std::vector<int> &bin)
{
    int d = n;

    if (n > 1)
    {
        d = n % 2;
        itob(n / 2, bin);
    }
    bin.push_back(d);
}

void btoi(unsigned int& n, std::vector<int> bin)
{
    n = 0;
    int k = 32;
    for(int i = 0; i < bin.size() ; i++){
        if(bin[i] == 1){
            long int num = pow(2,k);
            n += num;
        }
        k--;
    }
}

int x, y;

int width, height;
png_byte color_type;
png_byte bit_depth;

png_structp png_ptr;
png_infop info_ptr;
int number_of_passes;
png_bytep * row_pointers;

void read_png_file()
{
        unsigned char header[8];    // 8 is the maximum size that can be checked

        /* open file and test for it being a png */
        FILE *fp = fopen("image.png", "rb");
        if (!fp)
                abort_("[read_png_file] File %s could not be opened for reading", "image.png");
        fread(header, 1, 8, fp);
        if (png_sig_cmp(header, 0, 8))
                abort_("[read_png_file] File %s is not recognized as a PNG file", "image.png");


        /* initialize stuff */
        png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);

        if (!png_ptr)
                abort_("[read_png_file] png_create_read_struct failed");

        info_ptr = png_create_info_struct(png_ptr);
        if (!info_ptr)
                abort_("[read_png_file] png_create_info_struct failed");

        png_init_io(png_ptr, fp);
        png_set_sig_bytes(png_ptr, 8);

        png_read_info(png_ptr, info_ptr);

        width = png_get_image_width(png_ptr, info_ptr);
        height = png_get_image_height(png_ptr, info_ptr);
        color_type = png_get_color_type(png_ptr, info_ptr);
        bit_depth = png_get_bit_depth(png_ptr, info_ptr);

        number_of_passes = png_set_interlace_handling(png_ptr);
        png_read_update_info(png_ptr, info_ptr);

        row_pointers = (png_bytep*) malloc(sizeof(png_bytep) * height);
        for (y=0; y<height; y++)
                row_pointers[y] = (png_byte*) malloc(png_get_rowbytes(png_ptr,info_ptr));

        png_read_image(png_ptr, row_pointers);

        fclose(fp);
}


void process_file(void)
{
        if (png_get_color_type(png_ptr, info_ptr) == PNG_COLOR_TYPE_RGBA)
                abort_("[process_file] input file is PNG_COLOR_TYPE_RGB but must be PNG_COLOR_TYPE_RGB "
                       "(lacks the alpha channel)");

        if (png_get_color_type(png_ptr, info_ptr) != PNG_COLOR_TYPE_RGB)
                abort_("[process_file] color_type of input file must be PNG_COLOR_TYPE_RGB (%d) (is %d)",
                       PNG_COLOR_TYPE_RGBA, png_get_color_type(png_ptr, info_ptr));

        printf("width: %d\nheight: %d\n", width, height);

        int mHeader = 33; unsigned int mSize = 0;
        std::vector<int> mSizeByBites;
        for (y=0; y<height; y++) {
            png_byte* row = row_pointers[y];
            for (x=0; x<width; x++) {
                    png_byte* ptr = &(row[x*3]);
                    if(mHeader == 0){ break; }
                    mHeader-=3;

                    std::vector<int> b;

                    itob(ptr[0], b);
                    mSizeByBites.push_back(b[b.size()-1]);
                    b.clear();

                    itob(ptr[1], b);
                    mSizeByBites.push_back(b[b.size()-1]);
                    b.clear();

                    itob(ptr[2], b);
                    mSizeByBites.push_back(b[b.size()-1]);
                    b.clear();
                }
            if(mHeader == 0){ break; }
        }


        for(int i =0; i<mSizeByBites.size(); i++){
            printf("%d", mSizeByBites[i]);
        }
        btoi(mSize, mSizeByBites);
        printf(" = %i\n", mSize);

        std::vector<int> mDataBaBites;

        for (y=0; y<height; y++) {
            png_byte* row = row_pointers[y];
            for (x=0; x<width; x++) {
                if(mSize <= 0){ break; }

                png_byte* ptr = &(row[x*3]);
                std::vector<int> b;

                itob(ptr[0], b);
                mDataBaBites.push_back(b[b.size()-1]);
                b.clear();
                mSize--;
                if(mSize <= 0){ break; }
                itob(ptr[1], b);
                mDataBaBites.push_back(b[b.size()-1]);
                b.clear();
                mSize--;
                if(mSize <= 0){ break; }
                itob(ptr[2], b);
                mDataBaBites.push_back(b[b.size()-1]);
                b.clear();
                mSize--;
                if(mSize <= 0){ break; }
                printf("%i\n", mSize);
            }
            if(mSize<=0){ break; }
        }

        std::ofstream output("result.mp3", std::ios::out | std::ios::binary);

        printf("[D] Writing to file start:    %li\n", mDataBaBites.size());
        output.write( (char*)(&mDataBaBites[0]), mDataBaBites.size() );
        output.close();

}


int main(int argc, char **argv)
{

        read_png_file();
        process_file();

        return 0;
}
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#定义PNG_调试3
#包括
无效中止(常量字符*s,…)
{
va_列表参数;
va_启动(args,s);
vfprintf(stderr,s,args);
fprintf(标准格式,“\n”);
va_端(args);
中止();
}
无效itob(短n,标准::矢量和bin)
{
int d=n;
如果(n>1)
{
d=n%2;
itob(n/2,bin);
}
箱子。推回(d);
}
void btoi(无符号整数和数字,标准::矢量二进制)
{
n=0;
int k=32;
对于(int i=0;i
#include <fstream> //For std::min

std::ofstream mp3File( "restored.mp3", std::ios::out | std::ios::binary );
//Assuming rgbData is a char* with the mp3 data,
//and rgbDataSize is its size in bytes
mp3File.write( rgbData, rgbDataSize );
mp3File.close();
#包括//标准::分钟
流mp3文件的std::of(“restored.mp3”,std::ios::out | std::ios::binary);
//假设rgbData是带有mp3数据的字符*,
//rgbDataSize是以字节为单位的大小
mp3File.write(rgbData,rgbDataSize);
mp3File.close();
更新:当我们(程序员)说“二进制表示”时,我们几乎总是指字节,而不是位。根据您对解码过程的描述,我推断您应该比较每个像素的3个RGB分量,并将最小值保留为解码字节。为此:

#include <algorithm>

    //...

    std::vector<char> mDataBaBites;

    for (y=0; y<height; y++) {
        png_byte* row = row_pointers[y];
        for (x=0; x<width; x++) {
            png_byte red = row[x*3];
            png_byte green = row[x*3 + 1];
            png_byte blue = row[x*3 + 2];               
            png_byte minByte = std::min( std::min(red,green), blue );
            mDataBaBites.push_back( minByte );
            mSize -= 3;
        }
        if(mSize<=0){ break; }
    }

    std::ofstream output("result.mp3", std::ios::out | std::ios::binary);
    printf("[D] Writing to file start:    %li\n", mDataBaBites.size());
    output.write( (char*)(&mDataBaBites[0]), mDataBaBites.size() );
    output.close();
#包括
//...
std::载体mDataBaBites;

对于(y=0;yi如果mp3文件实际上只是png的位,你应该能够保存数据。只要确保你使用的是二进制模式。“二进制模式”是什么意思?他的意思是,你必须用std::ios::binary标志创建ofstream。我有一个向量来存储位:1010101000111…数据在位中。如果我这样做的话你说,我只是得到一个满是这些0和1的文件。\如果v[0]=1,也就是数字1,这将不起作用。该数字的二进制表示将被写入(根据目标平台,它将由32或64位组成),而不是1。你需要将mp3数据作为
向量
字符*
(纯二进制数据)读取。发布您的代码会很有帮助。通过更新的描述,我现在可以编写一个好的二进制文件。但仍然不是有效的mp3。这也不好。当我查看R、B或G值时,我必须将其转换为二进制。例如:R=10->1010。然后我必须选择最小的本地值:101[0]。这些数字来自每个像素,每个RGB值,连接起来,表示一个mp3文件数据。这就是我使用vector的原因。如果真的是这样,那么按8的组获取整数并从中创建一个字节如何?这就是更新2的作用。
    std::ofstream output("result.mp3", std::ios::out | std::ios::binary);
    printf("[D] Writing to file start:    %li\n", mDataBaBites.size());
    for( int i=0; i<mDataBaBites.size(); i+=8 ){
       char decodedByte = 0;
       for( int j=0; j<8; j++ )
          decodedByte |= (mDataBaBites[i+j] << j);
       output.write( (char*)(&mDataBaBites[0]), 1 );
    }
    output.close();