当前位置:   article > 正文

lxml学习笔记

lxm文本转xml节点

问题1:有一个XML文件,如何解析
问题2:解析后,如果查找、定位某个标签
问题3:定位后如何操作标签,比如访问属性、文本内容等

from lxml import etree -> 导入模块,该库常用的XML处理功能都在lxml.etree中

requests + lxml解析小

  1. from lxml import etree  
  2. import requests  
  3.   
  4. page = 1  
  5. url = 'http://www.jikexueyuan.com/course/?pageNum=' + str(page)  
  6. html = requests.get(url)  
  7.   
  8. selector = etree.HTML(html.text)  
  9. content_field = selector.xpath('//div[@class="lesson-list"]/ul/li')  
  10. print(content_field)
  11. 复制代码

Element类

Element是XML处理的核心类,Element对象可以直观的理解为XML的节点,大部分XML节点的处理都是围绕该类进行的。这部分包括三个内容:节点的操作、节点属性的操作、节点内文本的操作。

1. 节点操作
  1. 创建Element对象
    使用Element方法,参数即节点名称。
  1. >>> root = etree.Element('root')
  2. >>> print(root)
  3. <Element root at 0x2da0708>
  4. 复制代码
  1. 获取节点名称
    使用tag属性,获取节点的名称。
  1. >>> print(root.tag)
  2. root
  3. 复制代码
  1. 输出XML内容
    使用tostring方法输出XML内容,参数为Element对象。
  1. >>> print(etree.tostring(root))
  2. b'<root><child1/><child2/><child3/></root>'
  3. 复制代码
  1. 添加子节点
    使用SubElement方法创建子节点,第一个参数为父节点(Element对象),第二个参数为子节点名称。
  1. >>> child1 = etree.SubElement(root, 'child1')
  2. >>> child2 = etree.SubElement(root, 'child2')
  3. >>> child3 = etree.SubElement(root, 'child3')
  4. 复制代码
  1. 删除子节点
    使用remove方法删除指定节点,参数为Element对象。clear方法清空所有节点。
  1. >>> root.remove(child1) # 删除指定子节点
  2. >>> print(etree.tostring(root))
  3. b'<root><child2/><child3/></root>'
  4. >>> root.clear() # 清除所有子节点
  5. >>> print(etree.tostring(root))
  6. b'<root/>'
  7. 复制代码
  1. 以列表的方式操作子节点
    可以将Element对象的子节点视为列表进行各种操作:

    1. >>> child = root[0] # 下标访问
    2. >>> print(child.tag)
    3. child1
    4. >>> print(len(root)) # 子节点数量
    5. 3
    6. >>> root.index(child2) # 获取索引号
    7. 1
    8. >>> for child in root: # 遍历
    9. ... print(child.tag)
    10. child1
    11. child2
    12. child3
    13. >>> root.insert(0, etree.Element('child0')) # 插入
    14. >>> start = root[:1] # 切片
    15. >>> end = root[-1:]
    16. >>> print(start[0].tag)
    17. child0
    18. >>> print(end[0].tag)
    19. child3
    20. >>> root.append( etree.Element('child4') ) # 尾部添加
    21. >>> print(etree.tostring(root))
    22. b'<root><child0/><child1/><child2/><child3/><child4/></root>'
    23. 复制代码
  2. 获取父节点
    使用getparent方法可以获取父节点。

  1. >>> print(child1.getparent().tag)
  2. root
  3. 复制代码



属性操作

  1. 属性是以key-value的方式存储的,就像字典一样。
  2. 复制代码
1. 创建属性

可以在创建Element对象时同步创建属性,第二个参数即为属性名和属性值:

  1. >>> root = etree.Element('root', interesting='totally')
  2. >>> print(etree.tostring(root))
  3. b'<root interesting="totally"/>'
  4. 也可以使用set方法给已有的Element对象添加属性,两个参数分别为属性名和属性值:
  5. >>> root.set('hello', 'Huhu')
  6. >>> print(etree.tostring(root))
  7. b'<root interesting="totally" hello="Huhu"/>'
  8. 复制代码
2. 获取属性

属性是以key-value的方式存储的,就像字典一样。直接看例子

  1. # get方法获得某一个属性值
  2. >>> print(root.get('interesting'))
  3. totally
  4. # keys方法获取所有的属性名
  5. >>> sorted(root.keys())
  6. ['hello', 'interesting']
  7. # items方法获取所有的键值对
  8. >>> for name, value in sorted(root.items()):
  9. ... print('%s = %r' % (name, value))
  10. hello = 'Huhu'
  11. interesting = 'totally'
  12. 复制代码

也可以用attrib属性一次拿到所有的属性及属性值存于字典中:

  1. >>> attributes = root.attrib
  2. >>> print(attributes)
  3. {'interesting': 'totally', 'hello': 'Huhu'}
  4. >>> attributes['good'] = 'Bye' # 字典的修改影响节点
  5. >>> print(root.get('good'))
  6. Bye
  7. 复制代码



文本操作

  1. 标签及标签的属性操作介绍完了,最后就剩下标签内的文本了。可以使用text和tail属性、或XPath的方式来访问文本内容。
  2. 复制代码
1. text和tail属性

一般情况,可以用Element的text属性访问标签的文本。

  1. >>> root = etree.Element('root')
  2. >>> root.text = 'Hello, World!'
  3. >>> print(root.text)
  4. Hello, World!
  5. >>> print(etree.tostring(root))
  6. b'<root>Hello, World!</root>'```
  7. XML的标签一般是成对出现的,有开有关,但像HTML则可能出现单一的标签,如下面这段代码中的`<br/>`。
  8. `<html><body>Text<br/>Tail</body></html>`
  9. Element类提供了tail属性支持单一标签的文本获取。
  10. ```python
  11. >>> html = etree.Element('html')
  12. >>> body = etree.SubElement(html, 'body')
  13. >>> body.text = 'Text'
  14. >>> print(etree.tostring(html))
  15. b'<html><body>Text</body></html>'
  16. >>> br = etree.SubElement(body, 'br')
  17. >>> print(etree.tostring(html))
  18. b'<html><body>Text<br/></body></html>'
  19. # tail仅在该标签后面追加文本
  20. >>> br.tail = 'Tail'
  21. >>> print(etree.tostring(br))
  22. b'<br/>Tail'
  23. >>> print(etree.tostring(html))
  24. b'<html><body>Text<br/>Tail</body></html>'
  25. # tostring方法增加method参数,过滤单一标签,输出全部文本
  26. >>> print(etree.tostring(html, method='text'))
  27. b'TextTail'
  28. 复制代码
2. XPath方式
  1. # 方式一:过滤单一标签,返回文本
  2. >>> print(html.xpath('string()'))
  3. TextTail
  4. # 方式二:返回列表,以单一标签为分隔
  5. >>> print(html.xpath('//text()'))
  6. ['Text', 'Tail']
  7. 复制代码

方法二获得的列表,每个元素都会带上它所属节点及文本类型信息,如下:

  1. >>> texts = html.xpath('//text()'))
  2. >>> print(texts[0])
  3. Text
  4. # 所属节点
  5. >>> parent = texts[0].getparent()
  6. >>> print(parent.tag)
  7. body
  8. >>> print(texts[1], texts[1].getparent().tag)
  9. Tail br
  10. # 文本类型:是普通文本还是tail文本
  11. >>> print(texts[0].is_text)
  12. True
  13. >>> print(texts[1].is_text)
  14. False
  15. >>> print(texts[1].is_tail)
  16. True
  17. 复制代码



文件解析与输出

这部分讲述如何将XML文件解析为Element对象,以及如何将Element对象输出为XML文件。

1. 文件解析

文件解析常用的有fromstring、XML和HTML三个方法。接受的参数都是字符串。

  1. >>> xml_data = '<root>data</root>'
  2. # fromstring方法
  3. >>> root1 = etree.fromstring(xml_data)
  4. >>> print(root1.tag)
  5. root
  6. >>> print(etree.tostring(root1))
  7. b'<root>data</root>'
  8. # XML方法,与fromstring方法基本一样
  9. >>> root2 = etree.XML(xml_data)
  10. >>> print(root2.tag)
  11. root
  12. >>> print(etree.tostring(root2))
  13. b'<root>data</root>'
  14. # HTML方法,如果没有<html>和<body>标签,会自动补上
  15. >>> root3 = etree.HTML(xml_data)
  16. >>> print(root3.tag)
  17. html
  18. >>> print(etree.tostring(root3))
  19. b'<html><body><root>data</root></body></html>'
  20. 复制代码
2. 输出

输出其实就是前面一直在用的tostring方法了,这里补充xml_declaration和encoding两个参数,前者是XML声明,后者是指定编码。

  1. >>> root = etree.XML('<root><a><b/></a></root>')
  2. >>> print(etree.tostring(root))
  3. b'<root><a><b/></a></root>'
  4. # XML声明
  5. >>> print(etree.tostring(root, xml_declaration=True))
  6. b"<?xml version='1.0' encoding='ASCII'?>\n<root><a><b/></a></root>"
  7. # 指定编码
  8. >>> print(etree.tostring(root, encoding='iso-8859-1'))
  9. b"<?xml version='1.0' encoding='iso-8859-1'?>\n<root><a><b/></a></root>"
  10. 复制代码



ElementPath

讲ElementPath前,需要引入ElementTree类,一个ElementTree对象可理解为一个完整的XML树,每个节点都是一个Element对象。而ElementPath则相当于XML中的XPath。用于搜索和定位Element元素。

这里介绍两个常用方法,可以满足大部分搜索、查询需求,它们的参数都是XPath语句: findall():返回所有匹配的元素,返回列表
find():返回匹配到的第一个元素

  1. >>> root = etree.XML("<root><a x='123'>aText<b/><c/><b/></a></root>")
  2. # 查找第一个b标签
  3. >>> print(root.find('b'))
  4. None
  5. >>> print(root.find('a').tag)
  6. a
  7. # 查找所有b标签,返回Element对象组成的列表
  8. >>> [ b.tag for b in root.findall('.//b') ]
  9. ['b', 'b']
  10. # 根据属性查询
  11. >>> print(root.findall('.//a[@x]')[0].tag)
  12. a
  13. >>> print(root.findall('.//a[@y]'))
  14. []
  15. 复制代码

原文地址:

Python lxml教程-SKYue

参考资料

  1. 用lxml解析HTML
  2. Python中利用xpath解析HTML
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家小花儿/article/detail/591598
推荐阅读
相关标签
  

闽ICP备14008679号