swift 中的 alloc() 错误

alloc() error in swift

所以我刚刚将 swift 更新为 2.0,显然我的代码中出现了一些错误。 这就是我现在拥有的。

unc loadRssFeed(data: NSURL) {

    var myRssParser : ParserManager = ParserManager.alloc().initWithURL(data) as! ParserManager

    myRssFeed = myRssParser.feeds

    tableView.reloadData()
}

Fixit 建议将 'var' 更改为 'let' 并使用对象初始值设定项而不是 alloc()。但问题是我以前从未使用过它(假设我的 Swift 经验是四个星期)。

我应该遵循 fixit 说明吗? 以及如何解决 alloc() 问题。

谢谢!

更新机智解析器代码:

class ParserManager: NSObject, NSXMLParserDelegate {

var parser = NSXMLParser()
var feeds = NSMutableArray()
var elements = NSMutableDictionary()
var element = NSString()
var ftitle = NSMutableString()
var link = NSMutableString()
var fdescription = NSMutableString()
var fdate = NSMutableString()


func initWithURL(url :NSURL) -> AnyObject {
    startParse(url)
    return self
}
func startParse(url :NSURL) {
    feeds = []
    parser = NSXMLParser(contentsOfURL: url)!
    parser.delegate = self
    parser.shouldProcessNamespaces = false
    parser.shouldReportNamespacePrefixes = false
    parser.shouldResolveExternalEntities = false
    parser.parse()
}

func allFeeds() -> NSMutableArray {
    return feeds
}
func parser(parser: NSXMLParser, didStartElement elementName: String, namespaceURI: String?, qualifiedName qName: String?, attributes attributeDict: [String : String]) {

    self.element = elementName

    if self.element == "item" {

        self.ftitle = ""
        self.fdescription = ""

        self.fdate = ""

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

    if (elementName as NSString).isEqualToString("item") {
        if ftitle != "" {
            elements.setObject(ftitle, forKey: "title")
        }



        if fdescription != "" {
            elements.setObject(fdescription, forKey: "description")
        }

        if fdate != "" {
            elements.setObject(fdate, forKey: "pubDate")
        }

        feeds.addObject(elements)
    }

}
func parser(parser: NSXMLParser, foundCharacters string: String?) {

    if element.isEqualToString("title") {
        ftitle.appendString(string!)

    }else if element.isEqualToString("description") {
        fdescription.appendString(string!)
    }else if element.isEqualToString("pubDate") {
        fdate.appendString(string!)
    }
}


}

修复应该如下所示:

let myRssParser = ParserManager(URL: data) as! ParserManager

或者可能像这样,取决于 ParserManager 的编写方式:

let myRssParser = ParserManager(url: data) as! ParserManager

甚至有可能您不必使用参数名称:

let myRssParser = ParserManager(data) as! ParserManager

如上所述,您不应直接调用 allocinit... 方法。此外,ParserManager class 不应该有 initWithURL 方法,而是带有 URL 参数的 init 方法。现有代码似乎遵循 Objective-C 模式,但 Swift 有其自己的约定。

但更广泛地说,我建议不要执行同步网络请求,而当您调用 NSXMLParser(contentsOfURL:) 时,就会发生这种情况。最好异步请求数据,然后重构代码以遵循异步模式(例如 completionHandler 闭包)。例如:

class ParserManager: NSObject, NSXMLParserDelegate {

    /// The NSMutableArray used internally by this class

    private var feeds = NSMutableArray()

    /// Initiate parsing asynchronously; note, I'm returning the `NSURLSessionTask` in case you want to cancel it at some later point

    func parse(URL: NSURL, completionHandler: (NSMutableArray?, NSError?)->()) -> NSURLSessionTask {
        let task = NSURLSession.sharedSession().dataTaskWithURL(URL) { data, response, error in
            guard data != nil else {
                dispatch_async(dispatch_get_main_queue()) {
                    completionHandler(nil, error)
                }
                return
            }

            let parser = NSXMLParser(data: data!)
            parser.delegate = self

            if parser.parse() {
                dispatch_async(dispatch_get_main_queue()) {
                    completionHandler(self.feeds, nil)
                }
            } else {
                dispatch_async(dispatch_get_main_queue()) {
                    completionHandler(nil, parser.parserError)
                }
            }
        }
        task.resume()

        return task
    }

    // NSXMLParserDelegate methods implemented here

}

然后,您可以使用完成处理程序语法来指定在网络请求和解析完成后要执行的操作:

let URL = NSURL(string: "...")
let myRssParser = ParserManager()
myRssParser.parse(URL) { feeds, error in
    guard feeds != nil else {
        print(error)
        return
    }

    // use `feeds` here, e.g.

    self.feeds = feeds            // update your local property
    self.tableView.reloadData()   // and reload the table
}

// but don't use `feeds` here, since it won't be done by the time we get here