java使用opencv比较图片并生成so文件部署到linux生产环境

使用opencv版本2.4.1.1,java项目是springboot

https://opencv.org/releases/page/4/

 一、windows中使用

下载openCV-2.4.1.1windows版本,得到一个.exe文件,电脑中打开运行会生成一个opencv文件夹目录,打开目录中找到java使用的jar文件和.dll库文件(x86或x64文件中)

接下来项目中引入jar文件,可以创建一个文件夹存放

gradle中引用

compile fileTree(dir:'openCv',include:['*.jar'])

项目中初始化库文件

import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * @author zhy
 * @date 2019/8/1 15:27
 */
@Configuration
@ConfigurationProperties(prefix = "opencv.lib")
public class OpenCvLibConfig {

    @Value("${opencv.lib.path}")
    private String path;

    @Bean
    public void init() {
        System.load(path);
    }

}

application.properties

opencv.lib.path=D:/test/opencv_java2411.dll

创建工具类

import lombok.extern.slf4j.Slf4j;
import org.opencv.calib3d.Calib3d;
import org.opencv.core.*;
import org.opencv.features2d.*;
import org.opencv.highgui.Highgui;

import java.util.LinkedList;
import java.util.List;

/**
 * 图像识别
 *
 * @author zhy
 * @date 2019/7/31 14:56
 */
@Slf4j
public class ImageRecognition {

    private ImageRecognition() {
        throw new IllegalStateException("Utility class");
    }

    /**
     * 匹配图像
     *
     * @param templateFilePath 模板文件路径
     * @param originalFilePath 原始文件路径
     * @return 匹配的像素点总数
     */
    public static int matchImage(String templateFilePath, String originalFilePath, String processDiagram) {
        float nndrRatio = 0.7f;
        //读取图片文件
        Mat templateImage = Highgui.imread(templateFilePath, Highgui.CV_LOAD_IMAGE_COLOR);
        Mat originalImage = Highgui.imread(originalFilePath, Highgui.CV_LOAD_IMAGE_COLOR);
        if (templateImage.cols() == 0 || originalImage.cols() == 0 || templateImage.rows() == 0 || originalImage.rows() == 0) {
            log.error("图片文件路径异常,获取的图片大小为0,无法读取");
            return 0;
        }
        MatOfKeyPoint templateKeyPoints = new MatOfKeyPoint();
        //指定特征点算法SURF
        FeatureDetector featureDetector = FeatureDetector.create(FeatureDetector.SURF);
        //获取模板图的特征点
        featureDetector.detect(templateImage, templateKeyPoints);
        //提取模板图的特征点
        MatOfKeyPoint templateDescriptors = new MatOfKeyPoint();
        DescriptorExtractor descriptorExtractor = DescriptorExtractor.create(DescriptorExtractor.SURF);
        log.info("提取模板图的特征点");
        descriptorExtractor.compute(templateImage, templateKeyPoints, templateDescriptors);
        Mat outputImage = new Mat(templateImage.rows(), templateImage.cols(), Highgui.CV_LOAD_IMAGE_COLOR);
        log.info("在图片上显示提取的特征点");
        Features2d.drawKeypoints(templateImage, templateKeyPoints, outputImage, new Scalar(255, 0, 0), 0);
        MatOfKeyPoint originalKeyPoints = new MatOfKeyPoint();
        MatOfKeyPoint originalDescriptors = new MatOfKeyPoint();
        featureDetector.detect(originalImage, originalKeyPoints);
        log.info("提取原图的特征点");
        descriptorExtractor.compute(originalImage, originalKeyPoints, originalDescriptors);

        List<MatOfDMatch> matches = new LinkedList<>();
        DescriptorMatcher descriptorMatcher = DescriptorMatcher.create(DescriptorMatcher.FLANNBASED);
        log.info("寻找最佳匹配");
        descriptorMatcher.knnMatch(templateDescriptors, originalDescriptors, matches, 2);

        log.info("计算匹配结果");
        LinkedList<DMatch> goodMatchesList = new LinkedList<>();
        matches.forEach(match -> {
            DMatch[] dmatcharray = match.toArray();
            DMatch m1 = dmatcharray[0];
            DMatch m2 = dmatcharray[1];

            if (m1.distance <= m2.distance * nndrRatio) {
                goodMatchesList.addLast(m1);
            }
        });
        if (processDiagram != null && goodMatchesList.size() > 4) {
            matchPointOutput(templateKeyPoints, originalKeyPoints
                    , goodMatchesList, templateImage, originalImage, processDiagram);
        }
        return goodMatchesList.size();
    }

    /**
     * 输出特征点匹配过程
     *
     * @param templateKeyPoints MatOfKeyPoint
     * @param originalKeyPoints MatOfKeyPoint
     * @param goodMatchesList   LinkedList<DMatch>
     * @param templateImage     Mat
     * @param originalImage     Mat
     */
    private static void matchPointOutput(MatOfKeyPoint templateKeyPoints, MatOfKeyPoint originalKeyPoints
            , List<DMatch> goodMatchesList, Mat templateImage, Mat originalImage, String processDiagram) {
        List<KeyPoint> templateKeyPointList = templateKeyPoints.toList();
        List<KeyPoint> originalKeyPointList = originalKeyPoints.toList();
        LinkedList<Point> objectPoints = new LinkedList<>();
        LinkedList<Point> scenePoints = new LinkedList<>();
        goodMatchesList.forEach(goodMatch -> {
            objectPoints.addLast(templateKeyPointList.get(goodMatch.queryIdx).pt);
            scenePoints.addLast(originalKeyPointList.get(goodMatch.trainIdx).pt);
        });
        MatOfPoint2f objMatOfPoint2f = new MatOfPoint2f();
        objMatOfPoint2f.fromList(objectPoints);
        MatOfPoint2f scnMatOfPoint2f = new MatOfPoint2f();
        scnMatOfPoint2f.fromList(scenePoints);
        Mat homography = Calib3d.findHomography(objMatOfPoint2f, scnMatOfPoint2f, Calib3d.RANSAC, 3);
        Mat templateCorners = new Mat(4, 1, CvType.CV_32FC2);
        Mat templateTransformResult = new Mat(4, 1, CvType.CV_32FC2);
        templateCorners.put(0, 0, 0, 0);
        templateCorners.put(1, 0, templateImage.cols(), 0);
        templateCorners.put(2, 0, templateImage.cols(), templateImage.rows());
        templateCorners.put(3, 0, 0, templateImage.rows());
        //使用 perspectiveTransform 将模板图进行透视变以矫正图象得到标准图片
        Core.perspectiveTransform(templateCorners, templateTransformResult, homography);

        MatOfDMatch goodMatches = new MatOfDMatch();
        goodMatches.fromList(goodMatchesList);
        Mat matchOutput = new Mat(originalImage.rows() * 2, originalImage.cols() * 2, Highgui.CV_LOAD_IMAGE_COLOR);
        Features2d.drawMatches(templateImage, templateKeyPoints, originalImage, originalKeyPoints, goodMatches
                , matchOutput, new Scalar(0, 255, 0), new Scalar(255, 0, 0), new MatOfByte(), 2);
        Highgui.imwrite(processDiagram, matchOutput);
    }
}

测试使用

@Test
    public void zhy() {
        String templateFilePath = "D:/test/wd2.jpg";
        String originalFilePath = "D:/test/wd4.jpg";

        int i = ImageRecognition.matchImage(templateFilePath, originalFilePath, "D:/test/zhy.jpg");


        System.out.println("匹配的像素点总数:" + i);
    }

效果

二,部署到linux 

下载opencv源码压缩包

上传到linux服务器,并解压

 安装依赖包,采用yum命令安装,在终端机输入以下指令

yum install cmake gcc gcc-c++ gtk+-devel gimp-devel gimp-devel-tools gimp-help-browser zlib-devel libtiff-devel libjpeg-devel libpng-devel gstreamer-devel libavc1394-devel libraw1394-devel libdc1394-devel jasper-devel jasper-utils swig python libtool nasm build-essential ant

进入文件夹新建build

cd opencv-2.4.11
mkdir build
cd build

编译

cmake -D CMAKE_BUILD_TYPE=RELEASE -D CMAKE_INSTALL_PREFIX=/usr/local ..
make
sudo make install

生成jar

cmake -D CMAKE_BUILD_TYPE=RELEASE -D CMAKE_INSTALL_PREFIX=/usr/local -DBUILD_TESTS=OFF ..

最后运行

make -j8
sudo make install

得到.so库文件和jar

cd /usr/local/share/OpenCV/java/

部署时application.properties引用生成的.so文件即可

### 回答1: 编译Java环境的OpenCV SO和JAR文件涉及到以下几个步骤: 第一步,安装Java开发环境(JDK)和OpenCV库。 第二步,准备好编译Java环境的OpenCV SO文件。打开终端,切换到OpenCV源码的根目录,运行如下命令: cmake -DBUILD_SHARED_LIBS=OFF -DWITH_JAVA=ON \ -DJAVA_AWT_INCLUDE_PATH=/usr/lib/jvm/java-7-openjdk-amd64/include/ \ -DJAVA_AWT_LIBRARY=/usr/lib/jvm/java-7-openjdk-amd64/jre/lib/amd64/libjawt.so \ -DJAVA_INCLUDE_PATH=/usr/lib/jvm/java-7-openjdk-amd64/include/ \ -DJAVA_JVM_LIBRARY=/usr/lib/jvm/java-7-openjdk-amd64/jre/lib/amd64/server/libjvm.so \ -DCMAKE_INSTALL_PREFIX=/usr/local .. 这个过程会自动检测Java的版本,并在编译过程中使用Java相关的库文件。完成后,运行make命令进行编译。 第三步,将编译好的OpenCV SO文件复制到合适的目录下(例如/usr/local/lib/)。 第四步,使用Java中的JNI调用OpenCV库,并编译生成Java环境的OpenCV JAR文件。在Java项目的源码中,引用OpenCV库需要使用到"libopencv_java.so"和"opencv-xxx.jar"两个文件。其中"libopencv_java.so"是在第三步中编译出来的,而"opencv-xxx.jar"需要在编译的过程中生成。在终端中,切换到OpenCV源码的根目录,输入以下命令: cd build/bin jar -cf opencv-xxx.jar org/opencv/* 这个过程会将OpenCVJava API打包成一个JAR文件。在Java项目中引用该JAR文件即可。 以上就是编译Java环境的OpenCV SO和JAR文件的步骤。需要注意的是,在编译过程中如果遇到问题,需要参照实际情况进行适当的调整。 ### 回答2: 在Linux系统中编译Java环境的OpenCV .so和.jar文件,可以按照以下步骤进行: 1.下载OpenCV源代码,解压后进入opencv目录。 2.安装OpenCV所需的依赖包,包括gcc,g++,cmake,git,libgtk2.0-dev,pkg-config,libavcodec-dev,libavformat-dev,libswscale-dev等。 3.使用cmake进行编译。执行以下命令进行配置: mkdir build && cd build cmake -D CMAKE_BUILD_TYPE=RELEASE -D CMAKE_INSTALL_PREFIX=/usr/local .. 4.使用make命令进行编译: make -j4 5.编译完成后,安装OpenCV: sudo make install 6.将编译生成的.so文件拷贝到Java项目的lib目录下。 7.使用Java Native Interface(JNI)来调用OpenCV库,可以在Java项目中创建一个jni目录,并在其中创建NativeUtils.java文件。 8.在NativeUtils.java中定义native方法,并使用Javah命令生成.h文件javah -cp . NativeUtils 9.在生成的.h文件中实现native方法,调用OpenCV库中的方法。 10.编译Java项目,并将生成的.jar文件添加到项目中。 11.运行Java项目,在该项目中使用OpenCV库的相应方法即可。 总之,通过以上步骤,在Linux系统中编译Java环境的OpenCV .so和.jar文件使用OpenCV库中方法不再是困难的事情了。 ### 回答3: 首先,在Ubuntu或其他Linux发行版中安装Java开发工具包(JDK)和OpenCV开发包。可以使用apt-get命令安装这两个包。 安装完OpenCV后,进入opencv/build/java文件夹。这个文件夹包含OpenCV JAR文件和SO文件的源代码。 运行以下命令构建Java JAR文件: ant build-jar 该命令构建了包含OpenCV Java类的JAR文件。 接下来,编译Java原生库SO文件, 运行以下命令: ant build-native 该命令将编译Java原生库SO文件,然后将其复制到OpenCV JAR文件所在的lib目录下。 最后,将OpenCV JAR文件和SO文件添加到Java项目的classpath中。可以使用eclipse或其他Java IDE,或手动编辑项目的classpath。编辑后,可以使用Java代码调用OpenCV库中的函数。 通过上述步骤,便可以在Linux环境下编译Java环境的OpenCV SO和JAR文件
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值