gcc相关运行原理及linux系统下opencv使用

一、任务要求

一. 学习并掌握可执行程序的编译、组装过程。学习任务如下

1)阅读、理解和学习材料“用gcc生成静态库和动态库.pdf”和“静态库.a与.so库文件的生成与使用.pdf”,请在Linux系统(Ubuntu)下如实仿做一遍。

2)在第一次作业的程序代码基础进行改编,除了x2x函数之外,再扩展写一个x2y函数(功能自定),main函数代码将调用x2x和x2y ;将这3个函数分别写成单独的3个 .c文件,并用gcc分别编译为3个.o 目标文件;将x2x、x2y目标文件用 ar工具生成1个 .a 静态库文件, 然后用 gcc将 main函数的目标文件与此静态库文件进行链接,生成最终的可执行程序,记录文件的大小。

3)将x2x、x2y目标文件用 ar工具生成1个 .so 动态库文件, 然后用 gcc将 main函数的目标文件与此动态库文件进行链接,生成最终的可执行程序,记录文件的大小,并与之前做对比。

二. Gcc不是一个人在战斗。请说明gcc编译工具集中各软件的用途,了解EFF文件格式。学习任务如下:阅读、理解和学习材料“Linux GCC常用命令.pdf”和“GCC编译器背后的故事.pdf”,如实仿做一遍。

三. (综合实践)每一个程序背后都站着一堆优秀的代码库。通过学习opencv图像库编程,了解如何借助第三方库函数完成一个综合程序设计。“学了opencv,妈妈再不担忧你不会图像编程啦!”。 在Ubuntu16/18系统下练习编译、安装著名的C/C++图像处理开源软件库 Opencv3.x (过程多,耗时长,需要耐心和细心)。安装成功后:

  1. 编写一个打开图片进行特效显示的代码 test1.cpp(见opencv编程参考资料 ); 注意gcc编译命令: gcc test1.cpp -o test1 pkg-config --cflags --libs opencv 1)请解释这条编译命令,它是如何获得opencv头文件、链接lib库文件的路径的? 2)改用make+makefile方式编译 上述程序(用变量命名格式写makefile文件,并包括 clean选项)
  2. 练习使用opencv库编写打开摄像头压缩视频的程序。参考示例代码1和示例代码2。并回答:1)如果要求打开你硬盘上一个视频文件来播放,请问示例代码1第7行代码如何修改?2)在示例代码1第9行的while循环中,Mat是一个什么数据结构? 为什么一定要加一句waitKey延时代码,删除它行不行?3)示例代码1代码会在while循环中一直运行,你如果试图用鼠标关闭图像显示窗口,会发现始终关不掉。需要用键盘Ctrl+C 强制中断程序,非常不友好。如何改进? 3. 掌握git使用方法,在gitee/github网站创建自己的账号和仓库,将以上作业代码分类上传到仓库中。

二、实验过程

2.1学习并掌握可执行程序的编译、组装过程

2.1.1gcc生成可执行的动态静态库

A:创建hello.h、hello.c 和 main.c文件

程序 1: hello.h
#ifndef HELLO_H
#define HELLO_H
void hello(const char *name);
#endif //HELLO_H
程序 2: hello.c
#include <stdio.h>
void hello(const char *name)
{
	printf("Hello %s!\n", name);
}
程序 3: main.c
#include "hello.h"
int main()
{
	hello("everyone");
	return 0;
}

**B:**第 2 步:将 hello.c 编译成 .o文件

gcc -c hello.c

可以很清晰的看到生成了一个hello.o的文件

C:第 3 步:由.o 文件创建静态库

创建静态库用 ar 命令。在系统提示符下键入以下命令将创建静态库文件 libmyhello.a

ar -crv libmyhello.a hello.o

可以很清晰的看到生成了.a的文件

linux 加载opencv so文件 java_linux

D:第 4 步,程序中使用静态库

gcc -o main main.c libmyhello.a

得到结果

linux 加载opencv so文件 java_opencv_02

E:第 5 步:.s文件的建立

gcc -c -fpic hello.c
gcc -shared *.o -o libsofile.s

得到了.s的文件,如图:

linux 加载opencv so文件 java_linux_03

F:第6步,程序使用动态库

gcc -o hello main.c  libsofile.so
./hello

出现报错,报错原因./hello: error while loading shared libraries: libmyhello.so: cannot open shar ed object file: No such file or director

解决方案:是找不到动态库文件 libmyhello.so。程序在运行时, 会在/usr/lib 和/lib 等目录中查找需要的动态库文件。若找到,则载入动态库,否则将提 示类似上述错误而终止程序运行。我们将文件 libmyhello.so 复制到目录

/user/lib中即可

mv libsofile.so /user/lib
idconfig

最终结果

linux 加载opencv so文件 java_计算机视觉_04

2.1.2生成静态和动态文件并进行链接

A:创建sub1.h、sub2.h、main.c、sub1.c、sub2.c文件:

vi main.c 
vi sub1.h
vi sub2.h
vi sub1.c
vi sub2.c

对应的文件内容如下:

sub1.h:
#include<stdio.h>

int add(int a,int b);

sub2.h:
#include<stdio.h>

int minux(int a,int b);

sub1.c:
#include"sub1.h"

int add(int a,int b){
        return a+b;
}

sub2.c:
#include"sub1.h"

int minus(int a,int b){
        return a-b;
}

main.c:
#include<stdio.h>
#include"sub1.h"
#include"sub2.h"

int main()
{
   int a =1,b =2;
   int c = add(a,b);
   int d = minus(a,b);
   printf("%d\n",c);
   printf("%d",d);
   return 0;
}

B:静态库.a文件的使用

首先生成.o文件,指令如下:

gcc -c sub1.c sub2.c

之后生成静态库的.a文件,指令如下:

ar crv libafile.a sub1.o sub2.o

这样就生成了一个.a文件,如图:

linux 加载opencv so文件 java_静态库_05

使用.a文件创建可执行程序并执行

gcc -o main main.c libafile.a
./main

最终结果为3、-1结果完全正确。

使用size记录文件大小

linux 加载opencv so文件 java_计算机视觉_06

C:动态库.so文件使用

首先生成目标文件:

gcc -c -fpic sub1.c sub2.c

生成共享库.so 文件 :

gcc -shared *.o -o libsofile.so

使用.so 库文件,创建可执行程序 :

gcc -o test main.c libsofile.so
./test

出现报错

./test: error while loading shared libraries: libsofile.so: cannot open shared object file: No such file or director

出现这个的原因是由于 linux 自身系统设定的相应的设置的原因,即其只在/lib and /usr/lib 下搜索对应 的.so 文件,故需将对应 so 文件拷贝到对应的路径就行了

sudo cp libsofile.so /user/lib

再次执行./test,即可成功运行。

使用size查看可执行文件的大小:

linux 加载opencv so文件 java_opencv_07

上述.a和.so生成的文件有着细微的差别,但并没有很大的差距

2.2Gcc不是一个人在战斗。请说明gcc编译工具集中各软件的用途

(1)centos环境下安装gcc

要编译c语言程序,就必须要在linux系统下安装gcc,安装指令如下:

yum install gcc
(2)创建main.c文件并进行gcc操作的步骤

A:创建main.c文件,使用vi创建main.c文件

vi main.c

main.c文件内容如下:

#include <stdio.h>
//此程序很简单,仅仅打印一个 Hello World 的字符串。
int main(void)
{
printf("Hello World! \n");
return 0;
}

B:预处理过程

预处理过程会将mian.c转化为main.i文件,对应的指令如下:

gcc -E main.c -o main.i

我们可以打开main.i文件查看情况:

linux 加载opencv so文件 java_计算机视觉_08

预处理过程做的事情如下:

(1) 将所有的#define 删除,并且展开所有的宏定义,并且处理所有的条件预编 译指令,比如#if #ifdef #elif #else #endif 等。

(2) 处理#include 预编译指令,将被包含的文件插入到该预编译指令的位置。

(3) 删除所有注释“//”和“/* */”。

(4) 添加行号和文件标识,以便编译时产生调试用的行号及编译错误警告行号。

(5) 保留所有的#pragma 编译器指令,后续编译过程需要使用它们

C编译过程

编译过程就是对预处理完的文件进行一系列的词法分析,语法分析,语义分析及 优化后生成相应的汇编代码。

编译过程会将我们处理的mian.c文件得到的main.i文件做进一步的处理,得到main.s的文件:

gcc -S main.i -o main.s

打开main.s文件查看内容:

linux 加载opencv so文件 java_#include_09

main.s部分内容如图

D:汇编:汇编过程调用对汇编代码进行处理,生成处理器能识别的指令,保存在后缀为.o 的目标文件中。由于每一个汇编语句几乎都对应一条处理器指令,因此,汇编相 对于编译过程比较简单,通过调用 Binutils 中的汇编器 as 根据汇编指令和处理 器指令的对照表一一翻译即可

汇编可以将main.s的文件转化为main.o的文,指令如下:

gcc -c main.s -o main.o -v

linux 加载opencv so文件 java_计算机视觉_10

E动态链接和静态链接

(1) 静态链接是指在编译阶段直接把静态库加入到可执行文件中去,这样可执行 文件会比较大。链接器将函数的代码从其所在地(不同的目标文件或静态链 接库中)拷贝到最终的可执行程序中。为创建可执行文件,链接器必须要完 成的主要任务是:符号解析(把目标文件中符号的定义和引用联系起来)和 重定位(把符号定义和内存地址对应起来然后修改所有对符号的引用)。

(2) 动态链接则是指链接阶段仅仅只加入一些描述信息,而程序执行时再从系统 中把相应动态库加载到内存中去

执行动态链接指令如下:

gcc main.c -o main

生成的main的可执行文件,我们可以使用size查看动态链接的大小:

size main

对应的结果截图如图所示:

linux 加载opencv so文件 java_opencv_11

ldd可以查看连接了多少其他的动态库:

ldd main

对应的结果如下:

linux 加载opencv so文件 java_计算机视觉_12

执行生成静态链接的执行如下:

gcc -static main.c -o main

使用size指令查看大小:

size main

对应的结果如下所示:

linux 加载opencv so文件 java_opencv_13

我门可以看到静态链接的大小要远远大于动态链接生成的大小

我们使用ldd指令看是否链接 了动态库:

ldd main

linux 加载opencv so文件 java_计算机视觉_14

说明没有链接动态库

F分析ELF的文件

ELF 文件格式如下图所示,位于 ELF Header 和 Section Header Table 之间的都 是段(Section)。一个典型的 ELF 文件包含下面几个段: .text:已编译程序的指令代码段。 .rodata:ro 代表 read only,即只读数据(譬如常数 const)。 .data:已初始化的 C 程序全局变量和静态局部变量。 .bss:未初始化的 C 程序全局变量和静态局部变量。 .debug:调试符号表,调试器用此段的信息帮助调试 :

linux 加载opencv so文件 java_#include_15

可以使用 readelf -S 查看其各个 section 的信息:

readelf -S main

linux 加载opencv so文件 java_静态库_16

反编译:由于 ELF 文件无法被当做普通文本文件打开,如果希望直接查看一个 ELF 文件包 含的指令和数据,需要使用反汇编的方法。 使用 objdump -D 对其进行反汇编如:

objdump -D main

linux 加载opencv so文件 java_linux_17

使用 objdump -S 将其反汇编并且将其 C 语言源代码混合显示 :

gcc -o main -g main.c 
objdump -S main

linux 加载opencv so文件 java_静态库_18

2.3Linux系统下opencv图像编程

2.3.1编写一个打开图片进行特效显示的代码 test.cpp

首先创建test.cpp文件内容:

#include <opencv2/highgui.hpp>
#include <opencv2/opencv.hpp>
using namespace cv;
using namespace std;
int main(int argc, char** argv)
{
	CvPoint center;
    double scale = -3; 

	IplImage* image = cvLoadImage("lena.jpg");
	argc == 2? cvLoadImage(argv[1]) : 0;
	
	cvShowImage("Image", image);
	
	
	if (!image) return -1; 	center = cvPoint(image->width / 2, image->height / 2);
	for (int i = 0;i<image->height;i++)
		for (int j = 0;j<image->width;j++) {
			double dx = (double)(j - center.x) / center.x;
			double dy = (double)(i - center.y) / center.y;
			double weight = exp((dx*dx + dy*dy)*scale);
			uchar* ptr = &CV_IMAGE_ELEM(image, uchar, i, j * 3);
			ptr[0] = cvRound(ptr[0] * weight);
			ptr[1] = cvRound(ptr[1] * weight);
			ptr[2] = cvRound(ptr[2] * weight);
		}

	Mat src;Mat dst;
	src = cvarrToMat(image);
	cv::imwrite("test.png", src);

    cvNamedWindow("test",1);  	imshow("test", src);
	cvWaitKey();
	return 0;
}

使用指令编译并显示:

g++ test1.cpp -o test1 `pkg-config --cflags --libs opencv`
./test1

结果如下报错:

error: (-2:Unspecified error) Can't initialize GTK backend in function 'cvInitSystem

发生报错的原因是因为我这里使用的是远程主机进行的连接,使用的是root用户的权限进行操作,而我的主机此时也正在使用root用户,导致的报错。

解决办法更换远程连接用户即可,使得两边用户不一致即可解决问题。

运行结果如图所示:

linux 加载opencv so文件 java_opencv_19

:注意gcc编译命令: gcc test1.cpp -o test1 pkg-config --cflags --libs opencv 请解释这条编译命令,它是如何获得opencv头文件、链接lib库文件的路径的

:pck-config a. 检查库的版本号。如果所需要的库的版本不满足要求,它会打印出错误信息,避免链接错误版本的库文件。b. 获得编译预处理参数,如宏定义,头文件的位置。c. 获得链接参数,如库及依赖的其它库的位置,文件名及其它一些连接参数。d. 自动加入所依赖的其它库的设置。在该文件夹里面有个opencv.pc的文件,其实这就是pkg-config下OpenCV的配置文件。选项–cflags 它是用来指定程序在编译时所需要头文件所在的目录,选项 --libs则是指定程序在链接时所需要的动态链接库的目录。

2.3.1练习使用opencv库编写打开摄像头压缩视频的程序。

要编译此段代码,首先需要打开虚拟机的摄像头才能进行操作,否则就会发生报错,导致无法运行。

我这里使用的是workstation15.5的版本,具体版本操作近乎一致。

按照图中的顺序将在状态栏中显示勾上即可

linux 加载opencv so文件 java_linux_20

使用vim工具编写代码,代码如下:

#include<iostream>
#include <opencv2/opencv.hpp>
#include<opencv2/core/core.hpp>
#include<opencv2/highgui/highgui.hpp>
using namespace cv;
using namespace std;

void main()
{
	//打开电脑摄像头
	VideoCapture cap(0);
	if (!cap.isOpened())
	{
		cout << "error" << endl;
		waitKey(0);
		return;
	}

	//获得cap的分辨率
	int w = static_cast<int>(cap.get(CV_CAP_PROP_FRAME_WIDTH));
	int h = static_cast<int>(cap.get(CV_CAP_PROP_FRAME_HEIGHT));
	Size videoSize(w, h);
	VideoWriter writer("RecordVideo.avi", CV_FOURCC('M', 'J', 'P', 'G'), 25, videoSize);
	
	Mat frame;
	int key;//记录键盘按键
	char startOrStop = 1;//0  开始录制视频; 1 结束录制视频
	char flag = 0;//正在录制标志 0-不在录制; 1-正在录制

	while (1)
	{
		cap >> frame;
		key = waitKey(100);
		if (key == 32)//按下空格开始录制、暂停录制   可以来回切换
		{
			startOrStop = 1 - startOrStop;
			if (startOrStop == 0)
			{
				flag = 1;
			}
		}
		if (key == 27)//按下ESC退出整个程序,保存视频文件到磁盘
		{
			break;
		}

		if (startOrStop == 0 && flag==1)
		{
			writer << frame;
			cout << "recording" << endl;
		}
		else if (startOrStop == 1)
		{
			flag = 0;
			cout << "end recording" << endl;
			
		}
		imshow("picture", frame);
	}
	cap.release();
	writer.release();
	destroyAllWindows();
}

结果会生成一个avi的视频文件,使用指令编译并运行这个文件:

g++ test3.cpp -o test3 `pkg-config --cflags --libs opencv`
./test3

我们可以查看当前下是否存在对应的avi文件

linux 加载opencv so文件 java_#include_21

确实存在,说明成功。

:1)如果要求打开你硬盘上一个视频文件来播放,请问第7行代码如何修改?

2)在第9行的while循环中,Mat是一个什么数据结构? 为什么一定要加一句waitKey延时代码,删除它行不行?

3)此代码会在while循环中一直运行,你如果试图用鼠标关闭图像显示窗口,会发现始终关不掉。需要用键盘Ctrl+C 强制中断程序,非常不友好。如何改进?

:1)播放视频问津只需将代码中VideoCapture cap(0)中的0更改为播放的路径即可

2)在这个代码中Mat是一个类。由两部分数据组成:矩阵头(包括矩阵尺寸、存储方法、存储地址等信息)和一个指向所有像素值的矩阵(根据所选存储方法不同,矩阵可以是不同的维数)的指针。
waitkey是图像的播放时间,如果不使用,图片无法进行展现

3)使用if (key == 27)break;使用这个语句当我们按下esc的使用,按下esc就可以退出程序了。当然其中也可以将27改为其他的数字,只需按下键盘对应的键就可以退出

config --cflags --libs opencv`
./test3

我们可以查看当前下是否存在对应的avi文件

linux 加载opencv so文件 java_#include_22

确实存在,说明成功。

:1)如果要求打开你硬盘上一个视频文件来播放,请问第7行代码如何修改?

2)在第9行的while循环中,Mat是一个什么数据结构? 为什么一定要加一句waitKey延时代码,删除它行不行?

3)此代码会在while循环中一直运行,你如果试图用鼠标关闭图像显示窗口,会发现始终关不掉。需要用键盘Ctrl+C 强制中断程序,非常不友好。如何改进?

:1)播放视频问津只需将代码中VideoCapture cap(0)中的0更改为播放的路径即可

2)在这个代码中Mat是一个类。由两部分数据组成:矩阵头(包括矩阵尺寸、存储方法、存储地址等信息)和一个指向所有像素值的矩阵(根据所选存储方法不同,矩阵可以是不同的维数)的指针。
waitkey是图像的播放时间,如果不使用,图片无法进行展现

3)使用if (key == 27)break;使用这个语句当我们按下esc的使用,按下esc就可以退出程序了。当然其中也可以将27改为其他的数字,只需按下键盘对应的键就可以退出

三、相关总结

这次的实验对我来说难度还是比较大的,特别是安装虚拟机环境和opencv的环境下。一开始使用的centos系统一堆报错,无法运行。中途更换了ubuntu系统,才完成实验。下载opencv及运行时也出过各种各样的错误,但是在最终花了许多时间也还是解决了问题。