質問をすることでしか得られない、回答やアドバイスがある。

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

ただいまの
回答率

89.13%

C++,OpenCVをもちいて画像から粒の直径を求めたい

受付中

回答 1

投稿

  • 評価
  • クリップ 1
  • VIEW 1,251

tune1362

score 10

前提・実現したいこと

C++でプログラムを書こうとしたところうまく行きません.
当方C++については全くの初心者です.参考書,サイトを見ながら組んではみたもののうまく動きません.何卒ご助力のほどよろしくお願いいたします.

現在の状態といたしましては,main関数に上記のreturn n, return sの値が戻らず,式及び数値が表示されない状態です.

該当のソースコード

#include<opencv2/core/core.hpp>
#include<opencv2/highgui/highgui.hpp>
#include<opencv2/imgproc/imgproc.hpp>
#include<iostream>
#include<conio.h>  
#include "Source.h"
#include <string>
using namespace cv;
using namespace std;


int kazu() {
    Mat img = imread("seed.jpg", IMREAD_UNCHANGED);
    Mat gray_img;
    cvtColor(img, gray_img, CV_BGR2GRAY);
    Mat bin_img;
    adaptiveThreshold(gray_img, bin_img, 255, CV_ADAPTIVE_THRESH_GAUSSIAN_C, CV_THRESH_BINARY, 99, 8);
    bin_img = ~bin_img; 
    Mat element = Mat::ones(5, 5, CV_8UC1); 

    erode(bin_img, bin_img, element, Point(-1, -1), 1); 
    erode(bin_img, bin_img, element, Point(-1, -1), 1); 
    erode(bin_img, bin_img, element, Point(-1, -1), 1); 
    erode(bin_img, bin_img, element, Point(-1, -1), 1); 

    vector<vector<Point> > contours;
    findContours(bin_img, contours, CV_RETR_EXTERNAL, CV_CHAIN_APPROX_NONE);
    for (int i = 0; i < contours.size(); i++) {
        int count = contours.at(i).size();
        double x = 0; double y = 0;
        for (int j = 0; j < count; j++) {
            x += contours.at(i).at(j).x;
            y += contours.at(i).at(j).y;
        }
        x /= count;
        y /= count;
        circle(img, Point(x, y), 5, Scalar(0, 0, 255), 2, 4);
    }
        int n;
        n = contours.size();

    imshow("IMAGE", img); 
    return n;
}


double keisan() {
    Mat img = imread("seed.jpg", IMREAD_UNCHANGED);
    Mat gray_img;
    cvtColor(img, gray_img, CV_BGR2GRAY);
    Mat bin_img;
    adaptiveThreshold(gray_img, bin_img, 255, CV_ADAPTIVE_THRESH_GAUSSIAN_C, CV_THRESH_BINARY, 99, 8);
    bin_img = ~bin_img; 
    Mat element = Mat::ones(5, 5, CV_8UC1); 

    vector<vector<Point> > contours;
    findContours(bin_img, contours, CV_RETR_EXTERNAL, CV_CHAIN_APPROX_NONE);

    double s = 0;
    int max_area_contour = -1;
    for (int j = 0; j < contours.size(); j++) {
        double area = contourArea(contours.at(j));
        s = s+area;
    }
    return s;
}
int main() {
    int a = kazu();
    double b = keisan();
    double d;
    d = pow(4 * b / 3.14 / a, 0.5);
    cout << "個数" << a << endl << "面積" << b << endl<<"直径"<<d<<endl;
    return 0;
}

補足情報()

やりたいこととしては粒体の画像を読み取り,それの直径を求めるというものです.現在,動画をソフトで画像にした後その画像をプログラムで編集している状態です.可能であれば動画のままプログラムで読み込んで,プログラムの中で動画を一定の間隔で画像を抜き取り,それらの編集したいのですがこのようなことは可能でしょうか?

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

0

私の環境(Windows10, Visual Studio 2015)ではおそらく正常動作しました。
検証をしづらいので、『粒状の画像』とその直径などのテストデータを示していただきたいです。

main関数に上記のreturn n, return sの値が戻らず、式及び数値が表示されない状態です。

この発言の意味がわからないです。
常に0がreturnされてしまうのか、その前にランタイムエラーで止まってしまうのか、
エラーメッセージがあるならそれを追記してください。


プログラミングの経験があるようですから、もう少し読みやすいコードを書いて下さい。
kazu keisanが何をしようとしているのか、関数名から予測不能です。
一文字の変数も、少なくともデバッグを依頼するときには使わないでください。
また、もう少しコードをまとめてください。同じ処理を複数書いています。

警告も多く出ています。必要な部分は明示的にstatic_castしてください。
ただし、C4819は無視して結構です。


可能であれば動画のままプログラムで読み込んで,プログラムの中で動画を一定の間隔で画像を抜き取り,それらの編集したいのですがこのようなことは可能でしょうか?

可能です。OpenCVでは、『画像処理のループ』として動画を処理することが出来ます。
cv::VideoCapture()の使い方を調べてみてください。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

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

  • ただいまの回答率 89.13%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

同じタグがついた質問を見る