🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Visual Studio

Microsoft Visual StudioはMicrosoftによる統合開発環境(IDE)です。多種多様なプログラミング言語に対応しています。

OpenCV

OpenCV(オープンソースコンピュータービジョン)は、1999年にインテルが開発・公開したオープンソースのコンピュータビジョン向けのクロスプラットフォームライブラリです。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

2回答

3751閲覧

C++でMatの型が合わない問題

airu

総合スコア10

Visual Studio

Microsoft Visual StudioはMicrosoftによる統合開発環境(IDE)です。多種多様なプログラミング言語に対応しています。

OpenCV

OpenCV(オープンソースコンピュータービジョン)は、1999年にインテルが開発・公開したオープンソースのコンピュータビジョン向けのクロスプラットフォームライブラリです。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2019/10/23 11:08

前提・実現したいこと

画像を読み込んで離散コサイン変換した後、逆離散コサイン変換して出力するプログラムを作成しています。
警告が沢山出て、プログラムが止まってしまいます。
多分、Matの型が合っていないと思うのですが、理解が足りずどうすればいいのか分かりません。
初歩的な質問で申し訳ありませんが、よろしくお願いします。
以下のコードを参考にしました。
https://github.com/yoyoyo-yo/Gasyori100knock/blob/master/Question_31_40/answers_cpp/answer_36.cpp

発生している問題・エラーメッセージ

重大度レベル コード 説明 プロジェクト ファイル 行 抑制状態 警告 C26451 Arithmetic overflow: Using operator '*' on a 4 byte value and then casting the result to a 8 byte value. Cast the value to the wider type before calling operator '*' to avoid overflow (io.2). qim_1 c:\users\ak\downloads\opencv\build\install\include\opencv2\core\cuda.inl.hpp 301 警告 C26451 Arithmetic overflow: Using operator '+' on a 4 byte value and then casting the result to a 8 byte value. Cast the value to the wider type before calling operator '+' to avoid overflow (io.2). qim_1 c:\users\ak\downloads\opencv\build\install\include\opencv2\core\cuda.inl.hpp 301 警告 C26451 Arithmetic overflow: Using operator '*' on a 4 byte value and then casting the result to a 8 byte value. Cast the value to the wider type before calling operator '*' to avoid overflow (io.2). qim_1 c:\users\ak\downloads\opencv\build\install\include\opencv2\core\cuda.inl.hpp 482 警告 C26451 Arithmetic overflow: Using operator '+' on a 4 byte value and then casting the result to a 8 byte value. Cast the value to the wider type before calling operator '+' to avoid overflow (io.2). qim_1 c:\users\ak\downloads\opencv\build\install\include\opencv2\core\cuda.inl.hpp 482 …

該当のソースコード

#include <opencv2/core.hpp> #include <opencv2\opencv.hpp> #include <opencv2/highgui.hpp> #define _USE_MATH_DEFINES #include <iostream> #include <math.h> #include <complex> using namespace cv; using namespace std; const int height = 512, width = 512; // DCT hyper-parameter int T = 8; int K = 8; // Discrete Cosine transformation void dct(Mat img, Mat dct_coef) { double I; double F; double Cu, Cv; for (int ys = 0; ys < height; ys += T) { for (int xs = 0; xs < width; xs += T) { for (int v = 0; v < T; v++) { for (int u = 0; u < T; u++) { F = 0; if (u == 0) { Cu = 1. / sqrt(2); } else { Cu = 1; } if (v == 0) { Cv = 1. / sqrt(2); } else { Cv = 1; } for (int y = 0; y < T; y++) { for (int x = 0; x < T; x++) { I = (double)img.at<uchar>(ys + y, xs + x); F += 2. / T * Cu * Cv * I * cos((2. * x + 1) * u * M_PI / 2. / T) * cos((2. * y + 1) * v * M_PI / 2. / T); } } dct_coef.at<uchar>(ys + v, xs + u) = F; } } } } } // Inverse Discrete Cosine transformation void idct(Mat out, Mat dct_coef) { double f; double Cu, Cv; for (int ys = 0; ys < height; ys += T) { for (int xs = 0; xs < width; xs += T) { for (int y = 0; y < T; y++) { for (int x = 0; x < T; x++) { f = 0; for (int v = 0; v < K; v++) { for (int u = 0; u < K; u++) { if (u == 0) { Cu = 1. / sqrt(2); } else { Cu = 1; } if (v == 0) { Cv = 1. / sqrt(2); } else { Cv = 1; } f += 2. / T * Cu * Cv * dct_coef.at<uchar>(ys + v, xs + u) * cos((2. * x + 1) * u * M_PI / 2. / T) * cos((2. * y + 1) * v * M_PI / 2. / T); } } f = fmin(fmax(f, 0), 255); out.at<uchar>(ys + y, xs + x) = (uchar)f; } } } } } // Main int main(int argc, const char* argv[]) { // read original image Mat img = cv::imread("lenna.jpg", cv::IMREAD_COLOR); // グレースケール変換 Mat gray1; cvtColor(img, gray1, COLOR_RGB2GRAY); //std::cout << "image.channels(): " << gray1.channels() << std::endl; // output image Mat out = cv::Mat::zeros(img.rows, img.cols, CV_8U); imshow("gray", out); // DCT Mat dct_coef; dct(gray1, dct_coef); // IDCT idct(out, dct_coef); imwrite("out.jpg", out); waitKey(); }

試したこと

.at<>の型を色々変えてみましたが、うまくいかなかったので質問しました。

補足情報(FW/ツールのバージョンなど)

Visual Studio2019
OpenCV-4.1.0

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

提示されてる警告は、doubleとintを混在させて計算式にしているためでなんの問題もありません(計算結果が違う可能性はあるけど)

プログラムが止まるのはこれが原因ではないですね。

投稿2019/10/23 11:45

編集2019/10/23 11:46
y_waiwai

総合スコア88040

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

airu

2019/10/24 02:24

ずっとそこが問題なのかと思ってました…。 ありがとうございます。 下の方が指摘してくださっていたように、dct_coefの定義が問題だったみたいです。 直したら無事にちゃんと動きました! ありがとうございました。
guest

0

ベストアンサー

main()で空っぽのdct_coefをdct()の引数に渡す.

Mat dct_coef; dct(gray1, dct_coef);


dct()内では,

dct_coef.at<uchar>(ys + v, xs + u) = F;

として唐突な画素アクセスしている.

…という点がとりあえず問題ありに見える.

投稿2019/10/23 11:44

fana

総合スコア11990

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

airu

2019/10/24 02:29 編集

ありがとうございます。まさにその部分でした。 Mat dct_coef; を Mat dct_coef(img.size(), CV_64F, Scalar::all(0)); に直して、該当の型も直したら無事にちゃんと動きました! ありがとうございました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問