在实现文件操作的时候我意识到了一个事实:由于nsurl是可支持的路径,很多文件的操作实际上可以扩充到网络上去,比如从网上下载一个文件、拿取一些数据……想到这里,埋藏在心中很久的愿望又开始蠢蠢欲动:写一个新闻客户端!
我有一个常看的新闻网站(不要问我是哪个,要fq),官方客户端很难用,第三方广告满天飞,严重影响心情。所以当初开始学ios的时候就把重做一个客户端的目标放在了首位。最开始的时候想得太简单,兴冲冲地开了个project,搞了两下tableview,然后发现……对于我这种从来只写底层代码,swift/oc语法都不会的人,做这个简直是异想天开好嘛!
折腾的过程中,为了拿到数据,还跑去下了android的官方客户端反编译,find grep出来了人家的数据接口地址(论一个爱折腾的程序员是怎么给自己挖坑的),返回的是xml数据,连蒙带猜试了试,可以用!不过后来做ui受挫(其实不止做ui,挫败感太大),就把那份东西闲置起来了。这两天突然想起来,现在我貌似感觉有点良好,何不捡起来继续写我的客户端?
因为ui我还一窍不通,不懂怎么调试,也不懂怎么把数据显示出来,为了避免到处打印的麻烦,我选择playground。这样也就是需要用原生库,这个很简单,搜一下就有了:nsxmlparserdelegate,还有好些源码,虽然大多是oc的。
问题是有源码,我依!然!看!不!懂!怎么用!每个教程都是扔了源码上来,可没人解释清楚到底怎么运行。我研究了好一阵,终于明白了大概的机制:
引用
自己写一个类继承nsxmlparserdelegate,就叫它xmlreader吧;并且实现一些必须的callback函数,在xmlreader的init中调用.parse()方法,最后在new xmlreader的时候就会自动完成整个parse了
当然你也可以不把parse()放在init()中调用,而是在new了xmlreader之后强制调用,重要的是那些callback函数的实现。xmlreader中还可以声明一些变量,来保存解析过程中的状态,而具体的获得到的数据填充,也是需要你自己做的(数据结构自行定义)。
鉴于xml是一个有层级的带递归意味的数据结构,程序会层层解析下去直到结束,中间遇到的各种类似section开始、结束、拿到字符串等情况都会call一个固定的函数,将解析出来的数据作为参数传进去,而你要做的,就是实现这些callback以完成自己想要做的事。
下面是具体的代码示范。首先是定义一个新的类:
class xmlreader: nsobject, nsxmlparserdelegate {
var currentname :string? = nil
var level :int = 0
init(add :string ) {
super.init()
let url = ns!
guard let parserxml = nsxmlparser(contentsofurl: url) else {
return
}
parserxml.delegate = self
parserxml.parse()
}
func parserdidstartdocument(parser: nsxmlparser) {}
func parserdidenddocument(parser: nsxmlparser){}
func parser(parser: nsxmlparser, didstartelement elementname: string, namespaceuri: string?, qualifiedname qname: string?, attributes attributedict: [string : string]) {}
func parser(parser: nsxmlparser, didendelement elementname: string, namespaceuri: string?, qualifiedname qname: string?) {}
func parser(parser: nsxmlparser, var foundcharacters string: string) {}
func parser(parser: nsxmlparser, parseerroroccurred parseerror: nserror) {}
}
这是一个基础部分,里面定义了一些callback但没有实现它。我把xml的地址作为参数传给了init(),并且声明了两个变量currentname和level分别用来保存当前的标签名字和递归深度。然后我们基于下面这个简单的xml文件继续完成callback部分:
c#
vs
首先是文档开始和结束时:
//文档开始解析时触发,只触发一次
func parserdidstartdocument(parser: nsxmlparser) {
print(" start")
}
//文档结束时触发,只触发一次,通常需要在这里给出一个信号告诉上层或其他人解析已经结束
func parserdidenddocument(parser: nsxmlparser){
print(" end")
}
然后是标签的开始和结束事件:
//遇到一个开始标签触发,elementname为当前标签,如果当前标签有属性,则字典sttributedict不为空
func parser(parser: nsxmlparser, didstartelement elementname: string, namespaceuri: string?, qualifiedname qname: string?, attributes attributedict: [string : string]()) {
self.currentname = elementname
self.level
print("\(level) start, \(elementname), \(attributedict)")
if currentname == "language" { // 获取language属性的内容
//print("language: \(attributedict)")
}
}
//遇到结束标签触发,该方法主要是做一些清理工作,在这里我修改了当前的深度
func parser(parser: nsxmlparser, didendelement elementname: string, namespaceuri: string?, qualifiedname qname: string?) {
print("\(level) end, \(elementname)")
self.currentname = nil
self.level--
}
接下来是字符串值的handler
// 遇到字符串时触发
func parser(parser: nsxmlparser, var foundcharacters string: string) {
//删除首尾的回车符和空格
string = string.stringbytrimmingcharactersinset(nscharacterset.whitespaceandnewlinecharacterset())
if string.isempty {
return
}
print("---- \(string)")
}
最后是error handler
// 文档出错时触发
func parser(parser: nsxmlparser, parseerroroccurred parseerror: nserror) {
print(parseerror)
}
以上的callback函数实现都只放了些打印信息,在实际操作中需要把这些获取的信息一一填充进自己的数据结构中。说实话是挺麻烦的,难怪都说xml快要被json全面取代(其实对于曾被libxml折磨过的码农来说,这已经挺好了)。全部的代码传送门:
本文转自:
1 楼 2015-12-18 10:15