OpenCV-Python bindings是如何生成的(2)

OpenCV-Python bindings生成流程

通过上篇文章和opencv python模块中的CMakeLists.txt文件,可以了解到opencv-python bindings生成的整个流程:

  1. 生成headers.txt文件
    将每个模块的头文件添加到list中,通过一些关键词过滤掉一些不需要扩展的头文件,file(WRITE "${CMAKE_CURRENT_BINARY_DIR}/headers.txt" "${opencv_hdrs}")将过滤后的list写入文件;
    headers.txt保存需要转换的头文件路径,英文;隔开,注意文件末尾没有换行符

  2. 生成cv2.cpp中需要的头文件

    	set(cv2_generated_hdrs"${CMAKE_CURRENT_BINARY_DIR}/pyopencv_generated_include.h""${CMAKE_CURRENT_BINARY_DIR}/pyopencv_generated_funcs.h""${CMAKE_CURRENT_BINARY_DIR}/pyopencv_generated_types.h""${CMAKE_CURRENT_BINARY_DIR}/pyopencv_generated_type_reg.h""${CMAKE_CURRENT_BINARY_DIR}/pyopencv_generated_ns_reg.h")file(WRITE "${CMAKE_CURRENT_BINARY_DIR}/headers.txt" "${opencv_hdrs}")
    add_custom_command(OUTPUT ${cv2_generated_hdrs}COMMAND ${PYTHON_EXECUTABLE} "${PYTHON_SOURCE_DIR}/src2/gen2.py" ${CMAKE_CURRENT_BINARY_DIR} "${CMAKE_CURRENT_BINARY_DIR}/headers.txt"DEPENDS ${PYTHON_SOURCE_DIR}/src2/gen2.pyDEPENDS ${PYTHON_SOURCE_DIR}/src2/hdr_parser.pyDEPENDS ${CMAKE_CURRENT_BINARY_DIR}/headers.txtDEPENDS ${opencv_hdrs})
    

    通过调用gen2.py来实现的,第一个参数是生成头文件后保存的路径,第二个参数是headers.txt文件路径

  3. 将cv2.cpp编译成动态库

OpenCV-Python bindings实践操作

纸上得来终觉浅,绝知此事要躬行。整个事情的来龙去脉都已经讲清楚,单到底怎么应用到自己的项目中呢。比如将自己图像检测功能实现完成,老大要求将它写成Web服务,毕竟B/S模式非常流程。C++写Web?开国际玩笑吧,现在python非常流行的天下。那么接下来看看怎么讲c++功能函数类转换成python的扩展库吧。

网上查查资料,还真有相关的blog:

  • How to convert your OpenCV C++ code into a Python module
  • github code https://github.com/spmallick/learnopencv/tree/master/pymodule

博客写的很好,但是提供的源码千般尝试始终报错,于是乎决定自己从CMakeLists.txt看起,从gen2.py脚本看起,慢慢查实。

mkdir ovex
cd ovex
mkdir src
cp ../opencv-3.1.0/modules/python/src2/{pycompat.hpp,cv2.cpp,gen2.py,hdr_parser.py} ./
cp ../learnopencv-master/pymodule/src/* src/
cp ../learnopencv-master/pymodule/headers.txt ./

准备使用opencv-3.1.0源码中的pycompat.hpp,cv2.cpp,gen2.py,hdr_parser.py文件,使用上面博客中的opencv c++源文件,慢慢来

gen2.py ./ headers.txt命令,第一步就报错了IOError: [Errno 2] No such file or directory: 'src/bvmodule.hpp\n'
看到文件名后面居然有个换行符,哎,拿到windows下删掉换行符。

去掉headers.txt中的换行符在去执行上面的命令,报出另外一个错误

Traceback (most recent call last):File "./gen2.py", line 943, in <module>generator.gen(srcfiles, dstdir)File "./gen2.py", line 855, in genself.code_include.write( '#include "{0}"\n'.format(hdr[hdr.rindex('opencv2/'):]) )
ValueError: substring not found

hdr.rindex('opencv2/')取头文件中opencv/的位置,hdr[hdr.rindex('opencv2/'):])取头文件中opencv/之后的所有字符串。看样子脚本是opencv定制,应用需要改动脚本某些地方,通过不断尝试,几经修改,终于大功告成。gen2.py ./ headers.txt命令在当前目录下生成pyopencv_generated_funcs.h pyopencv_generated_include.h pyopencv_generated_ns_reg.h pyopencv_generated_type_reg.h pyopencv_generated_types.h文件。gen2.py修改如下
gen2.py修改

调整博客提供的编译命令,编译动态库

g++ -shared -rdynamic -g -O3 -Wall -fPIC \
-I . -I../ -I/usr/local/python2.7.14/lib/python2.7/site-packages/numpy/core/include \
cv2.cpp src/bvmodule.cpp \
-DNDEBUG \
`PKG_CONFIG_PATH=/usr/local/opencv_with_contrib3.1.0/lib/pkgconfig pkg-config --cflags --libs opencv` \
`/usr/local/python2.7.14/bin/python2.7-config --includes --ldflags` \
-L`/usr/local/python2.7.14/bin/python2.7-config --exec-prefix`/lib \
-o bv.so

出错

cv2.cpp:124: 错误:‘Stitcher’未声明
cv2.cpp:124: 错误:expected initializer before ‘Status’
cv2.cpp:474: 错误:ISO C++ 不允许声明无类型的‘Status’
cv2.cpp:474: 错误:expected ‘,’ or ‘...’ before ‘&’ token
cv2.cpp:474: 错误:‘PyObject* pyopencv_from(int)’的模板标识符‘pyopencv_from<>’不匹配任何模板声明
cv2.cpp:1257: 警告:‘int convert_to_char(PyObject*, char*, const char*)’定义后未使用

类型未声明,用不到的话就删掉吧
在这里插入图片描述
再次执行编译命令,期待的so动态库在当前目录生成了。那就试他一试吧,

ovex]$ python2.7
Python 2.7.14 (default, Jul 25 2018, 13:52:02) 
[GCC 4.4.7 20120313 (Red Hat 4.4.7-23)] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> import bv
Traceback (most recent call last):File "<stdin>", line 1, in <module>
ImportError: dynamic module does not define init function (initbv)

这个错误的出现不得不促使我去了解一下C/C++为python做扩展的整个流程,Extending and Embedding the Python Interpreter,注意python2与python3是有差异的。

很容易发现,模块需要一个对应初始化函数来初始化模块,比如PyInit_bv,那么在opencv中这个初始化函数在哪定义的呢?直接点在当前目录下执行grep -n --color=auto "cv2" -R *,在cv2.cpp中找到我们想要的函数了,接下来就是修改了
cv2.cpp修改
模块的名称是通过MODULESTR来定义的,python2以及python3中模块初始化函数的声明与定义。
再次编译成功通过,测试OK

$ python2.7
Python 2.7.14 (default, Jul 25 2018, 13:52:02) 
[GCC 4.4.7 20120313 (Red Hat 4.4.7-23)] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> import bv
>>> import numpy as np
>>> dir(bv)
['CV_16S', 'CV_16SC1', 'CV_16SC2', 'CV_16SC3', 'CV_16SC4', 'CV_16U', 'CV_16UC1', 'CV_16UC2', 'CV_16UC3', 'CV_16UC4', 'CV_32F', 'CV_32FC1', 'CV_32FC2', 'CV_32FC3', 'CV_32FC4', 'CV_32S', 'CV_32SC1', 'CV_32SC2', 'CV_32SC3', 'CV_32SC4', 'CV_64F', 'CV_64FC1', 'CV_64FC2', 'CV_64FC3', 'CV_64FC4', 'CV_8S', 'CV_8SC1', 'CV_8SC2', 'CV_8SC3', 'CV_8SC4', 'CV_8U', 'CV_8UC1', 'CV_8UC2', 'CV_8UC3', 'CV_8UC4', 'Filters', '__doc__', '__file__', '__name__', '__package__', '__version__', 'createTrackbar', 'error', 'fillHoles', 'setMouseCallback']
>>> import cv2
>>> img = cv2.imread("holes.jpg")
>>> bv.fillHoles(img)
>>> cv2.imwrite("filledHoles.jpg", img)
True
>>>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/508640.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【TensorFlow】学习资源汇总以及知识总结

官方资源 官方网站 https://tensorflow.org 非翻墙神器不能访问也&#xff08;关键是我用了翻墙神器也没能访问&#xff09;伪官方网站 https://tensorflow.google.cn/ 墙内的人可以查阅的资料github https://github.com/tensorflow/tensorflow官方提供的models以及tutorial h…

机器学习资源锦集

http://www.cnblogs.com/pinard 十年码农&#xff0c;对数学统计学&#xff0c;数据挖掘&#xff0c;机器学习&#xff0c;大数据平台&#xff0c;大数据平台应用开发&#xff0c;大数据可视化感兴趣。github 深度学习 【深度学习】批归一化&#xff08;Batch Normalization&…

获取训练数据的方式

下载搜狗词库 https://pinyin.sogou.com/dict/ 在官网搜索相关的词库下载&#xff0c;比如地名等&#xff0c;然后使用脚本将此条转换成txt保存&#xff0c; 来源 # -*- coding: utf-8 -*- import os import sys import struct # 主要两部分 # 1.全局拼音表&#xff0c;貌似…

浅谈python MRO与Mixin模式

MRO(Method Resolution Order) In object-oriented programming languages with multiple inheritance, the diamond problem (sometimes referred to as the “deadly diamond of death”) is an ambiguity that arises when two classes B and C inherit from A, and class D…

CentOS7开发环境搭建(2)

关闭SELinux # 查看 $ getenforce Disabled $ sestatus SELinux status: enabled SELinuxfs mount: /sys/fs/selinux SELinux root directory: /etc/selinux Loaded policy name: targeted Current mode: …

IntelliJ IDEA开发环境应用

安装 下载windows压缩包获取帮助: idea.medeming.com/jihuoma 常用设置 全局设置&#xff0c;对新建的工程生效 【File】【Other Settings】【Setings for New Projects…】 比如配置maven的路径以及配置文件的路径&#xff0c;基本设置一次即可&#xff0c;不需要每次新建工…

tcp状态机-三次握手-四次挥手以及常见面试题

TCP状态机介绍 在网络协议栈中&#xff0c;目前只有TCP提供了一种面向连接的可靠性数据传输。而可靠性&#xff0c;无非就是保证&#xff0c;我发给你的&#xff0c;你一定要收到。确保中间的通信过程中&#xff0c;不会丢失数据和乱序。在TCP保证可靠性数据传输的实现来看&am…

Visual studio Code的C/C++开发环境搭建

文章目录VS CodeC/C环境配置环境准备使用实例基于 VSCode 的远程开发平台环境准备参考VS Code Visual Studio Code&#xff08;简称VS Code&#xff09;是一个由微软开发&#xff0c;同时支持Windows 、 Linux和macOS等操作系统且开放源代码的代码编辑器&#xff0c;它支持测试…

Linux网络编程--文件描述符

文件描述符 在Unix和Unix-like操作系统中&#xff0c;文件描述符(file descriptor, FD)是一个文件或者像pipe或者network socket等之类的输入/输出源的唯一标识。 文件描述符通常是一个非负整数&#xff0c;负数通常代表无值或者错误。 文件描述符是POSIX API的一部分。每个除…

深信服 linux软件开发面试题整理

1、结构体可以进行比较 int memcmp ( const void * ptr1, const void * ptr2, size_t num ); Compare two blocks of memory Compares the first num bytes of the block of memory pointed by ptr1 to the first num bytes pointed by ptr2, returning zero if they all match…

大端小端模式判断以及数据转换

简介 在计算机系统中&#xff0c;我们是以字节为单位的&#xff0c;每个地址单元都对应着一个字节&#xff0c;一个字节为 8bit。但是在C语言中除了8bit的char之外&#xff0c;还有16bit的short型&#xff0c;32bit的long型&#xff08;要看具体的编译器&#xff09;&#xff…

MSYS2下搭建Qt开发环境

最近随意浏览了一下俺们大省会城市的招聘信息&#xff0c;发现C招聘中涉及Qt经验的要求有不少&#xff0c;为了牛奶和面包&#xff0c;决心深入一下Qt开发。本篇文章由此而出。 Qt 关于Qt的人生经历在这不在累赘&#xff0c;资料随处可得&#xff0c;这里只记录干货。 环境搭…

CentOS7开发环境搭建(1)

文章目录BIOS开启VT支持U盘安装系统(2019-03-11)CentOS DNS配置CentOS网络配置配置静态IP克隆虚拟机网卡名称变更 CentOS6.5时间配置安装VMWare-tools用户管理 (2019-03-15 7.6.1810)给一般账号 root 权限Samba服务配置安装必备软件获取本机公网ipyum源和第三方库源管理配置本地…

ACM 欧拉公式

给出一个数X&#xff0c;求小于X的与X互质的数的个数&#xff0c;使用欧拉公式。 如果x1*x2*...*xnX,则个数nX*(1-1/x1)*(1-/x2)*... 使用这个的题目&#xff0c;超典型 相遇周期(HDOJ)

HDU 1495 非常可乐(BFS)

思路 最难在于想到这道题是BFS&#xff0c;想到之后只有六种情况就很好理解了。 代码 #include<stdio.h> #include<string.h> #include<math.h> #include<queue> using namespace std; int a,b,s; struct shui {int count;int ha,hb,hs; }t,t1; int m…

NBU计算机专业期末考试记录

考试科目&#xff1a;操作系统 软件工程 数据库 计算机网络 JAVA高级应用 汇编 计算机算法设计 操作系统&#xff1a;题目比较简单&#xff0c;这学期的大题有写读写互斥的代码、求平均磁道数、银行家算法、页面调度算法的缺页次数计算。期中考试有参考价值&#xff0c;要看懂…

蚁群算法的若干记录

1、蚁群算法的特点&#xff1a; ① 结合了分布式算法、正反馈机制、贪婪式搜索的算法&#xff1a;正反馈可以快速发现较优解、分布式算法避免早熟收敛、贪婪式搜索有助于早期找出可解决方案&#xff1b; ② 蚁群算法具有很强的并行性&#xff1b; ③ 个体之间通过信息素合作…

蚁群算法之二

1、蚂蚁系统模型的建立 给定G(V,A)&#xff0c;其中V为定点集&#xff0c;A为各顶点互相连接组成的边集,已知各顶点之间的连接距离&#xff0c;要求确定一条长度最短的回路&#xff0c;仅遍历一次所有顶点的回路。引入记号&#xff1a; m&#xff1a;蚁群中蚂蚁的数量&#x…

ns2相关学习——tcl脚本编写(1)

新建一个仿真实例&#xff1a; set ns [new Simulator]为了让nam文件和trace文件有地方可以依托&#xff0c;我们要打开.nam文件进行写入&#xff0c;并且使用句柄nf set nf [open out.nam w] $ns namtrace-all $nf设置拓扑图 1、设置节点的脚本语言&#xff1a;建了两个节点&…

ns2相关学习——TCL脚本编写(2)

下面来学习更加复杂一点的TCL脚本的编写 简述&#xff1a;建立有4个节点的拓扑&#xff0c;其中一个节点作为路由器&#xff0c;用来将两个节点发出的数据包转发到第四个节点上面。 在这里将解释将两个节点的数据流区分开来的方法&#xff0c;展示如何去检测一个队列是否是满…