在tensorflow中,輸入數(shù)據(jù)格式為Tensor格式悬钳,但是使用opencv讀入的圖片是Mat格式,需要對其進行轉(zhuǎn)換。
1 Mat轉(zhuǎn)Tensor
1.1 方法一
使用循環(huán)進行賦值冀瓦,輸入的image為3通道彩色圖片,所以對應(yīng)的Tensor大小為1*Size*Size*3宾巍,同時image為BGR格式咕幻,轉(zhuǎn)化為RGB格式。
void mat2Tensor(Mat &image, Tensor &t) {
resize(image, image, Size(IMAGE_SIZE, IMAGE_SIZE)); //對圖片進行縮放
auto output = t.shaped<float, 4>({ 1, IMAGE_SIZE, IMAGE_SIZE, 3});
for (int i = 0; i < image.rows; ++i) {
for (int j = 0; j < image.cols; ++j) {
for (int k = 0; k < 3; ++k) {
output(0, i, j, k) = image.at<Vec3b>(i, j)[2-k];
}
}
}
}
1.2 方法二
首先將圖片由BGR轉(zhuǎn)化為RGB格式顶霞,然后創(chuàng)建一個指向Tensor變量數(shù)據(jù)的指針肄程,以這個指針來創(chuàng)建fake_mat對象,然后將image復制給fake_mat选浑,這樣就將數(shù)據(jù)放到Tensor的數(shù)據(jù)地址中了蓝厌。
cvtColor(image, image, cv::COLOR_BGR2RGB);
float *tensor_data_ptr = t.flat<float>().data();
cv::Mat fake_mat(image.rows, image.cols, CV_32FC(image.channels()), tensor_data_ptr);
image.convertTo(fake_mat, CV_32FC(image.channels()));
2 Tensor轉(zhuǎn)Mat
2.1 方法一
也是使用循環(huán),對其一個一個進行賦值古徒。
void tensor2Mat(Tensor &t, Mat &image) {
auto output = t.tensor<int, 3>(); // (1,512,512)
for (int i = 0; i < IMAGE_SIZE; ++i) {
for (int j = 0; j < IMAGE_SIZE; ++j) {
image.at<uchar>(i, j) = output(0, i, j);
}
}
}
2.2 方法二
拷貝地址拓提,分別使用指向Mat和Tensor的數(shù)據(jù)存放地址的指針,然后使用memcpy函數(shù)進行復制數(shù)據(jù)隧膘。
void tensor2Mat(Tensor &t, Mat &image) {
image.convertTo(image, CV_32FC1);
tensorflow::StringPiece tmp_data = t.tensor_data();
memcpy(image.data,const_cast<char*>(tmp_data.data()),IMAGE_SIZE * IMAGE_SIZE * sizeof(float));
image.convertTo(image, CV_8UC1);
}
2.3 方法三
這個方法我覺得是最高效且優(yōu)雅的代态,使用指針指向Tensor的數(shù)據(jù)地址,然后使用Mat的構(gòu)造函數(shù)疹吃,將這個地址傳進去蹦疑,就直接得到了Mat變量。
void tensor2Mat(Tensor &t, Mat &image) {
int *p = t.flat<int>().data();
image = Mat(IMAGE_SIZE, IMAGE_SIZE, CV_32SC1, p);
image.convertTo(image, CV_8UC1);
}