将OpenCV的代码从C++移植到C语言过程小记

2024-06-23 11:28

本文主要是介绍将OpenCV的代码从C++移植到C语言过程小记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

    • 1、需求背景
    • 2、移植成C语言
      • 2.0 移除namespace特性
      • 2.1 移除template特性
      • 2.2 移除class特性
      • 2.3 移除vector特性
      • 2.4 移除std函数
      • 2.5 移除引用传值
      • 2.6 移除auto类型推导
      • 2.7 C++中的关键词
      • 2.8 其他语言差异
    • 3、测试和杂项
      • 3.1 C语言和C++编译兼容
      • 3.2 GCC用C语言规则编译源文件
      • 3.3 修改文件名后缀
    • 4、最终代码

1、需求背景

最近有个项目要求将OpenCV的一个函数移植成由C语言编写,虽然我很不想干这件事情,但是毕竟是工作,还是得干。

本篇文章分享一下将C++代码移植成C语言的基本思路,和过程中遇到的问题。

2、移植成C语言

2.0 移除namespace特性

这个很简单,直接删。

2.1 移除template特性

C++的模板编程可以将多种类型的输入编译成分别对应的函数。

在C语言中,模板编程往往只会被解释成其中的一种。当然,如果被解释成多种,那就创建同样数量的类或函数就可以了。不同的类或函数可以增加不同的后缀以区分,例如 Point2f 表示 float 类型、Point2i 表示 int 类型。

C++代码:

template<typename _Tp> class Point_
{
public:_Tp x;_Tp y;Point_() : x(0), y(0) {}Point_(_Tp _x, _Tp _y) : x(_x), y(_y) {}Point_<_Tp> operator-(const Point_<_Tp>& other) {return Point_<_Tp>(x - other.x, y - other.y);}float cross(const Point_& other) const {return x * other.y - y * other.x;}
};typedef Point_<float> Point2f;

移除模板特性:

class Point2f
{
public:Point2f() = default;Point2f(float _x, float _y) : x(_x),  y(_y) {}float x = 0.f;float y = 0.f;Point2f operator-(const Point2f& other) const {return Point2f(x - other.x, y - other.y);}float cross(const Point2f& other) const {return x * other.y - y * other.x;}
};

2.2 移除class特性

C++的class主要实现了类属性内置方法重载操作符,这3个特性。

类属性可以用 typedef 的结构体实现。并且这一步可以不用一步到位,可以在所有的类方法逐渐移植完毕之后,最后再做修改。

内置方法我改为同前缀的一个一个单独的函数,将自己的结构体传入又传出。

重载运算符我采用Python命名风格,例如对减号的操作符重载,实现为 Point_sub 这样的形式。

C++代码:

// 定义类属性、重载运算符、和类方法
class Point2f
{
public:Point2f() = default;Point2f(float _x, float _y) : x(_x),  y(_y) {}float x = 0.f;float y = 0.f;Point2f operator-(const Point2f& other) const {return Point2f(x - other.x, y - other.y);}float cross(const Point2f& other) const {return x * other.y - y * other.x;}
};// 调用
Point2f p1, p2;
Point2f p3 = p2 - p1;
float result = p1.cross(p2);

移除class特性:

// 类变成结构体
typedef struct {float x, y;
} Point2f;// 重载运算符变成传参函数
Point2f Point2f_sub(const Point2f& self, const Point2f& other) {auto x = self.x, y = self.y;return Point2f{x - other.x, y - other.y};
}// 类方法变成传参函数
float Point2f_cross(const Point2f& self, const Point2f& other) {auto x = self.x, y = self.y;return x * other.y - y * other.x;
}// 调用
Point2f p1, p2;
Point2f p3 = Point2f_sub(p2, p1);
float result = Point2f_cross(p1, p2);

2.3 移除vector特性

vector 是动态数组,必然不可能重构成支持弹性数组的结构、同时还能支持 [] 符号按下标取出元素。因为弹性数组必然导致内存重新分配,那么首地址就会改变。而C语言中 [] 符号只是地址求值的语法糖,肯定会取出错误的结果。

完全等效的做法我可以实现一个 vector 结构体,里面包含数据长度(length)和数据区(data)两个属性。在 push_back 的时候实现内存重新分配,并对 length 和 data 重新赋值。在取元素的时候从 vector->data 位置取出数据,这样就可以实现内存重新分配也不会影响数据区的取值。

但是这样过于复杂,对源代码改动量也过大(求值部分需要依次增加 ->data),我不如评估一个数组的最大长度,然后将数组首地址数组长度的指针一起传入函数,在 vector 需要变化长度(例如 push_back)的时候,同步修改这个长度数据。

C++代码:

int foo(std::vector<Point2f> &points)
{if(!points.empty())points.resize(4);points.push_back(Point2f(xi, yi));int N = points.size();std::vector<float> distPt(N);float* pDist = distPt.data();
}// 调用
std::vector<Point2f> points;
foo(points);
printf("Size=%d\n", points.size());

移除vector依赖:

int foo(Point2f* points, int* points_size)
{if (points_size[0])points_size[0] = 4;points[points_size[0]++] = Point2f(xi, yi);int N = points_size[0];float distPt[N];float* pDist = distPt;
}// 调用
Point2f points[100];  // 申请足够大的空间
int points_size = 0;
foo(points, &points_size);
printf("Size=%d\n", points_size);

需要注意的是,*array_size++array_size[0]++ 的计算不等效,注意运算符的计算顺序。

另外,如果vector传入函数之后并未发生长度改变(例如没有 push_back),那么数据长度可以不用传入数字指针类型,直接传入数字本身也行。

2.4 移除std函数

需要注意 std::abs 对应的不是 abs 函数,而是 fabs,否则返回值会成为整数。maxmin 同理。

C++C
#include <iostream>#include <stdio.h>
std::cout << "Name: " << name << std::endl;printf(“Name: %s\n”, name);
std::cout << "Age: " << age << std::endl;printf(“Age: %d\n”, age);
#include <algorithm>#include <math.h>
std::absfabs
std::minfmin
std::maxfmax
std::ininfininf
std::isnanisnan
std::swapTYPE temp = a; a = b; b = temp;

2.5 移除引用传值

C++为了减少拷贝、和提高语法的可读性,引入了 &parameter 这样的引用传参形式。

在C语言中和传入 &parameter 完全等效的写法,是传入指针,然后在函数内求地址。

但是如果传入的是 const &parameter 类型,说明 &parameter 在函数内没有被更改过,所以直接把 & 去掉就可以了(也许C++这么写的考虑是减少内存拷贝次数?我不太清楚)。

2.6 移除auto类型推导

C语言中的 auto 关键字表示的实际上是 int,需要将C++中所有出现 auto 的地方替换为真正需要的类型。

至于怎么类型推导,那当然是人肉推导啦(笑)。

2.7 C++中的关键词

C++中比C语言多几个关键词,可以用宏定义来替代:

#define bool char
#define true 1
#define false 0
#define nullptr 0

2.8 其他语言差异

C++代码:

return Point2f(x, y);

移植为C语言:

Point2f result = {x, y};
return result;  // 无法直接赋值返回

3、测试和杂项

3.1 C语言和C++编译兼容

要在C++中调用C语言编译文件中的程序,需要在头文件中增加一段:

#ifdef __cplusplus
extern "C" {
#endif// 函数原型定义
int foo(int arg1, float arg2, char* arg3);#ifdef __cplusplus
}
#endif

3.2 GCC用C语言规则编译源文件

使用 -x 参数可以指定按照特定语言规则编译某种文件,多次指定 -x 参数后则可混合规则编译:

g++ -xc source.cpp -xc++ test.cpp -o test.exe

此命令可将 source.cpp 文件依照C语言规则编译,test.cpp 依照C++规则编译,最后通过C++规则进行链接,生成为 test.exe 文件。

3.3 修改文件名后缀

.cpp 后缀的文件修改为 .c 后缀即可,可避免某些编译器只能通过文件名后缀识别代码语言。

4、最终代码

最终结果用约400行C语言代码完成。

代码仓库可见GitHub:

https://github.com/znsoooo/opencv-calc-rotated-iou

分步修改步骤可见提交记录:

https://github.com/znsoooo/opencv-calc-rotated-iou/commits/master

这篇关于将OpenCV的代码从C++移植到C语言过程小记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1087026

相关文章

C语言中联合体union的使用

本文编辑整理自: http://bbs.chinaunix.net/forum.php?mod=viewthread&tid=179471 一、前言 “联合体”(union)与“结构体”(struct)有一些相似之处。但两者有本质上的不同。在结构体中,各成员有各自的内存空间, 一个结构变量的总长度是各成员长度之和。而在“联合”中,各成员共享一段内存空间, 一个联合变量

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

C++的模板(八):子系统

平常所见的大部分模板代码,模板所传的参数类型,到了模板里面,或实例化为对象,或嵌入模板内部结构中,或在模板内又派生了子类。不管怎样,最终他们在模板内,直接或间接,都实例化成对象了。 但这不是唯一的用法。试想一下。如果在模板内限制调用参数类型的构造函数会发生什么?参数类的对象在模板内无法构造。他们只能从模板的成员函数传入。模板不保存这些对象或者只保存他们的指针。因为构造函数被分离,这些指针在模板外

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

uniapp接入微信小程序原生代码配置方案(优化版)

uniapp项目需要把微信小程序原生语法的功能代码嵌套过来,无需把原生代码转换为uniapp,可以配置拷贝的方式集成过来 1、拷贝代码包到src目录 2、vue.config.js中配置原生代码包直接拷贝到编译目录中 3、pages.json中配置分包目录,原生入口组件的路径 4、manifest.json中配置分包,使用原生组件 5、需要把原生代码包里的页面修改成组件的方

C/C++的编译和链接过程

目录 从源文件生成可执行文件(书中第2章) 1.Preprocessing预处理——预处理器cpp 2.Compilation编译——编译器cll ps:vs中优化选项设置 3.Assembly汇编——汇编器as ps:vs中汇编输出文件设置 4.Linking链接——链接器ld 符号 模块,库 链接过程——链接器 链接过程 1.简单链接的例子 2.链接过程 3.地址和

C++必修:模版的入门到实践

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:C++学习 贝蒂的主页:Betty’s blog 1. 泛型编程 首先让我们来思考一个问题,如何实现一个交换函数? void swap(int& x, int& y){int tmp = x;x = y;y = tmp;} 相信大家很快就能写出上面这段代码,但是如果要求这个交换函数支持字符型

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

vcpkg安装opencv中的特殊问题记录(无法找到opencv_corexd.dll)

我是按照网上的vcpkg安装opencv方法进行的(比如这篇:从0开始在visual studio上安装opencv(超详细,针对小白)),但是中间出现了一些别人没有遇到的问题,虽然原因没有找到,但是本人给出一些暂时的解决办法: 问题1: 我在安装库命令行使用的是 .\vcpkg.exe install opencv 我的电脑是x64,vcpkg在这条命令后默认下载的也是opencv2:x6