Что я делаю не так здесь, возникли проблемы с XSLT с использованием C # - PullRequest
2 голосов
/ 02 февраля 2010

Пожалуйста, посмотрите на следующий код, поскольку я получаю сообщение об ошибке в этой строке:

xslt.Transform(mydoc.CreateReader(), writer);

Ошибка:

Шаг вперед: переход по не-пользовательскому коду 'System.Xml.Linq.XNode.CreateReader' Первое случайное исключение типа «System.NullReferenceException» произошло в System.Data.SqlXml.dll

((System.NullReferenceException) (ех))

PromotionsDataContext db = new PromotionsDataContext();
//XmlDocument myxml;

XElement Categories =
    new XElement("Promotions",
        from b in db.GetPromotions()
        select new XElement("Promotion",
            new XElement("Category", b.CategoryName),
               new XElement("Client", b.ClientName),
               new XElement("Title", b.Title)));



XDocument mydoc = new XDocument();
mydoc.Add(Categories);


try
{

    XDocument newTree = new XDocument();


    XmlWriterSettings settings = new XmlWriterSettings();
    settings.OmitXmlDeclaration = true;
    settings.ConformanceLevel = ConformanceLevel.Fragment;
    settings.CloseOutput = false;

    //using (XmlWriter writer = newTree.CreateWriter())
    using (XmlWriter writer = XmlWriter.Create(newTree.CreateWriter(), settings))
    {

        // Load the style sheet.
        XslCompiledTransform xslt = new XslCompiledTransform();


        xslt.Load(XmlReader.Create(new FileStream(@"C:\1\TransList.xslt", System.IO.FileMode.Open)));

        // Execute the transform and output the results to a writer.

        xslt.Transform(mydoc.CreateReader(), writer);
    }

    Console.WriteLine(newTree);
}
catch (Exception ex)
{
    Console.Write(ex);

}

Вот XSLT:

<xsl:stylesheet version='1.0' xmlns:xsl='http://www.w3.org/1999/XSL/Transform'>
  <xsl:output method='xml' />
  <xsl:key name='categories' match='Category' use='.' />
  <xsl:template match='/'>
    <xsl:for-each select="/Promotions/Promotion/Category[ 
        generate-id(.) = generate-id(key('categories', .)[1]) 
      ]">
      <xsl:variable name='cname' select='.' />
      <Category title='{.}'>
        <xsl:for-each select='/Promotions/Promotion[Category=$cname]'>
          <Title>
            <xsl:value-of select='Title' />
          </Title>
        </xsl:for-each>
      </Category>
    </xsl:for-each>
  </xsl:template>
</xsl:stylesheet>

Ошибка:

System.NullReferenceException: ссылка на объект не установлена ​​для экземпляра объекта. в System.Xml.Xsl.Runtime.XmlMergeSequenceWriter.StartTree (XPathNodeType rootType, IXmlNamespaceResolver nsResolver, XmlNameTable nameTable) в System.Xml.Xsl.Runtime.XmlQueryOutput.StartTree (XPathNodeType rootType) в System.Xml.Xsl.Runtime.XmlQueryOutput.WriteStartRoot () в корне (время выполнения XmlQueryRuntime {urn: schemas-microsoft-com: xslt-debug}) при выполнении (время выполнения XmlQueryRuntime {urn: schemas-microsoft-com: xslt-debug}) в System.Xml.Xsl.XmlILCommand.Execute (Объект defaultDocument, источники данных XmlResolver, XsltArgumentList argumentsList, средство записи XmlWriter, логическое значение closeWriter) в System.Xml.Xsl.XmlILCommand.Execute (XmlReader contextDocument, источники данных XmlResolver, XsltArgumentList argumentsList, результаты XmlWriter) в System.Xml.Xsl.XslCompiledTransform.Transform (вход XmlReader, аргументы XsltArgumentList, результаты XmlWriter) в Promo.Page_Load (Отправитель объекта, EventArgs e) в c: \ 1 \ promo.ascx.cs: строка 144

Теперь, если я сделаю это, это сработает:

StringWriter sw = new StringWriter(); 
xslt.Transform(mydoc.CreateReader(),null, sw);

Что я делаю не так с XmlWriter?

Значение xdoc непосредственно перед сбоем:

<Promotions>
  <Promotion>
    <Category>Arts &amp; Entertainment</Category>
    <Client>Client1</Client>
    <Title>Get your Free 2</Title>
  </Promotion>
  <Promotion>
    <Category>Arts &amp; Entertainment</Category>
    <Client>Client1</Client>
    <Title>Get your Free 4</Title>
  </Promotion>
  <Promotion>
    <Category>Arts &amp; Entertainment</Category>
    <Client>client1</Client>
    <Title>Get your Free 5</Title>
  </Promotion>
  <Promotion>
    <Category>Community &amp; Neighborhood</Category>
    <Client>Client2</Client>
    <Title>Get your Free 1</Title>
  </Promotion>
  <Promotion>
    <Category>Education</Category>
    <Client>Client3</Client>
    <Title>Get Your Free 3</Title>
  </Promotion>
</Promotions>

Ответы [ 5 ]

2 голосов
/ 02 февраля 2010

Я думаю, что проблема действительно в том, что вы, похоже, хотите создать фрагмент XML с вашей таблицей стилей и XmlWriter, в то время как объектная модель LINQ to XML (т.е. System.Xml.Linq.XDocument / XNode) не имеет какого-либо класса, представляющего фрагменты.

Код работает для меня безупречно, если я беру ваш ввод XML (например,

<Promotions>
  <Promotion>
    <Category>Arts &amp; Entertainment</Category>
    <Client>Client1</Client>
    <Title>Get your Free 2</Title>
  </Promotion>
  <Promotion>
    <Category>Arts &amp; Entertainment</Category>
    <Client>Client1</Client>
    <Title>Get your Free 4</Title>
  </Promotion>
  <Promotion>
    <Category>Arts &amp; Entertainment</Category>
    <Client>client1</Client>
    <Title>Get your Free 5</Title>
  </Promotion>
  <Promotion>
    <Category>Community &amp; Neighborhood</Category>
    <Client>Client2</Client>
    <Title>Get your Free 1</Title>
  </Promotion>
  <Promotion>
    <Category>Education</Category>
    <Client>Client3</Client>
    <Title>Get Your Free 3</Title>
  </Promotion>
</Promotions>

) и опубликована вторая таблица стилей (например,

)
<xsl:stylesheet version='1.0' xmlns:xsl='http://www.w3.org/1999/XSL/Transform'>
  <xsl:output method='xml' />
  <xsl:key name='categories' match='Category' use='.' />
  <xsl:template match='/'>
    <Categories>
      <!-- Added a root element here -->
      <xsl:for-each select="/Promotions/Promotion/Category[ 
            generate-id(.) = generate-id(key('categories', .)[1]) 
          ]">
        <xsl:variable name='cname' select='.' />
        <Category title='{.}'>
          <xsl:for-each select='/Promotions/Promotion[Category=$cname]'>
            <Title>
              <xsl:value-of select='Title' />
            </Title>
          </xsl:for-each>
        </Category>
      </xsl:for-each>
    </Categories>
  </xsl:template>
</xsl:stylesheet>

) и затем используйте следующий код C #:

    XDocument mydoc = XDocument.Load(@"..\..\XMLFile1.xml");

    XDocument newTree = new XDocument();

    using (XmlWriter writer = newTree.CreateWriter())
    {
        XslCompiledTransform xslt = new XslCompiledTransform();

        xslt.Load(@"..\..\XSLTFile2.xslt");

        xslt.Transform(mydoc.CreateReader(), writer);
        writer.Close();
    }

    Console.WriteLine(newTree);

Когда я использую оригинальную таблицу стилей (например,

<xsl:stylesheet version='1.0' xmlns:xsl='http://www.w3.org/1999/XSL/Transform'>
  <xsl:output method='xml' />
  <xsl:key name='categories' match='Category' use='.' />
  <xsl:template match='/'>
    <xsl:for-each select="/Promotions/Promotion/Category[ 
        generate-id(.) = generate-id(key('categories', .)[1]) 
      ]">
      <xsl:variable name='cname' select='.' />
      <Category title='{.}'>
        <xsl:for-each select='/Promotions/Promotion[Category=$cname]'>
          <Title>
            <xsl:value-of select='Title' />
          </Title>
        </xsl:for-each>
      </Category>
    </xsl:for-each>
  </xsl:template>
</xsl:stylesheet>

) Я получаю исключение InvalidOperationException, в котором говорится «Token StartElement в состоянии EndRootElement приведет к недопустимому документу XML. Убедитесь, что для параметра ConformanceLevel установлено значение ConformanceLevel.Fragment или ConformanceLevel.Auto, если вы хотите записать фрагмент XML.». Я подозреваю, что вы пытались противостоять этому, оборачивая свой XmlWriter в другой с помощью ConformanceLevel.Fragment, как вы это делали в своем коде C #, но я думаю, что это не работает, это приводит только к другому исключению.

Что должно работать на мой взгляд, это использовать CreateWriter () в экземпляре XElement, так как должна быть возможность добавить фрагмент в XElement. Однако мой тест по-прежнему вызывает исключение, поэтому я подал ошибку, см. https://connect.microsoft.com/VisualStudio/feedback/details/530052/xslt-stylesheet-writing-an-xml-fragment-to-an-xmlwriter-created-with-xelementinstance-createwriter-causes-nullreferenceexception

2 голосов
/ 02 февраля 2010

Похоже, что ваш XSLT создает фрагмент XML , содержащий список <Category> элементов, а не полный документ XML. И вы пытаетесь записать фрагмент в пустую XDocument. Это может привести к неверному документу XML, так как вам всегда нужен один корневой элемент в XML. Я не знаю, именно ли это является причиной вашего исключения, но вы должны увидеть, что произойдет, когда вы измените свой XSLT так:

<xsl:stylesheet version='1.0' xmlns:xsl='http://www.w3.org/1999/XSL/Transform'>
  <xsl:output method='xml' />
  <xsl:key name='categories' match='Category' use='.' />
  <xsl:template match='/'>
    <Categories> <!-- Added a root element here -->
        <xsl:for-each select="/Promotions/Promotion/Category[ 
            generate-id(.) = generate-id(key('categories', .)[1]) 
          ]">
          <xsl:variable name='cname' select='.' />
          <Category title='{.}'>
            <xsl:for-each select='/Promotions/Promotion[Category=$cname]'>
              <Title>
                <xsl:value-of select='Title' />
              </Title>
            </xsl:for-each>
          </Category>
        </xsl:for-each>
    </Categories>
  </xsl:template>
</xsl:stylesheet>
0 голосов
/ 02 февраля 2010

Вы хотели создать документ XML 2.0 в своем коде new XDeclaration("2.0","utf-8","true")? Я не знаю, является ли это причиной проблемы, но это, безусловно, странная вещь, поскольку XML 2.0 не существует. Попробуйте удалить его полностью; тебе это не нужно.

0 голосов
/ 02 февраля 2010

Поскольку XML создается через LINQ to XML, выполнение этого кода может быть отложено:

XElement Categories = new XElement(
    "Promotions",
    from b in db.GetPromotions()
    select new XElement("Promotion",
        new XElement("Category", b.CategoryName),
           new XElement("Client", b.ClientName),
           new XElement("Title", b.Title)));

Ваша ошибка, вероятно, происходит в этом коде. Попробуйте установить точку останова в вашем LINQ, чтобы пройти через процесс создания XElement. Или, чтобы убедиться, что выполнение не отложено, вы можете сделать следующее:

XElement Categories = new XElement(
    "Promotions",
    (
        from b in db.GetPromotions()
        select new XElement("Promotion",
            new XElement("Category", b.CategoryName),
            new XElement("Client", b.ClientName),
            new XElement("Title", b.Title))
    ).ToList()
);
0 голосов
/ 02 февраля 2010

Где вы определили mydoc? Могу поспорить, что это ноль.

...