Sending Images Using Sockets In Real Time
Introduction
Configuration
Contents
- Part 1
- Use OpenCV with Visual Studio 2017
- Build boost with VS2017
- Sending images with sockets
- Part 2
- Sockets on Windows and Linux
- Test Speed
- Increase transfer speed by compressing images
Use OpenCV with VS 2017
- Download binary files of OpenCV from internet.
- Unzip files.
- Add path of
bin
to environment variablePATH
- Include OpenCV headers and libs
- Add additional dependencies
Build Boost on Windows with VS2017
- Select architecture
- Select versions: x86 or x64
- Select
release
version ordebug
version - Select components
- Use boost in VS2017
Using sockets on Windows: winsock
Refs
http://www.binarytides.com/winsock-socket-programming-tutorial/
Using sockets on Linux
Refs
https://stackoverflow.com/questions/15445207/sending-image-jpeg-through-socket-in-c-linux
Connect Two PCs with Crossover Cable
Sending Image Files
Sending Uchar Array
Test Speed
Compress Images
My final code block:
std::string pic("BMW_M6_G-power_1082_1280x960.jpg");
Mat img;
img = imread(pic.c_str(), IMREAD_COLOR);
namedWindow("Display window", WINDOW_AUTOSIZE); // Create a window for display.
imshow("Display window", img); // Show our image inside it.
waitKey(0); //
uchar *ptr = img.data;
int len = img.rows * img.cols * img.channels();
std::cout << "size of image: " << " " << len << std::endl;
printf("\n");
//************Compress images*********
std::vector<uchar> buff;//buffer for coding
std::vector<int> param(2);
param[0] = cv::IMWRITE_JPEG_QUALITY;
param[1] = 80;//default(95) 0-100
auto start = std::chrono::high_resolution_clock::now();
cv::imencode(".jpeg", img, buff, param);
auto finish = std::chrono::high_resolution_clock::now();
float te = std::chrono::duration_cast<std::chrono::milliseconds>(finish - start).count();
printf("Compression Consumes %2.3fms\n", te);
std::cout << "Compressed Length = "<< buff.size() << std::endl;
uchar* imageBuf;
//imageBuf = (unsigned char *)realloc(imageBuf, buff.size());
//memcpy(imageBuf, &buff[0], buff.size());
cv::Mat imgbuf = Mat(1280, 960, CV_8UC3, buff.data());
cv::Mat imgMat = cv::imdecode(Mat(buff), CV_LOAD_IMAGE_COLOR);
std::cout << "de size = "<< imgMat.rows*imgMat.cols*imgMat.channels() <<std::endl;
//Mat dec;
//dec = imdecode(buff, cv::IMREAD_COLOR);
namedWindow("Display window2", WINDOW_AUTOSIZE);
imshow("Display window2", imgMat);
waitKey(0); //
References from internet:
std::vector<uchar> buff;//buffer for coding
std::vector<int> param(2);
param[0] = cv::IMWRITE_JPEG_QUALITY;
param[1] = 80;//default(95) 0-100
cv::imencode(".jpg", mat, buff, param);
C version:
#include <opencv/cv.h>
#include <opencv/highgui.h>
int
main(int argc, char **argv)
{
char *cvwin = "camimg";
cvNamedWindow(cvwin, CV_WINDOW_AUTOSIZE);
// setup code, initialization, etc ...
[ ... ]
while (1) {
// getImage was my routine for getting a jpeg from a camera
char *img = getImage(fp);
CvMat mat;
// substitute 640/480 with your image width, height
cvInitMatHeader(&mat, 640, 480, CV_8UC3, img, 0);
IplImage *cvImg = cvDecodeImage(&mat, CV_LOAD_IMAGE_COLOR);
cvShowImage(cvwin, cvImg);
cvReleaseImage(&cvImg);
if (27 == cvWaitKey(1)) // exit when user hits 'ESC' key
break;
}
cvDestroyWindow(cvwin);
}
// decode jpg (or other image from a pointer)
// imageBuf contains the jpg image
cv::Mat imgbuf = cv::Mat(480, 640, CV_8U, imageBuf);
cv::Mat imgMat = cv::imdecode(imgbuf, CV_LOAD_IMAGE_COLOR);
// imgMat is the decoded image
// encode image into jpg
cv::vector<uchar> buf;
cv::imencode(".jpg", imgMat, buf, std::vector<int>() );
// encoded image is now in buf (a vector)
imageBuf = (unsigned char *) realloc(imageBuf, buf.size());
memcpy(imageBuf, &buf[0], buf.size());
// size of imageBuf is buf.size();
>>> img_str = cv2.imencode('.jpg', img)[1].tostring()
>>> type(img_str)
'str'
>>> nparr = np.fromstring(STRING_FROM_DATABASE, np.uint8)
>>> img = cv2.imdecode(nparr, cv2.CV_LOAD_IMAGE_COLOR)
https://stackoverflow.com/questions/801199/opencv-to-use-in-memory-buffers-or-file-pointers