C#DataContracts是否足以正确描述GeoJSON?

时间:2012-02-17 11:32:37

标签: c# json geojson datacontract datacontractjsonserializer

所以我一直在尝试将GeoJSON映射到C#DataContracts,以便我们能够轻松支持在WCF / REST-apis中返回GeoJSON。我越是使用DataContracts和序列化,我越觉得它是一个我无法理解的魔法黑盒子。

无论如何,就我而言:

GeoJSON.cs:

namespace GeoJSON {

[DataContract]
[KnownType(typeof (Point))]
[KnownType(typeof (MultiPoint))]
[KnownType(typeof (LineString))]
[KnownType(typeof (MultiLineString))]
[KnownType(typeof (Polygon))]
[KnownType(typeof (MultiPolygon))]
[KnownType(typeof (Feature))]
[KnownType(typeof (FeatureCollection))]
[KnownType(typeof (Geometry))]
[KnownType(typeof(GeometryCollection))]
public abstract class GeoJson {

    // This is because WCF services can't read the DataContracts KnownType's above
    // so you need to break DRY and repeat yourself here and add this class
    // to a ServiceKnownType-attribute to every service method returning GeoJSON
    // Like this:
    // [ServiceKnownType("Types", typeof(GeoJson.KnownTypes))]
    public static class KnownTypes {
        public static Type[] Types(ICustomAttributeProvider provider) {
            return new[] {
                typeof (Point),
                typeof (MultiPoint),
                typeof (LineString),
                typeof (MultiLineString),
                typeof (Polygon),
                typeof (MultiPolygon),
                typeof (Feature),
                typeof (FeatureCollection),
                typeof (Geometry),
                typeof (GeometryCollection)
            };
        }
    }
}

#region Feature

// The idea here (and elsewhere) is that if i let the type field be enums the 
// deserializer might use that information to chose the right class. 
// Is this just crack?
[DataContract]
public enum FeatureType {
    [EnumMember] Feature
}

public class Feature : GeoJson {
    public Feature() {
        Type = FeatureType.Feature;
        Properties = new Dictionary<string, string>();
    }

    public Feature(Geometry geometry) {
        Type = FeatureType.Feature;
        Properties = new Dictionary<string, string>();
        Geometry = geometry;
    }

    [DataMember(Name = "type")]
    public FeatureType Type { get; set; }

    [DataMember(Name = "id")]
    public string Id { get; set; }

    [DataMember(IsRequired = true, Name = "geometry")]
    public Geometry Geometry { get; set; }

    // TODO: Make this support proper JSON-trees
    [DataMember(IsRequired = true, Name = "properties")]
    public IDictionary<string, string> Properties { get; set; }


}

#endregion Feature

#region FeatureCollection

[DataContract]
public enum FeatureCollectionType {
    [EnumMember]
    FeatureCollection
}

[DataContract(Name = "FeatureCollection")]
public class FeatureCollection : GeoJson {
    public FeatureCollection() {
        Type = FeatureCollectionType.FeatureCollection;
        Features = new Feature[0];
    }

    public FeatureCollection(params Feature[] features) {
        Type = FeatureCollectionType.FeatureCollection;
        Features = features ?? new Feature[0];
    }

    [DataMember(Name = "type")]
    private FeatureCollectionType Type { get; set; }

    [DataMember(Name = "features", IsRequired = true)]
    private Feature[] Features { get; set; }
}

#endregion FeatureCollection

#region GeometryCollection

[DataContract]
public enum GeometryCollectionType {
    [EnumMember] GeometryCollection
}

[DataContract(Name = "GeometryCollection")]
public class GeometryCollection : GeoJson {
    public GeometryCollection() {
        Type = GeometryCollectionType.GeometryCollection;
        Geometries = new Geometry[0];
    }

    public GeometryCollection(params Geometry[] geometries) {
        Type = GeometryCollectionType.GeometryCollection;
        Geometries = geometries ?? new Geometry[0];
    }

    [DataMember(Name = "type")]
    private GeometryCollectionType Type { get; set; }

    [DataMember(Name = "geometries", IsRequired = true)]
    private Geometry[] Geometries { get; set; }
}

#endregion GeometryCollection


#region Geometry

/* TODO: 
 *   - More constructors
 *   - Enforce some more invariants.
 */

[DataContract]
[KnownType(typeof(Point))]
[KnownType(typeof(MultiPoint))]
[KnownType(typeof(LineString))]
[KnownType(typeof(MultiLineString))]
[KnownType(typeof(Polygon))]
[KnownType(typeof(MultiPolygon))]
public abstract class Geometry : GeoJson { }

[DataContract]
public enum PointType {
    [EnumMember(Value="Point")] Point
}
[DataContract]
public class Point : Geometry {
    public Point() {
        Type = PointType.Point;
    }

    [DataMember(Name = "type", Order = 0)]
    public PointType Type { get; set; }

    [DataMember(Name = "coordinates", Order = 1)]
    public double[] Coordinates { get; set; }
}

[DataContract]
public enum MultiPointType {
    [EnumMember(Value = "MultiPoint")] MultiPoint
}
[DataContract]
public class MultiPoint : Geometry {
    public MultiPoint() {
        Type = MultiPointType.MultiPoint;
    }

    [DataMember(Name = "type", Order = 0)]
    public MultiPointType Type { get; set; }

    [DataMember(Name = "coordinates", Order = 1)]
    public double[][] Coordinates { get; set; }
}

[DataContract]
public enum LineStringType {
    [EnumMember] LineString
}
[DataContract]
public class LineString : Geometry {
    public LineString() {
        Type = LineStringType.LineString;
    }

    [DataMember(Name = "type", Order = 0)]
    public LineStringType Type { get; set; }

    [DataMember(Name = "coordinates", Order = 1)]
    public double[] Coordinates { get; set; }
}

[DataContract]
public enum MultiLineStringType {
    [EnumMember] MultiLineString
}
[DataContract]
public class MultiLineString : Geometry {
    public MultiLineString() {
        Type = MultiLineStringType.MultiLineString;
    }

    [DataMember(Name = "type", Order = 0)]
    public MultiLineStringType Type { get; set; }

    [DataMember(Name = "coordinates", Order = 1)]
    public double[][] Coordinates { get; set; }
}

[DataContract]
public enum PolygonType {
    [EnumMember] Polygon
}
[DataContract]
public class Polygon : Geometry {
    public Polygon() {
        Type = PolygonType.Polygon;
    }

    [DataMember(Name = "type", Order = 0)]
    public PolygonType Type { get; set; }

    [DataMember(Name = "coordinates", Order = 1)]
    public double[][] Coordinates { get; set; }
}

[DataContract]
public enum MultiPolygonType {
    [EnumMember] MultiPolygon
}
[DataContract]
public class MultiPolygon : Geometry {
    public MultiPolygon() {
        Type = MultiPolygonType.MultiPolygon;
    }

    [DataMember(Name = "type", Order = 0)]
    public MultiPolygonType Type { get; set; }

    [DataMember(Name = "coordinates", Order = 1)]
    public double[][][] Coordinates { get; set; }

}

#endregion Geometry 
}

的Program.cs:

namespace Test {
internal class Program {
    private static void Main(string[] args) {
        var linestring = new LineString {
            Coordinates = new[] {12.0, 57.1, 13, 14}
        };

        var point = new Point {
            Coordinates = new[] { 12.0, 57.1 }
        };

        var geometryCollection = new GeometryCollection(point, linestring);

        var feature = new Feature(linestring);

        var featureCollection = new FeatureCollection(
            feature,
            new Feature(point)
        );


        try {
            WriteObject("LineString", linestring);
            WriteObject("Point", point);
            WriteObject("GeometryCollection",geometryCollection);
            WriteObject("Feature", feature);
            WriteObject("FeatureCollection", featureCollection);
        }
        catch (Exception ex) {
            Console.WriteLine("An exception occured: " + ex.Message);
        }
        Console.ReadKey();
    }

    public static void WriteObject(string label, GeoJson json) {
        var stream = new MemoryStream();
        var ser = new DataContractJsonSerializer(json.GetType());
        ser.WriteObject(stream, json);
        stream.Position = 0;
        var sr = new StreamReader(stream);
        Console.Write("\n" + label + ":\n\t");
        Console.WriteLine(sr.ReadToEnd());
    }
}
}

程序的输出:

LineString:
    {"type":0,"coordinates":[12,57.1,13,14]}

Point:
    {"type":0,"coordinates":[12,57.1]}

GeometryCollection:
    {"geometries":[{"__type":"Point:#GeoJSON","type":0,"coordinates":[12,57.1]},{"__type":"LineString:#GeoJSON","type":0,"coordinates":[12,57.1,13,14]}],"type":0}

Feature:
    {"geometry":{"__type":"LineString:#GeoJSON","type":0,"coordinates":[12,57.1,13,14]},"id":null,"properties":[],"type":0}

FeatureCollection:
    {"features":[{"geometry":{"__type":"LineString:#GeoJSON","type":0,"coordinates":[12,57.1,13,14]},"id":null,"properties":[],"type":0},{"geometry":{"__type":"Point:#GeoJSON","type":0,"coordinates":[12,57.1]},"id":null,"properties":[], "type":0}],"type":0}

问题

  1. 我将这些__type标签洒在
  2. 周围
  3. 类型字段被序列化为“0”而不是实际名称。
  4. 在WCF服务中使用它们时,我也遇到了这些问题。有任何想法吗? 关于我是否会出现反序列化或者这个想法是否合理的任何暗示?

1 个答案:

答案 0 :(得分:0)

我可以肯定地说,除非你没有在多态方案中序列化类型,否则无法绕过__type发射。

一种可能的解决方案是创建一些不会调用多态的包装器操作,并通过该操作而不是通过poly方法返回对象。

JSON序列化程序确实有一个名为alwaysEmitTypeInformation的标志,但是你打开的东西总是发出__type。现在有办法将其关闭,主要是为了避免无意的用户错误。

另见我的回答:Rename __type-field in WCF services