網站首頁 編程語言 正文
在opencv中,reshape函數比較有意思,它既可以改變矩陣的通道數,又可以對矩陣元素進行序列化,非常有用的一個函數。
函數原型:
C++: Mat Mat::reshape(int cn, int rows=0) const
參數比較少,但設置的時候卻要千萬小心。
cn: 表示通道數(channels), 如果設為0,則表示保持通道數不變,否則則變為設置的通道數。
rows: 表示矩陣行數。 如果設為0,則表示保持原有的行數不變,否則則變為設置的行數。
首先設置一個初始矩陣:一個20行30列1通道的一個矩陣
int main() { Mat data = Mat(20, 30, CV_32F); //設置一個20行30列1通道的一個矩陣 cout << "行數: " << data.rows << endl; cout << "列數: " << data.cols << endl; cout << "通道: " << data.channels() << endl; system("pause"); return 1; }
輸出:
第一次變化:通道數不變,將矩陣序列化1行N列的行向量。
int main() { Mat data = Mat(20, 30, CV_32F); //設置一個20行30列1通道的一個矩陣 cout << "行數: " << data.rows << endl; cout << "列數: " << data.cols << endl; cout << "通道: " << data.channels() << endl; cout << endl; Mat dst = data.reshape(0, 1); cout << "行數: " << dst.rows << endl; cout << "列數: " << dst.cols << endl; cout << "通道: " << dst.channels() << endl; system("pause"); return 1; }
第二次變化:通道數不變,將矩陣序列化N行1列的列向量。
int main() { Mat data = Mat(20, 30, CV_32F); //設置一個20行30列1通道的一個矩陣 cout << "行數: " << data.rows << endl; cout << "列數: " << data.cols << endl; cout << "通道: " << data.channels() << endl; cout << endl; Mat dst = data.reshape(0, data.rows*data.cols); cout << "行數: " << dst.rows << endl; cout << "列數: " << dst.cols << endl; cout << "通道: " << dst.channels() << endl; system("pause"); return 1; }
可見,序列成列向量比行向量要麻煩一些,還得去計算出需要多少行。但我們可以先序列成行向量,再轉置
Mat dst = data.reshape(0, 1); //序列成行向量 Mat dst = data.reshape(0, 1).t(); //序列成列向量
第三次變化:通道數由1變為2,行數不變。
int main() { Mat data = Mat(20, 30, CV_32F); //設置一個20行30列1通道的一個矩陣 cout << "行數: " << data.rows << endl; cout << "列數: " << data.cols << endl; cout << "通道: " << data.channels() << endl; cout << endl; Mat dst = data.reshape(2, 0); cout << "行數: " << dst.rows << endl; cout << "列數: " << dst.cols << endl; cout << "通道: " << dst.channels() << endl; system("pause"); return 1; }
從結果可以看出,列數被分出一半,放在第二個通道里去了。
同理,如果通道數由1變為3,行數不變。則每通道的列數變為原來的三分之一。
需要注意的是,如果行保持不變,改變的通道數一定要能被列數整除,否則會出錯。
第四次變化:通道數由1變為2,行數變為原來的五分之一。
int main() { Mat data = Mat(20, 30, CV_32F); //設置一個20行30列1通道的一個矩陣 cout << "行數: " << data.rows << endl; cout << "列數: " << data.cols << endl; cout << "通道: " << data.channels() << endl; cout << endl; Mat dst = data.reshape(2, data.rows/5); cout << "行數: " << dst.rows << endl; cout << "列數: " << dst.cols << endl; cout << "通道: " << dst.channels() << endl; system("pause"); return 1; }
可見,不管怎么變,都遵循這樣一個等式:
變化之前的? rows*cols*channels = 變化之后的 rows*cols*channels
我們只能改變通道數和行數,列數不能改變,它是自動變化的。
但是要注意的是,在變化的時候,要考慮到是否整除的情況。如果改變的數值出現不能整除,就會報錯。
最后,我們再驗證一下:opencv在序列化的時候是行序列化還是列序列化呢?
我們知道,在matlab里面,是列序列化, 即取值為從上到下,從左到右,opencv又是怎么樣的呢
int main() { Mat data = (Mat_<int>(2, 3) << 1, 2, 3, 10, 20, 30); //2行3列的矩陣 cout << data << endl; Mat dst1 = data.reshape(0, 6); //通道不變,序列成列向量 cout <<endl<< dst1 << endl; Mat dst2 = data.reshape(0, 1); //通道不變,序列成行向量 cout << endl << dst2 << endl; system("pause"); return 1; }
從結果看出,不管是變化成行向量還是列向量,opencv都是行序列化,即從左到右,從上到下,與matlab是不一樣的。
簡單的一個函數,功能卻很強大!你會用了嗎?
原文鏈接:https://www.cnblogs.com/denny402/p/5035535.html
相關推薦
- 2022-04-17 uniapp 實現無感刷新token, 適應大多數項目
- 2022-06-12 Python取讀csv文件做dbscan分析_python
- 2023-02-05 如何用C#獲取計算機詳細的軟件和硬件信息_C#教程
- 2023-03-23 Nginx實現http自動跳轉到https_nginx
- 2022-04-24 Redis?整數集合的具體使用(intset)_Redis
- 2022-03-15 使用axios出現跨域問題Response to preflight request doesn‘t
- 2022-08-16 python切片操作方法的實例總結_python
- 2022-09-25 Shiro和SpringSecurity
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支