java-opencv分水岭算法分割硬币复现关键点

前言

java-opencv分水岭算法示例网上我是没找到java版本的教程。另外网上搜关于#基于分水岭算法的图像分割用法的教程基本都是官网基于那个有粘连的硬币分割图的。因为python语法简介灵活很多,移植到java的话稍不注意就会没那个效果的还不好排查,试了很多次才复现出示例的那个效果。教程很多我就不写明细了,说下例子了解的东西吧。理论我是不懂的想了解的网上找教程

关键点在这个步骤上

  1. 联通函数返回物体个数和对应的掩码(ret个数,markers的坐标值其实的是个数的序号0,1,2,3最大值是ret;0算是背景)
  2. markers[unknown==255] = 0;这步是个关键只是python语法太简洁,unknown来源就是背景(膨胀)-前景(腐蚀)的到的,把得到的联通域中之前的未知部分变为分水岭的未知区域。
ini 复制代码
     #Marker labelling`
     ret, markers = cv.connectedComponents(sure_fg)`
     # Add one to all labels so that sure background is not 0, but 1`
     markers = markers+1`
     # Now, mark the region of unknown with zero`
     markers[unknown==255] = 0`

流程了解的东西也不知道对不对

  1. 图像的二值化,就是将图像上的像素点的灰度值设置为0或255,也就是将整个图像呈现出明显的只有黑和白的视觉效果,二值化后的图片是单通道的。
scss 复制代码
//背景黑色前景白色THRESH_OTSU自动二值化
Imgproc.threshold(gray,gray,0,255,Imgproc.THRESH_BINARY_INV+Imgproc.THRESH_OTSU);

2.形态学处理,二值化后由于图像存在一些噪声(即白色的点)。所以需要形态学的开运算:腐蚀与膨胀平滑边界消除一些白点。

ini 复制代码
Mat element = Imgproc.getStructuringElement(Imgproc.MORPH_RECT, new Size(3,3));
Imgproc.morphologyEx(gray,gray,Imgproc.MORPH_OPEN,element);

3.得到背景、前景、未知区域。 膨胀是内容变大背景变小,物体边界增加到背景肯定是背景;腐蚀是内容变小,背景变大,这部分内容肯定是内容;然后膨胀后的图像-腐蚀后的=得到未知区域(如果物体不相连的话应该就是所谓的边界)。这个未知区域后面会用到

ini 复制代码
Mat sure_bg=new Mat();
Imgproc.dilate(gray, sure_bg, element, new Point(-1, -1), 3);
Mat sure_fg=new Mat();
Imgproc.erode(gray, sure_fg, element, new Point(-1, -1), 3);
Mat unknown=new Mat();
Core.subtract(sure_bg, sure_fg, unknown);
  1. 物体有相连接的部分需要用到距离变换,注意这里要使用第二次二值化处理。
ini 复制代码
//上次开运算得到的
Imgproc.distanceTransform(gray, dist_transform, Imgproc.CV_DIST_L2, 5);
//归一化
Core.normalize(dist_transform, dist_transform, 0, 1, Core.NORM_MINMAX);
Mat sure_fg2=new Mat();
Core.MinMaxLocResult mmr=Core.minMaxLoc(dist_transform);
//再次二值化
Imgproc.threshold(dist_transform,sure_fg2,0.5*mmr.maxVal,255,Imgproc.THRESH_BINARY);
  1. 使用分水岭算法

markers = markers+1因为分水岭中0是未知区域 makrers在java中里面坐标表示的图像的个数

markers[unknown==255] = 0把上面得到的未知区域,在得到的连通域中表示为未知的地方

unknown是第一次得到的背景-第二次二值化的大前景

css 复制代码
int count=Imgproc.connectedComponents(sure_fg2,mask);
for(int i=0;i<mask.rows();i++){
    for(int j=0;j<mask.cols();j++){
        mask.put(i,j,new int[]{(int)mask.get(i,j)[0]+1});
        if(unknown.get(i,j)[0]==255){
            mask.put(i,j,new int[]{0});
        };
    }
}
Imgproc.watershed(source,mask);
相关推荐
老华带你飞17 分钟前
建筑材料管理|基于springboot 建筑材料管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端·学习·spring
木心爱编程25 分钟前
【Qt 5.14.2 新手实战】QTC++入门筑基——按钮与标签联动:QPushButton + QLabel 实现图片切换器
java·c++·qt
椰羊~王小美1 小时前
setScale没传roudingmode,为什么会报错
java
WizLC1 小时前
【JAVA】JVM类加载器知识笔记
java·jvm·笔记
喝汽水的猫^1 小时前
Java实现Excel 导出(多 Sheet、复杂格式)
java·excel
技术净胜1 小时前
MATLAB进行图像分割从基础阈值到高级分割
opencv·计算机视觉·matlab
毕设源码-朱学姐1 小时前
【开题答辩全过程】以 基于JavaWeb的疾病查询系统的设计与实现为例,包含答辩的问题和答案
java·eclipse
雨中飘荡的记忆1 小时前
Java面向对象编程详解
java·开发语言
zhangyifang_0091 小时前
Spring中的BeanFactory类
java·后端·spring
大学生资源网2 小时前
java毕业设计之面向校园的助力跑腿系统设计与实现源码(源码+文档+数据库)
java·数据库·mysql·毕业设计·源码·springboot