Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/csharp/304.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
C# 同时编写和验证XML_C#_.net_Xml_Xsd - Fatal编程技术网

C# 同时编写和验证XML

C# 同时编写和验证XML,c#,.net,xml,xsd,C#,.net,Xml,Xsd,我有一个Write方法,可以序列化使用XmlAttributes的对象。这很标准,就像这样: private bool WriteXml(DirectoryInfo dir) { var xml = new XmlSerializer(typeof (Composite)); _filename = Path.Combine(dir.FullName, _composite.Symbol + ".xml"); using (var xmlFile = File.Creat

我有一个Write方法,可以序列化使用XmlAttributes的对象。这很标准,就像这样:

private bool WriteXml(DirectoryInfo dir)
{
    var xml = new XmlSerializer(typeof (Composite));
    _filename = Path.Combine(dir.FullName, _composite.Symbol + ".xml");
    using (var xmlFile = File.Create(_filename))
    {
          xml.Serialize(xmlFile, _composite);
    }
    return true;
}
除了尝试读取我刚刚写出来的文件(使用模式验证器),我还可以在编写XML时执行XSD验证吗


在将内存流写入磁盘之前,我可以处理内存流,但在.Net中似乎通常有一种优雅的方法来解决大多数问题。

我这样做的方式对任何感兴趣的人来说都是这样的:

public Composite Read(Stream stream)
{
    _errors = null;
    var settings = new XmlReaderSettings();
    using (var fileStream = File.OpenRead(XmlComponentsXsd))
    {
        using (var schemaReader = new XmlTextReader(fileStream))
        {
            settings.Schemas.Add(null, schemaReader);
            settings.ValidationType = ValidationType.Schema;
            settings.ValidationEventHandler += OnValidationEventHandler;
            using (var xmlReader = XmlReader.Create(stream, settings))
            {
                var serialiser = new XmlSerializer(typeof (Composite));
                return (Composite) serialiser.Deserialize(xmlReader);
            }
        }
    }
}

private ValidationEventArgs _errors = null;
private void OnValidationEventHandler(object sender, ValidationEventArgs validationEventArgs)
{
    _errors = validationEventArgs;
}
然后,使用内存流执行以下操作,而不是将XML写入文件:

private bool WriteXml(DirectoryInfo dir)
{
    var xml = new XmlSerializer(typeof (Composite));
    var filename = Path.Combine(dir.FullName, _composite.Symbol + ".xml");
    // first write it to memory
    var memStream = new MemoryStream();
    xml.Serialize(memStream, _composite);
    memStream.Position = 0;
    Read(memStream);
    if (_errors != null)
    {
        throw new Exception(string.Format("Error writing to {0}. XSD validation failed : {1}", filename, _errors.Message));
    }
    memStream.Position = 0;
    using (var outFile = File.OpenWrite(filename))
    {
        memStream.CopyTo(outFile);
    }
    memStream.Dispose();
    return true;
}

这样,在将任何内容写入磁盘之前,您总是根据架构进行验证。

不要在问题的标题中添加标记。有关更多信息,请参阅。XML序列化程序是否已生成兼容的XML?再次检查它有什么意义?针对业务不同部分共享的XSD。我想知道,如果我更改了代码中的某些内容,我没有违反XSD规则。因此,实际上,这是一个自动集成测试,当代码更改时,您只需对生成的XML运行一次。部分是的,但是XSD中存在限制,在某些字符串上使用regex等。我不希望重复这些签入代码,特别是因为XSD将来可能会更改。除非有人总是记得检查代码,否则它可能会失败。那也没那么糟糕,除非我需要马上知道失败是否发生了,而不是等到别人读了以后。