Q&A
前提・実現したいこと
Java版OpenCV3.4.2で二値画像にラベリングをし、連結成分の面積が一番大きな部分以外を黒で塗りつぶしたいです(切り抜き)。
connectedComponentsWithStatsを用いて面積情報を取得するところまではできたのですが
指定したラベルの連結成分の色を変えることができずに困っています。
該当のソースコード
Java
1 public void getframe(Mat src_img, Mat dst_img){ 2 3 int rows = src_img.width(); 4 int cols = src_img.height(); 5 6//二値化 7 Mat gray_img = new Mat(cols,rows, CvType.CV_8UC1); 8 Imgproc.cvtColor(src_img,gray_img,Imgproc.COLOR_RGB2GRAY); 9 int blockSize=Math.min(cols,rows)/40*2+1; 10 Imgproc.adaptiveThreshold(gray_img,dst_img,255,Imgproc.ADAPTIVE_THRESH_GAUSSIAN_C,Imgproc.THRESH_BINARY_INV,blockSize,3); 11 12 13//ラベリング実行 14 Mat labelImage = new Mat(cols,rows,CvType.CV_32S); 15 Mat stats = new Mat(); 16 Mat centroids = new Mat(); 17 18 int nLabels = Imgproc.connectedComponentsWithStats(dst_img,labelImage,stats,centroids,8,CvType.CV_32S); 19 20 21 22//面積情報を取得 23int[] result = new int[nLabels]; 24 int max = 0; 25 int count = 0; 26 27 for(int i=1;i<nLabels;i++){ 28 stats.get(i,Imgproc.CC_STAT_AREA,result); 29 if(max < result[i]){ 30 max = result[i]; 31 count = i; 32 } 33 } 34 35//連結成分の中で一番面積が大きいもの以外を黒塗り 36 for(int i=1;i<nLabels;i++){ 37 if(i != count){ 38 //黒で塗りつぶす 39 } 40 } 41 42}
試したこと
c++でのconnectedComponentsにおける指定したラベルの塗りつぶしを参考にしています。
しかし、Java版OpenCVに画素値を変更するVec3bが無いようで、
代わりに指定した座標の画素値を取り出す「mat.get()」と
指定した座標の画素値を変更する「mat.put()」を利用しようと試みたのですがどうにもうまくいきません。
そもそも参考にしているサイトにおいて
std::vector<cv::Vec3b> colors(nLabels);
というVec3b型の可変長配列を作っているところまでは理解できるのですが
colors[0] = cv::Vec3b(0, 0, 0);
上記のようにcolols[ラベルの番号]だけで指定したラベルにアクセスできているというのが理解できません。
colorsはあくまでnLabels個のVec3b型の配列として定義されただけなのに
なぜラベル番号を引数に入れるだけで該当ラベルの画素値を操作できるようになるのでしょうか。
伝わりづらいしし質問で申し訳ありませんがご教授いただけると幸いです。
c++
1std::vector<cv::Vec3b> colors(nLabels); 2 colors[0] = cv::Vec3b(0, 0, 0); 3 for(int label = 1; label < nLabels; ++label) 4 { 5 colors[label] = cv::Vec3b((rand()&255), (rand()&255), (rand()&255)); 6 } 7 8 // ラベリング結果の描画 9 cv::Mat dst(src.size(), CV_8UC3); 10 for(int y = 0; y < dst.rows; ++y) 11 { 12 for(int x = 0; x < dst.cols; ++x) 13 { 14 int label = labelImage.at<int>(y, x); 15 cv::Vec3b &pixel = dst.at<cv::Vec3b>(y, x); 16 pixel = colors[label]; 17 } 18 19
回答2件
あなたの回答
tips
プレビュー
下記のような回答は推奨されていません。
このような回答には修正を依頼しましょう。