Разбор иерархического XML в Swift с использованием NSXMLParser

У меня действительно возникают проблемы с возвращением иерархических значений XML в форме, которую я действительно могу использовать, поэтому любая помощь будет принята с благодарностью. Я довольно новичок в разработке Swift и IOS, поэтому, если честно, я не до конца понимаю парсер, но надеюсь, что после этого я это сделаю!

Вот пример XML, который я пытаюсь проанализировать, он возвращается из мыльного веб-сервиса. Вся эта сторона с точки зрения подключения и получения данных работает отлично

<s:Envelope
    xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
    <s:Body>
        <get_Entry_DetailsResponse
            xmlns="http://tempuri.org/">
            <get_Entry_DetailsResult>
                <ApiResult
                    xmlns="">
                    <Status>Success</Status>
                    <Parameters>
                        <TotalRecords>1</TotalRecords>
                        <Records>
                            <Entry>
                                <Entry_ID>1234</Entry_ID>
                                <Title>This is the Title</Title>
                                <Description>This is a description.</Description>
                                <Charge_USD>3</Charge_USD>
                                <Charge_GBP>1.5</Charge_GBP>
                                <Classifications>
                                    <Classification_Type>No. Of Colours</Classification_Type>
                                    <Description>10</Description>
                                    <Classification_Type>Height</Classification_Type>
                                    <Description>16.712</Description>
                                    <Classification_Type>Width</Classification_Type>
                                    <Description>1.485</Description>
                                    <Classification_Type>Width Count</Classification_Type>
                                    <Description>11</Description>
                                    <Classification_Type>Pages</Classification_Type>
                                    <Description>6</Description>
                                    <Classification_Type>Type</Classification_Type>
                                    <Description>Type Description</Description>
                                    <Classification_Type>Topic</Classification_Type>
                                    <Description>Transport</Description>
                                    <Classification_Type>Item</Classification_Type>
                                    <Description>Shop Item</Description>
                                    <Classification_Type>Material</Classification_Type>
                                    <Description>Metal</Description>
                                    <Classification_Type>Manufacturer</Classification_Type>
                                    <Description>Unknown</Description>
                                </Classifications>
                            </Entry>
                        </Records>
                    </Parameters>
                </ApiResult>
            </get_Entry_DetailsResult>
        </get_Entry_DetailsResponse>
    </s:Body>
</s:Envelope>

Таким образом, я могу получить такие элементы, как Entry_ID, Title, Description, Charge_GBP и Charge_USD. Это то, как вернуть классификации в код, чтобы я мог использовать его для вывода на страницу.

Итак, вот мой connectionDidFinishLoading, где я запускаю и запускаю парсер:

func connectionDidFinishLoading(connection: NSURLConnection!) {
    var xmlParser = NSXMLParser(data: mutableData)
    xmlParser.delegate = self
    xmlParser.parse()
    xmlParser.shouldResolveExternalEntities = true
}

Тогда вот материал делегата XMLParser:

var mutableData:NSMutableData  = NSMutableData.alloc()
var currentElementName:NSString = ""
var foundCharacters = ""
var appParsedData = [Dictionary<String, String>]()
var currentDataDictionary = Dictionary<String, String>()

func parser(parser: NSXMLParser!, didStartElement elementName: String!, namespaceURI: String!, qualifiedName qName: String!, attributes attributeDict: NSDictionary!) {

        currentElementName = elementName

    }

func parser(parser: NSXMLParser, didEndElement elementName: String!, namespaceURI: String!, qualifiedName qName: String!) {

    if !foundCharacters.isEmpty {
        currentDataDictionary[currentElementName] = foundCharacters
        foundCharacters = ""

        //Last Element so add to main Dictionary
        //Cannot find last element yet on XML so use Charge_GBP for testing until fixed
        if currentElementName == "Charge_GBP" {
            appParsedData.append(currentDataDictionary)
        }
    }
}

func parser(parser: NSXMLParser, foundCharacters string: String!) {
    if (currentElementName == "Entry_ID") || (currentElementName == "Title") || currentElementName == "Description" || currentElementName == "Charge_USD" || currentElementName == "Charge_GBP" || currentElementName == "Classification_Type" {
        foundCharacters += string
    }
}

func parserDidEndDocument(parser: NSXMLParser!) {

    self.setupItemsOnView()

}

Затем моя функция setupItemsOnView, в которой я собираюсь получить значения и отобразить их на контроллере представления:

func setupItemsOnView() {

    for (currentElementName) in appParsedData {
        println("\(currentElementName):")
    }

    let test = appParsedData[0] as Dictionary<String, String>
    let test_entryid = test["Entry_ID"]
    println("Entry ID is \(test_entryid)")

}

Фу, хорошо, так что в основном я могу получить значения Entry_ID, Title, Description, Charge_GBP и Charge_USD в функции setupItemsOnView, но, используя этот метод, я не могу получить значения классификаций и описаний обратно значимым способом из родительского объекта классификации. В настоящее время я использую словарь для хранения данных, так что мне интересно, правильно ли использовать словарь, если у вас есть иерархический XML, но я понятия не имею, что еще можно попробовать.

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

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

ура

D

1 ответ

Решение

То, что вы хотели бы, это что-то сравнимое с jaxb для Swift - но я не знаю, что это существует. В любом случае, синтаксический анализ в Swift похож на использование SAX-парсера. Предположительно, в вашем коде Swift есть иерархия классов, которая соответствует иерархии элементов в XML. Подход, который я использовал, основывался на состоянии - просто создайте экземпляр класса, соответствующий анализируемому элементу, и сохраните его в переменной, предназначенной для этой цели. Затем ваш код синтаксического анализа может заполнить данные класса так, как они встречаются в последующих обратных вызовах. Конечно, вы должны сделать это иерархически, и использовать didEndElement обратный вызов для определения завершения.

Другие вопросы по тегам