opencv二进制数据jpg图像到cv :: Mat

时间:2014-10-07 07:38:49

标签: c++ postgresql opencv image-processing libpqxx

我想在来自postgresql数据库的c ++ opencv中加载图像。 图像jpg扩展名作为二进制数据(bytea类型)存储在基础中,由于libpqxx,我可以访问它。

问题是我不知道如何将数据转换为cv :: Mat实例。使用常规图像我可以使用imread('myImage.jpg',...),但在这种情况下,我甚至无法在Mat的数据属性中加载数据库图像,因为它是jpeg而不是bmp。

有什么想法吗?是否有一些我可以使用的opencv方法可以直接理解二进制数据并将其转换为适当的结构? imdecode()函数似乎用于位图数据。

编辑:Berak,使用向量imdecode函数返回null Matrice会发生什么“如果缓冲区太短或包含无效数据,则返回空矩阵/图像。”这是代码:

pqxx::result r=bdd::requete("SELECT image FROM lrad.img WHERE id=3",1);//returns the bytea image in r[0]["image"]
const char* buffer=r[0]["image"].c_str();    
vector<uchar>::size_type size = strlen((const char*)buffer);
vector<uchar> jpgbytes(buffer, buffer+size);
Mat img = imdecode(jpgbytes, CV_LOAD_IMAGE_COLOR); 
//jpgbytes.size()=1416562 img.size()=[0 x 0]

我错过了什么?

2 个答案:

答案 0 :(得分:3)

仍然使用imdecode。它可以处理png,jpg,bmp,ppm,webp,jp2,exr,但没有gif。

vector<uchar> jpgbytes; // from your db
Mat img = imdecode(jpgbytes);

(你应该对bmp或任何其他支持的格式做同样的事情,不要乱用Mat的原始数据指针!)

答案 1 :(得分:1)

好的我有将bytea数据转换为cv :: Mat的过程,这里是代码。

inline int dec(uchar x){ //convert uchar to int
    if (x>='0'&&x<='9') return (x-'0');
    else if (x>='a'&&x<='f') return (x-'a'+10);
    else if (x>='A'&&x<='F') return (x-'A'+10);
    return 0;
}
cv::Mat bytea2Mat(const pqxx::result::field& f){
    const char* buffer=f.c_str();  
    vector<uchar>::size_type size = strlen((const char*)buffer);
    vector<uchar> jpgbytes(size/2-1); 

    for (size_t i=0; i!=size/2-1;i++) { 
        jpgbytes[i]=(dec(buffer[2*(i+1)])<<4)+dec(buffer[2*(i+1)+1]);
    }
    cout <<size/2<<";"<<jpgbytes.size()<<endl;
    return imdecode(jpgbytes, CV_LOAD_IMAGE_COLOR);
}

对于原始输入字符串“a b0”(hexa形式),bytea输出被加密为char *,类似于“\ x41204230”。 (根据数据输入,\ x可能不存在) 要获取原始数据,您需要计算两个char的原始输入,('4','1'= 0x41 = 65)。向量是char *的一半大小。