笔者在公司项目中,需要解析一个200K左右的XML文件(有3000个XML结点),发现解析会导致整个App卡顿。于是猜想是不是在主线程中进行了解析,导致卡顿。因此,xml的代理方法didStartElement
,即以下方法,打印了当前线程,发现代理是异步执行的,就以为不是这个问题了。
func parser(_ parser: XMLParser, didStartElement elementName: String, namespaceURI: String?, qualifiedName qName: String?, attributes attributeDict: [String : String])
但是打断点调试发现,当xml解析完毕时,整个项目就正常运行了。走审代码后,发现parse()
方法是在主线程调用的!
开启异步线程调用parse()
OperationQueue().addOperation {
parser.parse()
}
PS:XMLParser是一个线程安全的类,后续代理执行的方法都会在同一个线程处理的。
虽然代理方法didStartElement
是在异步执行的,但parse()
却是在主线程调用的。因此可以猜想,parse()
调用后,会异步开启线程处理代理方法进行解析,但是主线程会继续等在解析的过程,导致了卡顿~所以parse()
也必须在异步线程开启,不然当解析的数据太大,将会导致App的卡顿。笔者项目解析220K的xml文件,需要25秒左右