有没有一种方法可以在数据库中存储OpenCV/javacv Mat对象?

zvokhttg  于 2024-01-05  发布在  Java
关注(0)|答案(4)|浏览(202)

在阅读了无数的文档并试图理解有关opencv/javacv提取关键点的示例之后,使用一些DescriptorExtractor计算特征以将输入图像与一堆图像进行匹配,以查看输入图像是其中之一还是其中之一的一部分,我认为,我们应该在计算后存储Mat对象。
我将使用Emily Webb'的代码作为示例:
String smallUrl =“rsz_our-mobile-planet-us-infographic_infographics_lg_unerela.jpg”; String largeUrl =“our-mobile-planet-us-infographic_infographics_lg.jpg”;

  1. IplImage image = cvLoadImage(largeUrl,CV_LOAD_IMAGE_UNCHANGED );
  2. IplImage image2 = cvLoadImage(smallUrl,CV_LOAD_IMAGE_UNCHANGED );
  3. CvMat descriptorsA = new CvMat(null);
  4. CvMat descriptorsB = new CvMat(null);
  5. final FastFeatureDetector ffd = new FastFeatureDetector(40, true);
  6. final KeyPoint keyPoints = new KeyPoint();
  7. final KeyPoint keyPoints2 = new KeyPoint();
  8. ffd.detect(image, keyPoints, null);
  9. ffd.detect(image2, keyPoints2, null);
  10. System.out.println("keyPoints.size() : "+keyPoints.size());
  11. System.out.println("keyPoints2.size() : "+keyPoints2.size());
  12. // BRISK extractor = new BRISK();
  13. //BriefDescriptorExtractor extractor = new BriefDescriptorExtractor();
  14. FREAK extractor = new FREAK();
  15. extractor.compute(image, keyPoints, descriptorsA);
  16. extractor.compute(image2, keyPoints2, descriptorsB);
  17. System.out.println("descriptorsA.size() : "+descriptorsA.size());
  18. System.out.println("descriptorsB.size() : "+descriptorsB.size());
  19. DMatch dmatch = new DMatch();
  20. //FlannBasedMatcher matcher = new FlannBasedMatcher();
  21. //DescriptorMatcher matcher = new DescriptorMatcher();
  22. BFMatcher matcher = new BFMatcher();
  23. matcher.match(descriptorsA, descriptorsB, dmatch, null);
  24. System.out.println(dmatch.capacity());

字符串
我的问题是:我如何在DB中存储mptorsA(或mptorsB)--在opencv的java实现中-?(它们是在extractor.compute(image,keyPoints,mptorsA)之后获得的Mat对象;)
我知道Mat对象在Java实现中不是可序列化的对象,但可以肯定的是,如果你想将一个图像与一组存档图像相匹配,你必须提取存档的描述符并将它们存储在某个地方以供功能使用。

vecaoik1

vecaoik11#

经过一些更多的搜索,我发现一些链接在http://answers.opencv.org/question/8873/best-way-to-store-a-mat-object-in-android/
虽然答案主要针对Android设备,并参考了之前关于保存关键点(Saving ORB feature vectors using OpenCV4Android (java API))的问题,但下面代码中的答案“从Mat对象到xml和xml到Mat对象”似乎是有效的:

  1. import org.opencv.core.CvType;
  2. import org.opencv.core.Mat;
  3. import org.w3c.dom.Document;
  4. import org.w3c.dom.Element;
  5. import org.w3c.dom.Node;
  6. import org.w3c.dom.NodeList;
  7. import javax.xml.parsers.DocumentBuilderFactory;
  8. import javax.xml.transform.OutputKeys;
  9. import javax.xml.transform.Transformer;
  10. import javax.xml.transform.TransformerFactory;
  11. import javax.xml.transform.dom.DOMSource;
  12. import javax.xml.transform.stream.StreamResult;
  13. import java.io.File;
  14. import java.util.Locale;
  15. import java.util.Scanner;
  16. public class TaFileStorage {
  17. // static
  18. public static final int READ = 0;
  19. public static final int WRITE = 1;
  20. // varaible
  21. private File file;
  22. private boolean isWrite;
  23. private Document doc;
  24. private Element rootElement;
  25. public TaFileStorage() {
  26. file = null;
  27. isWrite = false;
  28. doc = null;
  29. rootElement = null;
  30. }
  31. // read or write
  32. public void open(String filePath, int flags ) {
  33. try {
  34. if( flags == READ ) {
  35. open(filePath);
  36. }
  37. else {
  38. create(filePath);
  39. }
  40. } catch(Exception e) {
  41. e.printStackTrace();
  42. }
  43. }
  44. // read only
  45. public void open(String filePath) {
  46. try {
  47. file = new File(filePath);
  48. if( file == null || file.isFile() == false ) {
  49. System.err.println("Can not open file: " + filePath );
  50. }
  51. else {
  52. isWrite = false;
  53. doc = DocumentBuilderFactory.newInstance().newDocumentBuilder().parse(file);
  54. doc.getDocumentElement().normalize();
  55. }
  56. } catch(Exception e) {
  57. e.printStackTrace();
  58. }
  59. }
  60. // write only
  61. public void create(String filePath) {
  62. try {
  63. file = new File(filePath);
  64. if( file == null ) {
  65. System.err.println("Can not wrtie file: " + filePath );
  66. }
  67. else {
  68. isWrite = true;
  69. doc = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
  70. rootElement = doc.createElement("opencv_storage");
  71. doc.appendChild(rootElement);
  72. }
  73. } catch(Exception e) {
  74. e.printStackTrace();
  75. }
  76. }
  77. public Mat readMat(String tag) {
  78. if( isWrite ) {
  79. System.err.println("Try read from file with write flags");
  80. return null;
  81. }
  82. NodeList nodelist = doc.getElementsByTagName(tag);
  83. Mat readMat = null;
  84. for( int i = 0 ; i<nodelist.getLength() ; i++ ) {
  85. Node node = nodelist.item(i);
  86. if( node.getNodeType() == Node.ELEMENT_NODE ) {
  87. Element element = (Element)node;
  88. String type_id = element.getAttribute("type_id");
  89. if( "opencv-matrix".equals(type_id) == false) {
  90. System.out.println("Fault type_id ");
  91. }
  92. String rowsStr = element.getElementsByTagName("rows").item(0).getTextContent();
  93. String colsStr = element.getElementsByTagName("cols").item(0).getTextContent();
  94. String dtStr = element.getElementsByTagName("dt").item(0).getTextContent();
  95. String dataStr = element.getElementsByTagName("data").item(0).getTextContent();
  96. int rows = Integer.parseInt(rowsStr);
  97. int cols = Integer.parseInt(colsStr);
  98. int type = CvType.CV_8U;
  99. Scanner s = new Scanner(dataStr);
  100. s.useLocale(Locale.US);
  101. if( "f".equals(dtStr) ) {
  102. type = CvType.CV_32F;
  103. readMat = new Mat( rows, cols, type );
  104. float fs[] = new float[1];
  105. for( int r=0 ; r<rows ; r++ ) {
  106. for( int c=0 ; c<cols ; c++ ) {
  107. if( s.hasNextFloat() ) {
  108. fs[0] = s.nextFloat();
  109. }
  110. else {
  111. fs[0] = 0;
  112. System.err.println("Unmatched number of float value at rows="+r + " cols="+c);
  113. }
  114. readMat.put(r, c, fs);
  115. }
  116. }
  117. }
  118. else if( "i".equals(dtStr) ) {
  119. type = CvType.CV_32S;
  120. readMat = new Mat( rows, cols, type );
  121. int is[] = new int[1];
  122. for( int r=0 ; r<rows ; r++ ) {
  123. for( int c=0 ; c<cols ; c++ ) {
  124. if( s.hasNextInt() ) {
  125. is[0] = s.nextInt();
  126. }
  127. else {
  128. is[0] = 0;
  129. System.err.println("Unmatched number of int value at rows="+r + " cols="+c);
  130. }
  131. readMat.put(r, c, is);
  132. }
  133. }
  134. }
  135. else if( "s".equals(dtStr) ) {
  136. type = CvType.CV_16S;
  137. readMat = new Mat( rows, cols, type );
  138. short ss[] = new short[1];
  139. for( int r=0 ; r<rows ; r++ ) {
  140. for( int c=0 ; c<cols ; c++ ) {
  141. if( s.hasNextShort() ) {
  142. ss[0] = s.nextShort();
  143. }
  144. else {
  145. ss[0] = 0;
  146. System.err.println("Unmatched number of int value at rows="+r + " cols="+c);
  147. }
  148. readMat.put(r, c, ss);
  149. }
  150. }
  151. }
  152. else if( "b".equals(dtStr) ) {
  153. readMat = new Mat( rows, cols, type );
  154. byte bs[] = new byte[1];
  155. for( int r=0 ; r<rows ; r++ ) {
  156. for( int c=0 ; c<cols ; c++ ) {
  157. if( s.hasNextByte() ) {
  158. bs[0] = s.nextByte();
  159. }
  160. else {
  161. bs[0] = 0;
  162. System.err.println("Unmatched number of byte value at rows="+r + " cols="+c);
  163. }
  164. readMat.put(r, c, bs);
  165. }
  166. }
  167. }
  168. }
  169. }
  170. return readMat;
  171. }
  172. public void writeMat(String tag, Mat mat) {
  173. try {
  174. if( isWrite == false) {
  175. System.err.println("Try write to file with no write flags");
  176. return;
  177. }
  178. Element matrix = doc.createElement(tag);
  179. matrix.setAttribute("type_id", "opencv-matrix");
  180. rootElement.appendChild(matrix);
  181. Element rows = doc.createElement("rows");
  182. rows.appendChild( doc.createTextNode( String.valueOf(mat.rows()) ));
  183. Element cols = doc.createElement("cols");
  184. cols.appendChild( doc.createTextNode( String.valueOf(mat.cols()) ));
  185. Element dt = doc.createElement("dt");
  186. String dtStr;
  187. int type = mat.type();
  188. if(type == CvType.CV_32F ) { // type == CvType.CV_32FC1
  189. dtStr = "f";
  190. }
  191. else if( type == CvType.CV_32S ) { // type == CvType.CV_32SC1
  192. dtStr = "i";
  193. }
  194. else if( type == CvType.CV_16S ) { // type == CvType.CV_16SC1
  195. dtStr = "s";
  196. }
  197. else if( type == CvType.CV_8U ){ // type == CvType.CV_8UC1
  198. dtStr = "b";
  199. }
  200. else {
  201. dtStr = "unknown";
  202. }
  203. dt.appendChild( doc.createTextNode( dtStr ));
  204. Element data = doc.createElement("data");
  205. String dataStr = dataStringBuilder( mat );
  206. data.appendChild( doc.createTextNode( dataStr ));
  207. // append all to matrix
  208. matrix.appendChild( rows );
  209. matrix.appendChild( cols );
  210. matrix.appendChild( dt );
  211. matrix.appendChild( data );
  212. } catch(Exception e) {
  213. e.printStackTrace();
  214. }
  215. }
  216. private String dataStringBuilder(Mat mat) {
  217. StringBuilder sb = new StringBuilder();
  218. int rows = mat.rows();
  219. int cols = mat.cols();
  220. int type = mat.type();
  221. if( type == CvType.CV_32F ) {
  222. float fs[] = new float[1];
  223. for( int r=0 ; r<rows ; r++ ) {
  224. for( int c=0 ; c<cols ; c++ ) {
  225. mat.get(r, c, fs);
  226. sb.append( String.valueOf(fs[0]));
  227. sb.append( ' ' );
  228. }
  229. sb.append( '\n' );
  230. }
  231. }
  232. else if( type == CvType.CV_32S ) {
  233. int is[] = new int[1];
  234. for( int r=0 ; r<rows ; r++ ) {
  235. for( int c=0 ; c<cols ; c++ ) {
  236. mat.get(r, c, is);
  237. sb.append( String.valueOf(is[0]));
  238. sb.append( ' ' );
  239. }
  240. sb.append( '\n' );
  241. }
  242. }
  243. else if( type == CvType.CV_16S ) {
  244. short ss[] = new short[1];
  245. for( int r=0 ; r<rows ; r++ ) {
  246. for( int c=0 ; c<cols ; c++ ) {
  247. mat.get(r, c, ss);
  248. sb.append( String.valueOf(ss[0]));
  249. sb.append( ' ' );
  250. }
  251. sb.append( '\n' );
  252. }
  253. }
  254. else if( type == CvType.CV_8U ) {
  255. byte bs[] = new byte[1];
  256. for( int r=0 ; r<rows ; r++ ) {
  257. for( int c=0 ; c<cols ; c++ ) {
  258. mat.get(r, c, bs);
  259. sb.append( String.valueOf(bs[0]));
  260. sb.append( ' ' );
  261. }
  262. sb.append( '\n' );
  263. }
  264. }
  265. else {
  266. sb.append("unknown type\n");
  267. }
  268. return sb.toString();
  269. }
  270. public void release() {
  271. try {
  272. if( isWrite == false) {
  273. System.err.println("Try release of file with no write flags");
  274. return;
  275. }
  276. DOMSource source = new DOMSource(doc);
  277. StreamResult result = new StreamResult(file);
  278. // write to xml file
  279. Transformer transformer = TransformerFactory.newInstance().newTransformer();
  280. transformer.setOutputProperty(OutputKeys.INDENT, "yes");
  281. // do it
  282. transformer.transform(source, result);
  283. } catch(Exception e) {
  284. e.printStackTrace();
  285. }
  286. }
  287. }

字符串

展开查看全部
gt0wga4j

gt0wga4j2#

由于Thorben提出的代码在我的情况下很慢,我使用序列化提出了以下代码。

  1. public final void saveMat(String path, Mat mat) {
  2. File file = new File(path).getAbsoluteFile();
  3. file.getParentFile().mkdirs();
  4. try {
  5. int cols = mat.cols();
  6. float[] data = new float[(int) mat.total() * mat.channels()];
  7. mat.get(0, 0, data);
  8. try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(path))) {
  9. oos.writeObject(cols);
  10. oos.writeObject(data);
  11. oos.close();
  12. }
  13. } catch (IOException | ClassCastException ex) {
  14. System.err.println("ERROR: Could not save mat to file: " + path);
  15. Logger.getLogger(this.class.getName()).log(Level.SEVERE, null, ex);
  16. }
  17. }
  18. public final Mat loadMat(String path) {
  19. try {
  20. int cols;
  21. float[] data;
  22. try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream(path))) {
  23. cols = (int) ois.readObject();
  24. data = (float[]) ois.readObject();
  25. }
  26. Mat mat = new Mat(data.length / cols, cols, CvType.CV_32F);
  27. mat.put(0, 0, data);
  28. return mat;
  29. } catch (IOException | ClassNotFoundException | ClassCastException ex) {
  30. System.err.println("ERROR: Could not load mat from file: " + path);
  31. Logger.getLogger(this.class.getName()).log(Level.SEVERE, null, ex);
  32. }
  33. return null;
  34. }

字符串
对于描述符你OpenCV使用垫的浮动,在其他情况下,你必须相应地修改代码这个列表中找到here

  1. CV_8U and CV_8S -> byte[]
  2. CV_16U and CV_16S -> short[]
  3. CV_32S -> int[]
  4. CV_32F -> float[]
  5. CV_64F-> double[]

展开查看全部
35g0bw71

35g0bw713#

搜索完所有的答案后,我编辑了一些代码,它似乎工作。我用它来存储筛选描述符到HBase。

  1. public static byte[] serializeMat(Mat mat) {
  2. ByteArrayOutputStream bos = new ByteArrayOutputStream();
  3. try {
  4. float[] data = new float[(int) mat.total() * mat.channels()];
  5. mat.get(0, 0, data);
  6. ObjectOutput out = new ObjectOutputStream(bos);
  7. out.writeObject(data);
  8. out.close();
  9. // Get the bytes of the serialized object
  10. byte[] buf = bos.toByteArray();
  11. return buf;
  12. } catch (IOException ioe) {
  13. ioe.printStackTrace();
  14. return null;
  15. }
  16. }

字符串

展开查看全部
qnakjoqk

qnakjoqk4#

我还遇到了Javaorg.opencv没有原生OpenCV的FileStorage功能的问题。

  1. package net.sf.jaer.util;
  2. import com.esotericsoftware.yamlbeans.YamlException;
  3. import com.esotericsoftware.yamlbeans.YamlReader;
  4. import com.esotericsoftware.yamlbeans.YamlWriter;
  5. import java.io.FileReader;
  6. import java.io.FileNotFoundException;
  7. import java.io.FileWriter;
  8. import java.io.IOException;
  9. import java.util.ArrayList;
  10. import java.util.List;
  11. import java.util.Map;
  12. import java.util.logging.Logger;
  13. import org.opencv.core.CvType;
  14. import org.opencv.core.Mat;
  15. /**
  16. * Adapted from
  17. * https://stackoverflow.com/questions/34297676/how-to-load-opencv-matrices-saved-with-filestorage-in-java
  18. *
  19. * using
  20. * https://github.com/EsotericSoftware/yamlbeans
  21. *
  22. * @author https://stackoverflow.com/users/3303546/cecilia
  23. */
  24. public class YamlMatFileStorage {
  25. private static final Logger log = Logger.getLogger("YamlMatLoader");
  26. // Mat cannot be used directly because it is a native object and yamlbeans only deals with Java objects
  27. protected static class MatStorage {
  28. public int rows;
  29. public int cols;
  30. public String dt;
  31. public List<String> data;
  32. // The empty constructor is required by YamlReader
  33. public MatStorage() {
  34. }
  35. public double[] getData() {
  36. double[] dataOut = new double[data.size()];
  37. for (int i = 0; i < dataOut.length; i++) {
  38. dataOut[i] = Double.parseDouble(data.get(i));
  39. }
  40. return dataOut;
  41. }
  42. public void putData(Mat m) {
  43. rows = m.rows();
  44. cols = m.cols();
  45. data = new ArrayList();
  46. for (int r = 0; r < rows; r++) {
  47. for (int c = 0; c < cols; c++) {
  48. data.add(Double.toString(m.get(r, c)[0]));
  49. }
  50. }
  51. }
  52. }
  53. /**
  54. * Reads a Mat from a YAML file
  55. *
  56. * @param path path to YAML file
  57. * @return the Mat
  58. */
  59. public Mat readMatYml(String path) throws FileNotFoundException, YamlException, IOException {
  60. Mat m=null;
  61. try (YamlReader reader = new YamlReader(new FileReader(path))) {
  62. MatStorage matStorage = (MatStorage) reader.read();
  63. // Create a new Mat to hold the extracted data
  64. m = new Mat(matStorage.rows, matStorage.cols, CvType.CV_32FC1);
  65. m.put(0, 0, matStorage.getData());
  66. log.info(String.format("From file %s loaded Mat %s", path, m));
  67. }
  68. return m;
  69. }
  70. // Loading function
  71. /**
  72. * Reads a Mat from a YAML file
  73. *
  74. * @param path path to YAML file
  75. * @param m the Mat to write
  76. * @throws java.io.IOException
  77. * @throws com.esotericsoftware.yamlbeans.YamlException
  78. */
  79. public void writeMatYml(String path, Mat m) throws IOException, YamlException {
  80. try (YamlWriter writer = new YamlWriter(new FileWriter(path))) {
  81. // write the Mat
  82. MatStorage ms = new MatStorage();
  83. ms.putData(m);
  84. writer.write(ms);
  85. }
  86. log.info(String.format("Wrote Mat %s to file %s", m, path));
  87. }
  88. }

字符串
存储的文件看起来像这样:

  1. !net.sf.jaer.util.YamlMatFileStorage$MatStorage
  2. cols: 3
  3. data:
  4. - 1735.0830676085973
  5. - 0.0
  6. - 173.80815300223972
  7. - 0.0
  8. - 1720.885058317215
  9. - 147.99521279976508
  10. - 0.0
  11. - 0.0
  12. - 1.0
  13. rows: 3

展开查看全部

相关问题