Передать экземпляр QXmlStreamReader в класс

Я пытаюсь прочитать документ XML с помощью QXmlStreamReader. Я сталкиваюсь с проблемами, когда я пытаюсь передать указатель на читателя другим классам. Я создаю экземпляр читателя в одном классе. Этот класс читает XML, пока не дойдет до блока, определяющего новый класс. Я создаю экземпляр нового класса и затем вызываю функцию в этом классе, чтобы продолжить чтение специфичного для него XML. Например:

void SF_UnitClass::ReadModes()
{
    Q_ASSERT(XML.isStartElement() && XML.name() == MODES);
    NumModes = XML.attributes().value(COUNT).toInt();            
    while (XML.readNextStartElement())                                          
    {
        if (XML.name() == MODE)                                            
        {                                                                      
            ModeClass* pMode = new ModeClass(this);                          
            ModeList += XML.attributes().value(ID).toString();                
            pMode->ReadXML(&XML);                                              
            {                                                                  
            }
        }
        else                                                                    
        {
            XML.raiseError(QObject::tr("Something other than Mode block encountered in Modes block"));
        }
    }
}

void ModeClass::ReadXML(QXmlStreamReader* pXML)                                 
{                                                                               
    Q_ASSERT(pXML->isStartElement() && pXML->name() == MODE);                   
    while (pXML->readNextStartElement())                                        
    {                                                                           
        int nameIndex = ModeList.indexOf(pXML->name().toString());            
        if(nameIndex != -1)                                                     
        {                                                                       
            switch(nameIndex)                                                   
            {                                                                   
                case 0:                                                         
                {                                                               
    qDebug() << "Mode: Receivers";                                              
                    NumReceivers = pXML->readElementText().toInt();             
                    break;                                                      
                }                                                               
                case 1:                                                         
                {                                                               
    qDebug() << "Mode: Channels";                                               
                    ReadChannels(pXML);                                         
                    break;                                                      
                }                                                               
                case 2:                                                         
                {                                                               
    qDebug() << "Mode: Servos";                                                 
                    ReadServos(pXML);                                           
                    break;                                                      
                }                                                               
            }                                                                   
        }                                                                       
        else                                                                    
        {                                                                       
    qDebug() << "Mode: Error " << name;                                         
            pXML->raiseError(QObject::tr("Unrecognized keyword for Mode"));     
        }                                                                       
    }                                                                           
}                                                                               

void ModeClass::ReadChannels(QXmlStreamReader* pXML)                            
{                                                                               
    Q_ASSERT(pXML->isStartElement() && pXML->name() == CHANNELS);               
    NumChannels = pXML->attributes().value(COUNT).toInt();                      
    while (pXML->readNextStartElement())                                        
    {                                                                           
        if (pXML->name() == CHANNEL)                                            
        {                                                                       
    qDebug() << "Mode: Read Channel";                                           
            ChannelClass* pChannel = new ChannelClass(this);                    
            pChannel->ReadXML(pXML);                                            
        }                                                                       
        else                                                                    
        {                                                                       
            pXML->raiseError(QObject::tr("Something other than Channel block enc
        }                                                                       
    }                                                                           
}                                                                               

void ModeClass::ReadServos(QXmlStreamReader* pXML)                              
{                                                                               
    Q_ASSERT(pXML->isStartElement() && pXML->name() == SERVOS);                 
    NumServos = pXML->attributes().value(COUNT).toInt();                        
    while (pXML->readNextStartElement())                                        
    {                                                                           
        if (pXML->name() == SERVO)                                              
        {                                                                       
    qDebug() << "Mode: Read Servo";                                             
            ServoClass* pServo = new ServoClass(this);                          
            pServo->ReadXML(pXML);                                              
        }                                                                       
        else                                                                    
        {                                                                       
    qDebug() << "Servos: raiseError";                                           
            pXML->raiseError(QObject::tr("Something other than Servo block encou
        }                                                                       
    }                                                                           
    qDebug() << "Mode: Read Servos Exit";                                       
}                                                                               

void ChannelClass::ReadXML(QXmlStreamReader* pXML)                              
{                                                                               
    Q_ASSERT(pXML->isStartElement() && pXML->name() == CHANNEL);                
    ChannelNumber = pXML->attributes().value(ID).toInt();                       
    while (pXML->readNextStartElement())                                        
    {                                                                           
        int nameIndex = ChannelList.indexOf(pXML->name().toString());           
        if(nameIndex != -1)                                                     
        {                                                                       
            switch(nameIndex)                                                   
            {                                                                   
                case 0:                                                         
                {                                                               
                    UserName = pXML->readElementText();                         
                    break;                                                      
                }                                                               
                case 1:                                                         
                {                                                               
                    EndPointHold = String2Bool(pXML->readElementText());        
                    break;                                                      
                }                                                               
                case 2:                                                         
                {                                                               
                    ServoPriority = String2Bool(pXML->readElementText());       
                    break;                                                      
                }                                                               
            }                                                                   
        }                                                                       
        else                                                                    
        {                                                                       
            pXML->raiseError(QObject::tr("Unrecognized keyword for Channel"));  
        }                                                                       
    }                                                                           
}                                                                               

void ServoClass::ReadXML(QXmlStreamReader* pXML)                                
{                                                                               
    Q_ASSERT(pXML->isStartElement() && pXML->name() == SERVO);                  
    ServoNumber = pXML->attributes().value(ID).toInt();                         
    while (pXML->readNextStartElement())                                        
    {                                                                           
        int nameIndex = ServoList.indexOf(pXML->name().toString());             
    qDebug() << "Servo: NameIndex: " << nameIndex;                              

        if(nameIndex != -1)                                                     
        {                                                                       
            switch(nameIndex)                                                   
            {                                                                   
                case 0:                                                         
                {                                                               
                    Offset = pXML->readElementText().toInt();                   
                    break;                                                      
                }                                                               
                case 1:                                                         
                {                                                               
                    PosFactor = pXML->readElementText().toInt();                
                    break;                                                      
                }                                                               
                case 2:                                                         
                {                                                               
                    NegFactor = pXML->readElementText().toInt();                
                    break;                                                      
                }                                                               
                case 3:                                                         
                {                                                               
                    SecServo = pXML->readElementText().toInt();                 
                    break;                                                      
                }                                                               
                case 4:                                                         
                {                                                               
                    Unit = pXML->readElementText().toInt();                     
                    break;                                                      
                }                                                               
                case 5:                                                         
                {                                                               
                    Ser1 = pXML->readElementText().toInt();                     
                    break;                                                      
                }                                                               
                case 6:                                                         
                {                                                               
                    Ser2 = pXML->readElementText().toInt();                     
                    break;                                                      
                }                                                               
                case 7:                                                         
                {                                                               
                    Ser3 = pXML->readElementText().toInt();                     
                    break;                                                      
                }                                                               
            }                                                                   
        }                                                                       
        else                                                                    
        {                                                                       
    qDebug() << "Servo: raiseError";                                            
            pXML->raiseError(QObject::tr("Unrecognized keyword for Servo block")
        }                                                                       
    }                                                                           
    qDebug() << "Servo: Exit ";                                                 
}                                                                               

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE xml>
<SmartFlyUnit version="1.0">
    <UnitName>PowerExpander Eq10E</UnitName>
    <UnitCode>PE-5</UnitCode>
    <UnitID>1</UnitID>
    <MinRev>1.0</MinRev>
    <MaxRev>1.9</MaxRev>
    <Servos>36</Servos>
    <Outputs>32</Outputs>
    <UnitMode>Chan</UnitMode>
    <Modes Count="2">
        <Mode ID="Chan">
            <Receivers>1</Receivers>
            <Channels Count="10">
                <Channel ID="0">
                    <UserName></UserName>
                    <EndPtHold>Off</EndPtHold>
                    <ServoPri>Off</ServoPri>
                </Channel>
                <Channel ID="1">
                    <UserName></UserName>
                    <EndPtHold>Off</EndPtHold>
                    <ServoPri>Off</ServoPri>
                </Channel>
            </Channels>
            <Servos Count="36">
                <Servo ID="0">
                    <Offset>0</Offset>
                    <PosFact>1.0000</PosFact>
                    <NegFact>1.0000</NegFact>
                    <Reverse>Off</Reverse>
                    <SecServo>0xFF</SecServo>
                    <Unit>0xFF</Unit>
                    <Ser1>0xFF</Ser1>
                    <Ser2>0xFF</Ser2>
                    <Ser3>0xFF</Ser3>
                </Servo>
                <Servo ID="1">
                    <Offset>0</Offset>
                    <PosFact>1.0000</PosFact>
                    <NegFact>1.0000</NegFact>
                    <Reverse>Off</Reverse>
                    <SecServo>0xFF</SecServo>
                    <Unit>0xFF</Unit>
                    <Ser1>0xFF</Ser1>
                    <Ser2>0xFF</Ser2>
                    <Ser3>0xFF</Ser3>
                </Servo>
            </Servos>
        </Mode>
    </Modes>
    <MCUs Count="1">
        <MCU ID="0">
            <FileName>/PE-5/PE-5_M0_1_00.sffw</FileName>
        </MCU>
    </MCUs>
    <FPGAs Count="1">
        <FPGA ID="0">
            <Configs Count="2">
                <Config ID="0">
                    <FileName>/PE-5/PE-5_F0C0_1_00.sffw</FileName>
                </Config>
                <Config ID="1">
                    <FileName>/PE-5/PE-5_F0C1_1_00.sffw</FileName>
                </Config>
            </Configs>
        </FPGA>
    </FPGAs>
</SmartFlyUnit>

Я добавил код, который ModeClass использует для распознавания канала и сервоблока. Я также добавил код из ChannelClass и ServoClass. Наконец, я добавил отредактированную версию своего XML-файла, но достаточно, чтобы показать проблему. Этот XML отлично читается в моем коде TreeWidget (не показан). Что происходит, когда я читаю его в приведенном выше коде, так это то, что все каналы читаются правильно, затем он читает первый сервоблок и возвращается полностью к основному циклу ModeClass вместо чтения следующего сервопривода в ModeClass::ReadServos. Я знаю, что это много кода, но это был единственный способ показать проблему. Спасибо,

1 ответ

Qt документы говорят:

Читает до следующего начального элемента в текущем элементе. Возвращает true при достижении начального элемента. Когда достигнут конечный элемент или произошла ошибка, возвращается false.

Поэтому, когда вы сталкиваетесь с начальным элементом, используя readNextStartElement() будет искать внутри элемента, который вы нашли

т.е.

<xml>
</xml>
<xml>
</xml>

Что будет здесь:

  1. readNextStartElement() остановится на первом <xml> узел
  2. readNextStartElement() вернет false, как он находит </xml>

в немного измененном примере:

<xml>
  <child>
  </child>
</xml>
<xml>
</xml>
  1. readNextStartElement() остановится на первом <xml> узел
  2. readNextStartElement() остановится на <child> узел
  3. readNextStartElement() вернет false, так как внутри нет узла XML <child> узел

Что вы должны сделать, спросите вы?

после того как вы доберетесь до узла, в котором вы заинтересованы, и прочитав всю необходимую информацию, вы должны позвонить skipCurrentElement() так QXmlStreamReader будет идти до конца текущего элемента

Итак, немного изменив первый пример:

<xml>
</xml>
<xml>
</xml>
  1. readNextStartElement() остановится на первом <xml> узел
  2. skipCurrentElement() остановится при закрытии первого <xml> узел
  3. readNextStartElement() остановится на втором <xml> узел
Другие вопросы по тегам