java中的OpenCV Mat对象序列化

时间:2014-11-21 15:42:44

标签: java opencv serialization

我试图序列化对的映射并获得如下的异常:

 java.io.NotSerializableException: org.opencv.core.Mat

有没有某种方法来序列化这个?

4 个答案:

答案 0 :(得分:2)

不,不那么容易。

实际数据保存在c ++ native中,所以你的serialize()不会达到。

可以做什么:

Mat mat = ...
byte[] bytes = new byte[mat.total()*mat.elemSize()];
mat.get(0,0,bytes);
// now somehow save mat.type(), mat.rows(), mat.cols() and the bytes, later restore it:
Mat m2 = new Mat(rows,cols,type);
m2.put(0,0, bytes);

答案 1 :(得分:2)

我从here做了一些改进。经过测试和工作 SerializationUtils类here

public static String matToJson(Mat mat){
    JsonObject obj = new JsonObject();

    if(mat.isContinuous()){
        int cols = mat.cols();
        int rows = mat.rows();
        int elemSize = (int) mat.elemSize();
        int type = mat.type();

        obj.addProperty("rows", rows);
        obj.addProperty("cols", cols);
        obj.addProperty("type", type);

        // We cannot set binary data to a json object, so:
        // Encoding data byte array to Base64.
        String dataString;

        if( type == CvType.CV_32S || type == CvType.CV_32SC2 || type == CvType.CV_32SC3 || type == CvType.CV_16S) {
            int[] data = new int[cols * rows * elemSize];
            mat.get(0, 0, data);
            dataString = new String(Base64.encodeBase64(SerializationUtils.toByteArray(data)));
        }
        else if( type == CvType.CV_32F || type == CvType.CV_32FC2) {
            float[] data = new float[cols * rows * elemSize];
            mat.get(0, 0, data);
            dataString = new String(Base64.encodeBase64(SerializationUtils.toByteArray(data)));
        }
        else if( type == CvType.CV_64F || type == CvType.CV_64FC2) {
            double[] data = new double[cols * rows * elemSize];
            mat.get(0, 0, data);
            dataString = new String(Base64.encodeBase64(SerializationUtils.toByteArray(data)));
        }
        else if( type == CvType.CV_8U ) {
            byte[] data = new byte[cols * rows * elemSize];
            mat.get(0, 0, data);
            dataString = new String(Base64.encodeBase64(data));
        }
        else {

            throw new UnsupportedOperationException("unknown type");
        }
        obj.addProperty("data", dataString);

        Gson gson = new Gson();
        String json = gson.toJson(obj);

        return json;
    } else {
        System.out.println("Mat not continuous.");
    }
    return "{}";
}

public static Mat matFromJson(String json){


    JsonParser parser = new JsonParser();
    JsonObject JsonObject = parser.parse(json).getAsJsonObject();

    int rows = JsonObject.get("rows").getAsInt();
    int cols = JsonObject.get("cols").getAsInt();
    int type = JsonObject.get("type").getAsInt();

    Mat mat = new Mat(rows, cols, type);

    String dataString = JsonObject.get("data").getAsString();
    if( type == CvType.CV_32S || type == CvType.CV_32SC2 || type == CvType.CV_32SC3 || type == CvType.CV_16S) {
        int[] data = SerializationUtils.toIntArray(Base64.decodeBase64(dataString.getBytes()));
        mat.put(0, 0, data);
    }
    else if( type == CvType.CV_32F || type == CvType.CV_32FC2) {
        float[] data = SerializationUtils.toFloatArray(Base64.decodeBase64(dataString.getBytes()));
        mat.put(0, 0, data);
    }
    else if( type == CvType.CV_64F || type == CvType.CV_64FC2) {
        double[] data = SerializationUtils.toDoubleArray(Base64.decodeBase64(dataString.getBytes()));
        mat.put(0, 0, data);
    }
    else if( type == CvType.CV_8U ) {
        byte[] data = Base64.decodeBase64(dataString.getBytes());
        mat.put(0, 0, data);
    }
    else {

        throw new UnsupportedOperationException("unknown type");
    }
    return mat;
}

答案 2 :(得分:0)

我改变了其他代码,看起来很有用。 希望它可以提供帮助。

 public static byte[] serializeMat(Mat mat) {
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            try {
                float[] data = new float[(int) mat.total() * mat.channels()];
                mat.get(0, 0, data);
                ObjectOutput out = new ObjectOutputStream(bos);
                out.writeObject(data);
                out.close();
                // Get the bytes of the serialized object
                byte[] buf = bos.toByteArray();
                return buf;
            } catch (IOException ioe) {
                ioe.printStackTrace();
                return null;
            }
        }

答案 3 :(得分:0)

您可以将内容转储为字符串序列化,如下所示:

Mat mat = new Mat();
ObjectWriter ow = new ObjectMapper().writer().withDefaultPrettyPrinter();
String json = ow.writeValueAsString(mat.dump());