Вывод XML в неправильном порядке - PullRequest
5 голосов
/ 08 февраля 2012

Я написал простой веб-сервис WCF на C #, который возвращает записи из базы данных.

WCF использует следующий метод: getQuestionnaireForID?id=(questionnaireID).Веб-служба возвращает все правильные записи из базы данных, однако они отображаются в неправильном порядке.

В таком порядке я бы хотел, чтобы XML был:

<QuestionnaireXML xmlns="http://schemas.datacontract.org/2004/07/QuestionnaireDescriptor" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
    <QuestionnaireName>Sample Questionnaire</QuestionnaireName>
        <Questions>
            <Question>
                <QuestionID>297</QuestionID>
                <QuestionName>What is your favorite type of tree?</QuestionName>
                <Answers>
                    <Answer>
                        <AnswerTitle>Beech</AnswerTitle>
                    </Answer>
                    <Answer>
                        <AnswerTitle>Oak</AnswerTitle>
                    </Answer>
                </Answers>
            </Question>
            <Question>
                <QuestionID>298</QuestionID>
                <QuestionName>Windows or Mac?</QuestionName>
                <Answers>
                    <Answer>
                        <AnswerTitle>Mac</AnswerTitle>
                    </Answer>
                    <Answer>
                        <AnswerTitle>Windows</AnswerTitle>
                    </Answer>
                </Answers>
            </Question>
        </Questions>
</QuestionnaireXML>

Новместо этого в настоящее время возвращается в следующем порядке:

<QuestionnaireXML xmlns="http://schemas.datacontract.org/2004/07/QuestionnaireDescriptor" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
    <QuestionnaireName>Hello sir how do you do today?</QuestionnaireName>
    <Questions>
        <Question>
            <Answers>
                <Answer>
                    <AnswerTitle>Beech</AnswerTitle>
                </Answer>
                <Answer>
                    <AnswerTitle>Oak</AnswerTitle>
                </Answer>
            </Answers>
            <QuestionID>297</QuestionID>
            <QuestionName>What is your favorite type of tree?</QuestionName>
        </Question>
        <Question>
            <Answers>
                <Answer>
                    <AnswerTitle>Mac</AnswerTitle>
                </Answer>
                <Answer>
                    <AnswerTitle>Windows</AnswerTitle>
                </Answer>
            </Answers>
            <QuestionID>298</QuestionID>
            <QuestionName>Windows or Mac?</QuestionName>
        </Question>
    </Questions>
</QuestionnaireXML>

Я не уверен, связано ли это с неустановленным XML-документом или потому, что у меня циклы в неправильном порядке.Вот мой xml-дескриптор WCF.

[DataContract]
    public class QuestionnaireXML
    {
        OsqarSQL getData;
        DataTable DT;
        private String _questionnaireName;
        private List<Question> _Question = new List<Question>();

        public QuestionnaireXML(int QuestionnaireID)
        {
            // Loop through datatable for questionIDs and load the questions
            getData = new OsqarSQL();
            _questionnaireName = getData.GetQuestionnaireName(QuestionnaireID);
            DT = getData.GetQuestionIDWCF(QuestionnaireID);
            for (int i = 0; i < DT.Rows.Count; i++)
            {
                _Question.Add(new Question(Int32.Parse(DT.Rows[i][0].ToString())));
            }
        }

        // Define DataMembers for XML output
        [DataMember]
        public String QuestionnaireName
        {
            get { return _questionnaireName; }
            set { _questionnaireName = value; }
        }

        [DataMember]
        public List<Question> Questions
        {
            get { return _Question; }
            set { _Question = value; }
        }
    }

    [DataContract]
    public class Question
    {
        private Int32 _questionId;
        private String _questionName;
        OsqarSQL getData;
        DataTable DT;
        DataTable AT;
        private List<Answer> _Answer = new List<Answer>();

        public Question(int QuestionID)
        {
            getData = new OsqarSQL();
            DT = getData.GetQuestionNameWCF(QuestionID);
            _questionId = (int)DT.Rows[0][0];
            _questionName = DT.Rows[0][1].ToString();
            AT = getData.GetAnswerTitle(QuestionID);
            for (int i = 0; i < AT.Rows.Count; i++)
            {
                _Answer.Add(new Answer(Int32.Parse(AT.Rows[i][0].ToString())));
            }
        }

        // Define DataMembers for XML output
        [DataMember]
        public Int32 QuestionID
        {
            get { return _questionId; }
            set { _questionId = value; }
        }

        [DataMember]
        public String QuestionName
        {
            get { return _questionName; }
            set { _questionName = value; }
        }

        [DataMember]
        public List<Answer> Answers
        {
            get { return _Answer; }
            set { _Answer = value; }
        }
    }

    [DataContract]
    public class Answer
    {
        private Int32 _answerId;
        private String _answerTitle;
        OsqarSQL getData;
        DataTable DT;

        // Constructor Logic
        public Answer(int AnswerID)
        {
            getData = new OsqarSQL();
            DT = getData.GetAnswerTitleWcf(AnswerID);
            _answerTitle = DT.Rows[0][1].ToString();   
        }

        // Define DataMembers for XML output
        [DataMember]
        public String AnswerTitle
        {
            get { return _answerTitle; }
            set { _answerTitle = value; }
        }
    }

Как я могу решить эту проблему?Вызывает ли это проблемы при попытке анализа XML?

Ответы [ 3 ]

5 голосов
/ 08 февраля 2012

В таком порядке я бы хотел, чтобы XML был:

Почему? Это необходимо? Держу пари, что это почти наверняка не нужно.

WCF DataContractSerializer сериализует - по умолчанию - в алфавитном порядке. Вы заметите, что у XML-файла, который вы видите, есть узлы, упорядоченные в алфавитном порядке. Неважно, в каком порядке вы перечисляете свойства вашего класса.

При необходимости можно указать порядок элементов, используя атрибут Order (см. Порядок элементов данных MSDN ).

Другой вариант - использовать XMLSerializer, который более гибкий, но не такой простой, как DataContractSerializer.

Если не будет подлинной необходимости , чтобы переупорядочить эти элементы, то я бы не стал этим заниматься.

0 голосов
/ 08 февраля 2012

Я не вижу проблем с результатом, который вы получаете.Одним из преимуществ использования XML над, скажем, записями переменной или фиксированной длины является то, что ваши данные содержат достаточно метаданных, что позволяет синтаксическому анализу быть устойчивым к полям в неправильном порядке или любым другим типам структуры.Я не уверен, есть ли парсеры, которые обеспечивают такое поведение, но они не должны.Порядок элементов не важен в XML.

0 голосов
/ 08 февраля 2012

Это заданное и задокументированное поведение:

В предыдущей таблице перечислены интерфейсы сбора в порядке убывания приоритета.Это означает, например, что если тип реализует как IList, так и Generic IEnumerable, коллекция сериализуется и десериализуется в соответствии с правилами IList:

При десериализации все коллекции десериализуются, сначала создав экземпляр типавызывая конструктор по умолчанию, который должен присутствовать в сериализаторе для обработки типа коллекции как коллекции во время сериализации и десериализации.

Посмотрите раздел «Расширенные правила коллекции» коллекции Типы в контрактах данных документация для всех применяемых правил и для возможных объяснений ПОЧЕМУ они существуют.

...