QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 3888|回复: 2
打印 上一主题 下一主题

Python爬虫常用库总结

[复制链接]
字体大小: 正常 放大
杨利霞        

5273

主题

82

听众

17万

积分

  • TA的每日心情
    开心
    2021-8-11 17:59
  • 签到天数: 17 天

    [LV.4]偶尔看看III

    网络挑战赛参赛者

    网络挑战赛参赛者

    自我介绍
    本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。

    群组2018美赛大象算法课程

    群组2018美赛护航培训课程

    群组2019年 数学中国站长建

    群组2019年数据分析师课程

    群组2018年大象老师国赛优

    跳转到指定楼层
    1#
    发表于 2021-7-8 14:55 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
      r9 v/ w" \% U$ o
    Python爬虫常用库总结8 X/ \' @* Z& A* p/ H7 _
    文章目录  K; e0 E/ Q  h
    requests
    : C( \, C/ w2 `/ e, T4 P, \( qrequests基础2 \; ~6 W+ {% g$ ~0 a: @, b9 W
    requests模块发送get请求
    3 M8 q& u* r0 w' [8 s$ Nresponse响应对象- ~3 {% t! G/ x$ ]8 a
    response.text 和response.content的区别) j# V# [' v( b. |: r
    解决中文乱码; |+ J# W& k8 W' J
    response响应对象的其它常用属性或方法
    3 N; r" |4 V9 l# N. grequests实操5 G; G8 E( G" g+ `" u) a* t' v# E6 v9 ], _
    requests模块发送请求
    : t( s0 J7 U4 h' ^5 ?发送带参数的请求" b4 h+ Y2 f( ~/ U' `/ V4 v
    超时参数timeout的使用
    / q( o: L8 {- n$ G7 Xrequests发送post请求的方法9 [  P4 t/ ?( i3 Z  f
    BeautifulSoup) I" `* ^/ `+ f0 Q: v0 z
    常见解释器的优缺点. _; O. [* X9 Z" a4 n* F
    常用操作# l$ ^6 `2 v! w9 u& m
    几个简单的浏览结构化数据的方法
    / X$ e; m$ f' K0 y. N从文档中找到所有的< a>标签的链接; d# ^/ S6 s! A" S+ _
    在文档中获取所有的文字内容! q' r+ b3 }) i* ^9 {8 a
    通过标签和属性获取% w: y3 k8 z, ?) z' r7 w5 D
    Name属性* U( c2 w$ B0 x; j  r
    多个属性
    ! |! T# q' q; y( S8 v; m4 p% I4 g7 k多值属性' y# _* f/ D  B' y  D
    可以遍历的字符串- t& \& j, {' k
    注释及特殊字符串4 |' h* Y6 _# A$ I( ?% _7 L
    遍历文档树
    ( y& V6 t  k4 V* }6 W+ H; D子节点
    # o1 K& [  C8 p/ o/ o9 Kfind_all方法
    0 Z% J* O! A: C  K; j0 U! ]. X.contents和.children
    1 o+ `) W9 _& z7 v# \/ Y% G- }. bselenium
    0 o) N+ k9 h& ^: J# E, _selenium介绍
      `) b2 d( ?2 }4 r$ F3 Gchrome浏览器的运行效果
    6 B. O4 j6 |% h$ L. u2 ^9 ^) l( y  ]phantomjs无界面浏览器的运行效果5 h0 ~7 _7 e# l$ W' |4 {
    selenium的作用和工作原理( e" h5 d& s. _8 u* s
    selenium的安装以及简单使用/ E  t+ Z& j$ ~
    selenium的简单使用
    ) Z4 z* M6 a7 ^8 R& hlxml% g2 \3 F* ]  P: A0 X: w
    记得安装快速第三方库,Python经常需要安装第三方库,原始的下载速度很慢,使用国内的镜像就很快啦7 N* W! P8 n9 [7 F2 U

    & K  C7 H0 }+ F1 O; A1 ^

    ) {3 l2 j2 u; V# Y1 c# npip3 install -i https://pypi.tuna.tsinghua.edu.cn/simple 包名& Q) Q* Q# W5 S/ p
    1$ {) H1 O; N1 n* a
    快速下载模块
    4 {, q5 Q( A/ M/ `2 ?7 @! d( M7 L
    , B' n* O: ^  F7 r5 v
    ( G* D# o9 d( @; S5 j% }
    官方网址:
    0 Y% q6 Q- P. L% f/ L$ h: [3 S, t3 y
    8 U; V$ y9 W: e- s5 ?' `3 y
    Requests: 让 HTTP 服务人类9 @2 I4 r7 z( ~8 _+ ^
    Beautiful Soup 4.4.0 文档/ a0 g2 e- Y5 m' e. D! m9 Y
    Selenium官网2 w. _) H, c% B
    lxml - XML and HTML with Python
    - I! |! F/ {6 H0 |' R+ Irequests
    ' X4 B" J" u9 B; T# a% prequests官方文档 https://docs.python-requests.org/zh_CN/latest/
    6 m' ?* u6 C' l" @7 L9 T: C, D6 I% c2 z1 c" t" n6 Z7 s
    $ ~+ V2 A6 i( ^2 e3 w: p" d4 O2 l
    3 H+ c0 B$ F/ q0 o
    2 V* U8 g3 z8 y& _5 U0 Z; Q8 Q
    ) Y" b, ?6 W) a! ]" i
    ' V- z$ ~) S) D' C
    进行爬虫,首先要对网址进行请求,这个时候就要用刀我们的requests模块了。requests是python的一个HTTP客户端库,跟urllib,urllib2类似。与urllib,urllib2相比,requests模块语法更加简单。正如他的官网所说:) I  I  K% k0 M" O

    / A, Z: h" Q+ I: A

    9 h0 c( {  ?2 r& _8 e9 H+ S! ]4 |7 L1 B9 p: H
    7 {3 P# C# S. U5 A
    requests模块介绍1 [, H2 f* h% w8 M. f
    9 n0 q8 ^( A; `5 y  ?& |

    + F0 Y* T/ ?1 |  y发送http请求,获取响应数据- O+ g9 F$ c/ J" L; s

    - v' q" H1 P0 q# ]' Z5 {

    0 k. ^) O1 ^/ r& c8 arequests模块是一个第三方模块,需要在你的python(虚拟)环境中额外安装
    ( o1 G; @/ P+ [- r# S% ^. ~4 V
    2 g' j5 q$ R6 A: R0 F- v: f

    , T3 U9 f% H; a" _* p/ Spip/pip3 install requests
    * v# X% o6 {- \; ~
    1 p; D- D* _. l% Z) e- a. u$ ]

    5 P) f, w. G- O' Y8 |% X' k2 brequests基础
    , |/ _# ]% c5 N; F) wrequests模块发送get请求
    9 P! u2 `! E2 n8 c4 O3 R#https://beishan.blog.csdn.net/
    7 K& j9 o  E4 M; b/ _import requests
    6 s' A0 X2 m: w4 Y: s# 目标url
    . E  {9 }' Y* curl = 'https://www.baidu.com' " o1 V  V) }7 P4 i5 d4 B# e; C# o8 I( N  _
    # 向目标url发送get请求: W% ~2 d% v, e  a9 T
    response = requests.get(url)# y& L' t/ G' Y3 b
    # 打印响应内容
    * c& K* |, B5 u1 I( ]print(response.text)
    % X6 z* _0 k0 a* S' y1
    ' C# |9 S, w6 a8 j/ G6 _2
    ( t3 D: J) ]; ^  v& O2 w3
    % L! N  I8 H+ o4) {: T" q5 s* ^) Y; S
    5
    + R% y2 I/ X! H- R0 H8 T60 X' v  ]8 e# O1 H/ H
    7; q' {! e5 G5 T
    8% Y9 i8 ~) j. K2 C' F9 O
    response响应对象
    + N$ A) m1 l/ |  w观察上边代码运行结果发现,有好多乱码;这是因为编解码使用的字符集不同早造成的;我们尝试使用下边的办法来解决中文乱码问题
    & j# ]- U- ]& b, |/ F2 V, M5 \
    ; Y$ x, V- u& i% S
    0 m9 {5 s4 d+ D* I
    import requests
    1 D+ S9 b7 h3 ]6 w" [url = 'https://www.baidu.com'
    " d. {" {( ~) `* u, w# 向目标url发送get请求
    1 h2 l$ ]4 q9 w  Wresponse = requests.get(url)! J1 [. q  J7 F1 \: q6 r$ i
    # 打印响应内容; D; o- e" b* R1 _* J
    # print(response.text)+ Y: L8 V1 o( J# X( i5 ~
    print(response.content.decode()) # 注意这里!3 t. V3 A# n- T
    1# |% y0 N& u$ T
    22 ^7 o4 g5 t" |2 C4 H
    3
    4 h( G! Z9 e+ t& Q, x4$ q+ T  X' E; i, s5 r# Y
    5
    8 |0 l7 \1 }9 t* t5 B1 T3 j6- Y1 X7 v! i9 H
    72 u4 }: G) z2 ?, w2 W( q& ]
    response.text是requests模块按照chardet模块推测出的编码字符集进行解码的结果
    * a$ @% O& m0 J, Z& e! K6 N网络传输的字符串都是bytes类型的,所以response.text = response.content.decode(‘推测出的编码字符集’)9 V- j4 b: w8 e; e' X! |6 j
    我们可以在网页源码中搜索charset,尝试参考该编码字符集,注意存在不准确的情况
    ! a+ e$ e+ n, U. E$ U# oresponse.text 和response.content的区别3 X9 x5 c8 Z6 S+ `! O& z7 ?. d
    response.text. w# J4 G6 H  S
    类型:str
    : `# d/ ~3 j. i3 q3 g' v0 G解码类型: requests模块自动根据HTTP 头部对响应的编码作出有根据的推测,推测的文本编码
    9 H2 s4 `, Q- B9 K% ?+ M  cresponse.content
    : O6 u5 d3 D' r* z7 j( T* c类型:bytes  B) B. i8 e; }7 C4 r
    解码类型: 没有指定% Z3 B  O" w8 i3 @6 Y* v; b& ]
    解决中文乱码2 j- `; H. W0 [: w
    通过对response.content进行decode,来解决中文乱码, l+ b: h' \9 ]% j
    5 W( d( I6 R' ~4 `
    3 x4 I& t) e$ K2 r8 c/ u- ^7 Q
    response.content.decode() 默认utf-8
    / X( {+ x4 J% h0 J1 Bresponse.content.decode("GBK")7 z& e, r4 X$ G, [& Z
    常见的编码字符集3 K* v* g0 L2 e1 N9 ^
    utf-83 ?7 W& O" G9 u" S; V6 y
    gbk
    & ~3 }( v, i4 ^* X  G  S9 wgb2312
    ) e. Q$ q& ^7 _  g. G8 ?ascii (读音:阿斯克码)
      q8 e5 H/ Q( V2 \- O1 riso-8859-1
    $ a! v$ l/ G# [% M7 Jresponse响应对象的其它常用属性或方法/ M( C6 \+ ?2 D* n1 e
    #https://beishan.blog.csdn.net/- L/ N" e9 Y, ~) p* N' x, x' L6 C
    # 1.2.3-response其它常用属性4 A& U' q/ [6 M1 j9 ^4 @0 o% d. s
    import requests
    $ F/ s: u" J7 P6 f; T" s* o8 B. \) [
    ! \* U- j% A2 I' [
    # 目标url
    7 @- D$ g; b; w1 Y. w! surl = 'https://www.baidu.com'
    ; W1 \+ E6 a7 r7 M/ w6 _, {
    $ `4 \- Q: m% i" ~  V' A' T

    % V  p4 M5 B3 u  H& o. B# 向目标url发送get请求  W  ~3 ?1 U; Y+ B3 [, S
    response = requests.get(url)+ r5 O' \- E( q# R6 c# x1 j8 a$ j
      p6 \* b1 Q$ n( Q& `2 J

    ! ~/ u  Q( A* O! }, |2 Q# 打印响应内容
    6 Z: D8 P8 S9 I% T# print(response.text)% l! X5 x4 n* y, |7 E) e4 Q
    # print(response.content.decode())                         # 注意这里!
    ! J# _6 v: v) X. X/ s; Zprint(response.url)                                                        # 打印响应的url
    + W) {8 D8 `5 b  W, aprint(response.status_code)                                        # 打印响应的状态码
    ) H, |7 U/ P+ Z; Q+ w' G. w" ]print(response.request.headers)                                # 打印响应对象的请求头
    ( e% O+ f/ e0 pprint(response.headers)                                                # 打印响应头
    8 M6 v7 T' S) r  T8 |print(response.request._cookies)                        # 打印请求携带的cookies
    + j6 e- p4 R! ?5 Vprint(response.cookies)                                                # 打印响应中携带的cookies
    7 l4 z" v2 C6 _0 }, G: c1/ R0 B5 H# c5 \2 s  _: E0 J+ ?" P
    2& g/ i' O) c6 V, I3 d
    3
    0 Z1 N( p* [( n0 z) z4
    7 k- q$ u  d# F* b# ]* t3 y, T* p5+ N! m2 k" g- [# L% x6 {2 S' A
    6
    + g' b, K' I% L( ^! T7
    8 F- e/ x) g6 v! r8, F( o4 }% Q5 N) W) K
    9; L5 N8 S$ A' G2 y6 E" ~! n
    10
    ; _4 J2 U9 T9 e: Y11
    # ]0 o$ \" u8 f! Z$ K12
    3 T( O  O- U$ Z2 D( W. _8 N13) P0 }; j. ?% d, x6 K/ f
    14
    , {6 {% Z- x# L* Q" L/ Y15/ J% m8 z1 P* t  ^- Y) i* a3 P
    16
    / n) Q* o& ~1 T( ~% D% P17
    7 ^) y5 \$ Z+ I5 {. e' C' {1 U2 A18
    ; L3 j+ W3 a% `& R) h- r) M: a19* e7 z8 t4 Q0 L( I: I4 O/ d
    requests实操
    2 Y$ M2 n4 \$ {2 _8 g: Frequests模块发送请求2 C4 O. M; [$ s% H
    发送带header的请求0 f' t3 l' S$ J) q

      ~+ l* t8 B2 C/ P

    8 C6 r6 ^; B9 H  N5 N: G- Z1 f3 s+ l6 x我们先写一个获取百度首页的代码, ^9 j' @# i" G  Y6 a
    7 F* i) P9 K$ h9 y: D  D, S$ Z, w

    # L: P% L2 b/ R, J, y$ J; jimport requests5 ?+ I! u: [$ l
    url = 'https://www.baidu.com'. I0 v4 }! s; y2 m( }+ H: D+ B
    response = requests.get(url)
    1 L4 ^( ^4 g" z8 m# ~6 P% ?print(response.content.decode())8 e# |% J3 r' S$ G- \' T( s" w
    # 打印响应对应请求的请求头信息$ b7 V2 G( n5 @) x) Y
    print(response.request.headers)
    / N; A3 ]4 d9 m* ]1
    : f7 y0 B  x' F7 n+ ^# |5 T2; }! E- A# `$ h$ H
    34 ]5 y6 M3 @& s1 z' b
    4
    ; @% J/ E. C! d8 X5$ E% }6 T% w4 S* N$ F9 r2 O
    6* e, }- x/ U$ s* C
    从浏览器中复制User-Agent,构造headers字典;完成下面的代码后,运行代码查看结果
    3 `  }2 {2 G1 E" @) {  w
    # t& [% S. y$ A; V

    $ w( P) C) @& x6 J2 h/ Timport requests( f- Q3 `6 a! E5 n$ a4 `

    0 A1 l5 h2 }* {3 J) V

    . [% U% J# r4 U6 g; O. p+ U' turl = 'https://www.baidu.com': P4 X' u  b+ i5 k1 L2 }2 f/ F- Q

    0 Y6 l) e& g6 |1 x8 o5 D( h2 O+ q+ d

    / L; f; q! @% P3 G' Kheaders = {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.99 Safari/537.36"}
    + \. Q) R' L+ {2 \% N5 h/ a( i9 `/ Z0 |% y) ^: D

    & E, e% u! f* s& @9 [# 在请求头中带上User-Agent,模拟浏览器发送请求
    1 B' b' |) ?# x, e; l9 d& Xresponse = requests.get(url, headers=headers) $ j% a9 |5 [- F

    ! ~: k; U$ o: [( s8 h
    % h/ s9 Y8 _' {3 A, E" m, m6 e' r
    print(response.content)
    4 m) }& K8 P" R1 z8 \. C/ w% O, B* l6 d5 t; ^
    " Y/ V$ T1 F4 S+ L) u- J, Z$ F) |6 k
    # 打印请求头信息
    6 i1 y2 _  H/ k8 ~0 y4 ~* Fprint(response.request.headers)0 L/ m9 U: K9 i( U) |9 w8 v
    17 g" q/ v2 I8 ?  ?, ^. d/ ~, F
    2" ?, a" R$ t0 ^
    3! T: a& v1 s$ p( ]1 h( U/ Q3 N% u" |
    4
    " l5 s6 \7 n8 @" q5
    2 \1 i( G2 ~4 g0 t6 @, T7 p6
    . X, e3 X! j+ O/ l0 s7
    4 u. Y. Z, m* n- Q, a: t! L5 x8
    % F" a) R  X5 z% x" h# }9" h4 Q& H% o% `' w4 A$ N
    10
    * G3 Q" z( Q. z5 l8 U$ @9 f4 F; R- j114 J  {2 u- T/ g
    12# [9 K7 S5 e6 m2 c
    13' ?/ }6 @$ A( @3 P' Y' a
    发送带参数的请求" k) T% I! A$ F
    我们在使用百度搜索的时候经常发现url地址中会有一个 ?,那么该问号后边的就是请求参数,又叫做查询字符串
    / G9 u. k! X4 e# S# k" N) ?, w+ V# n) n/ y: Y- ^4 U
    ( ?, R$ g& k% e  V' K' O7 ?
    在url携带参数,直接对含有参数的url发起请求
    ( d9 \- q: r. |- l, p# X+ D( g, X% M; L" a+ S7 t2 q! H

    : D/ F7 m3 F" D1 @/ a" ?import requests3 r) l$ J! _$ T- E" f1 U& W# p

    9 g/ q; ]6 H/ n# U, n( ^5 W

    + `+ H& L- |5 Lheaders = {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.99 Safari/537.36"}
    ) Z+ P# c& R. {% Q! B! D
    2 J0 Z  o' z4 @# K" ^3 l/ X
    4 j, [% P/ B; k' A, n1 s) o+ H
    url = 'https://www.baidu.com/s?wd=python'' M$ n# D2 y7 H6 k( b7 L" z
    : `2 q' ^: t6 ^/ K! L$ `: j) Z

    % G( E! v( A; O* l3 u. ]: [. hresponse = requests.get(url, headers=headers). b/ M& P! e( F

    , E. ^* B* ?$ Q" n" J, Y

    8 R- O6 Z' s! o# n/ c$ M0 g1
    6 @  l" P- p) f- t8 m+ ~2
    6 L- L5 \  M3 ?( M# ?8 z3
    6 }$ q1 f& e$ t; l- o" E; S4
    ( E9 E- [  Q6 k# f. c9 x/ M* H5
    " M" K, r9 G, R, h0 `; f9 P  k6
    9 h; F: [8 ]. O$ u( d1 [71 N; [+ M, N# d+ C  j
    8. N7 `6 z  ~  S8 Y# C
    通过params携带参数字典# ?8 L5 B- @: ]- ]% S# Y

    ; b! @3 V* x3 H: `# e
    ! m% p2 M% N6 R+ V: B( \6 F
    ​ 1.构建请求参数字典/ N8 ^; b! `4 r" o
    # \* p( o0 J0 o. [
      a1 M7 M0 S3 V* `& k
    ​ 2.向接口发送请求的时候带上参数字典,参数字典设置给params
    ( N, K1 k0 R4 G! x5 e8 o' t
    * }. \( r- h, h  |3 w) W5 j

    1 }1 L" F# T2 j4 f% pimport requests& Z6 R- x. t) N

    : U0 c# W+ t: r, R

    1 i( L7 b9 z$ b/ u# c6 V  F5 z( Jheaders = {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.99 Safari/537.36"}
    0 O, N. N: p1 q& J1 k
    6 E3 ~% |1 f2 R! x

    ! v0 V1 `! |) o' a0 H) d( I# 这是目标url5 ^0 k9 o3 r: S# f
    # url = 'https://www.baidu.com/s?wd=python'
    ' X' u8 {% |" o6 ~& ]: P; e4 O6 U( r
    : f! j6 d2 }9 H' L
    " [+ D5 S% n2 D; W+ _! G8 p  d( ~
    # 最后有没有问号结果都一样
    1 f: ~! a" Q4 G/ x7 ?, g- Murl = 'https://www.baidu.com/s?'
    - g; ]0 ?: S0 f. s/ T8 p* X* _2 r3 `& H- \- Z5 ^$ j# |
    # {! w# [5 {1 m) t8 A! S  w' R  S& X
    # 请求参数是一个字典 即wd=python
    2 ?1 U" \3 ]& V3 n$ w6 ckw = {'wd': 'python'}
    : i( S* f! |. f$ r+ B7 k
    0 [6 \. }' f4 r7 U* Y3 `0 D

    3 p; u7 ?" w; N  k2 z* L: g! ?: D, {" T# 带上请求参数发起请求,获取响应
    / e1 z4 I# U* tresponse = requests.get(url, headers=headers, params=kw)" i% [& l: S! P% }# O! Y

    * X5 m' D! ?" U2 e- y+ E" s4 ]
    5 r8 R% ?% \3 J" Q" B" R2 ^
    print(response.content)
    5 N. N2 Y2 o8 ^& G5 I, s$ \1$ e" D1 C) ]& z; j
    22 ^0 T7 M6 v4 n1 U9 n* b, `+ m, [
    3( z, J9 _* ~( q, p  r8 @
    4: C. B% Z6 c( y; l% k& k& l# b% x5 K& W
    5- y; p* V% V$ X/ d, d& S: j
    6
    5 Q9 A. u6 t- W* j7 L3 O3 f$ b7
    9 S1 Y' o& B  P, A1 I/ {7 p5 l8 |/ A3 ]80 d) Z( ~$ }6 b# r; H6 C
    9
    / M3 P: ?/ k5 C+ U+ ]9 d2 `10
    . H3 v* z! _/ d6 M" C; J: U$ O11
    : G- J9 C  ?( q- l& k8 [; n$ w/ K127 s& \% h# }( H# E' h
    13% Z; {% W8 _( y, `
    14- \: S$ c; h+ _0 T% C) U
    15
    * Y/ w: b2 ^/ R4 ]: J16
    4 U- b7 B0 n0 `173 ~) F5 k) k. ?7 M1 k* S2 l
    从浏览器中复制User-Agent和Cookie
    7 Q; v" t  d: G5 L浏览器中的请求头字段和值与headers参数中必须一致
    + a7 }/ Y- q/ q4 K+ vheaders请求参数字典中的Cookie键对应的值是字符串
    & C( X. X# H9 W- ]! i$ x# e& W+ Kimport requests
    6 s7 R2 |" c: `& ?( ^( r
    & x0 W) [0 }& @  ]

    4 O' o- Z4 C) |+ B3 {/ uurl = 'https://github.com/USER_NAME'4 x! i3 J/ Q+ b$ `2 @

    # M5 p3 z9 _6 }

    5 Z) K. x1 j2 j% Z5 F# 构造请求头字典
    - K: E4 P! K' |7 Sheaders = {5 u0 O$ e: ?' h$ z$ X2 w- d
        # 从浏览器中复制过来的User-Agent
    1 m0 v1 G; I0 |3 M7 Z* c$ A2 l    'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/67.0.3396.87 Safari/537.36',6 c7 s+ M9 A1 w+ W9 w8 R6 C
        # 从浏览器中复制过来的Cookie$ _/ }$ m: f( E2 ?
        'Cookie': 'xxx这里是复制过来的cookie字符串'
    - ^$ K) f9 }  ]: K4 e# A8 w7 H) W, S}  h1 P! F; L7 w' [. S  Z

    ) V- ^  Q4 e( Y' K( \9 t

    " B  U+ q6 ]: J$ X) v2 D# 请求头参数字典中携带cookie字符串
    3 B: B) T/ F+ m( Jresp = requests.get(url, headers=headers)* `* T' Q, J/ x
    " H+ `" m0 C3 X$ u
    8 s, T( [" @- ~7 `
    print(resp.text)4 J7 q$ O3 k* n; A
    1; y2 {& S+ f+ I% r$ `8 \
    2( O  l9 Q' m4 E. a$ B: V
    3
    1 Y$ `& b( ]( r; X3 i+ T4% ]8 ~" K$ \0 |$ O+ p: ?: x6 b6 q
    5+ J* z) G1 |+ s9 f" T
    6; b# N7 W, x! E  j' k' ?
    7
    7 X) E" q+ R( ?0 o% y$ b8
    . r9 v" }' C2 V/ S* g6 p: ?9
      h! ?9 J% L! J* L6 a, o10
    7 }3 w/ w9 Q* w& F, r110 `) N) q5 p. i9 M# y5 d& m
    12
    4 `9 L; j; e: M13# d! O" n, \4 t6 c
    14; o1 P4 v& t: U2 K, k
    15" W3 ~0 I5 j: |$ t$ P( ], U  _6 r
    16
    ! N) g9 Q: p5 r# W6 {超时参数timeout的使用
    , ]- K9 C$ [# S' `在平时网上冲浪的过程中,我们经常会遇到网络波动,这个时候,一个请求等了很久可能任然没有结果。0 Z3 b- f8 k$ T+ x$ U# I, _

    3 b) t' L' P5 i9 \+ k) }! D
    ; i) M0 v6 q( a$ P& o1 Q6 W! W
    在爬虫中,一个请求很久没有结果,就会让整个项目的效率变得非常低,这个时候我们就需要对请求进行强制要求,让他必须在特定的时间内返回结果,否则就报错。2 H, [* H; W7 `' Z
    ' q4 a4 H' E  E( X

    ' w% a' h. y+ E6 Z超时参数timeout的使用方法7 S0 u! v# |% o9 }
    % e9 u: _9 x# o

    , D( @. p3 P  t3 t6 o/ Oresponse = requests.get(url, timeout=3)
    - x4 [6 ~# z1 D: W8 e9 `4 H  C; N5 @# t4 f# Z

    / n+ H1 S3 W4 C  _timeout=3表示:发送请求后,3秒钟内返回响应,否则就抛出异常
    " U! Y7 _7 c# e: V* Z4 T+ I9 s4 H1 P% a# i3 _: R- |
    " u4 O1 F, F3 ^9 l) [! H2 U3 U5 @
    import requests
      _, u% L+ A1 m! E* r, p) f
    : g1 F/ J. ^3 e5 \

    & F3 E; n6 ]0 `7 o' g! n! g! q8 j! b( H8 D" M7 x9 C0 T+ J" K" v
    ) {4 i. q. E* \% s; F
    url = 'https://twitter.com'
    ' Q, w. N' Q5 y8 G/ eresponse = requests.get(url, timeout=3)     # 设置超时时间
    : ^* S9 A: O  t9 A) ~2 x7 R7 l7 [% w+ J: V( k- |
    $ H% F( c- V6 m+ f
    19 _! d. D) v. T
    23 g* W% \3 ^0 F
    3& u4 y; O# Z. X! L, w
    46 a8 A; @+ O9 |0 o' a5 S  q
    5
    ! Q) g% ^: \4 |: _0 g4 y6* I# @6 O4 P3 _0 S" |6 x! \+ u8 ^; U
    requests发送post请求的方法0 |3 |( y- z; ^* ?- ^
    response = requests.post(url, data)
    - t! _8 p, Y  s
    ! C- e+ W% w, P2 T* \- i) u
    & ~6 b+ P$ n8 x
    data参数接收一个字典4 w) v3 k9 d% Z/ E- d

    # P0 w3 T. R8 u% L; W2 B5 Y

    : |- a! H( L% orequests模块发送post请求函数的其它参数和发送get请求的参数完全一致
    ; `$ \# v, i) |6 x! v& u) B5 ]- X
    2 w- A% }5 d% p% i5 Q
    BeautifulSoup
    ! K0 A3 B8 M! ^# R# S; zBeautifulSoup官方文档 https://beautifulsoup.readthedocs.io/zh_CN/v4.4.0/% k; k* a# k: e* `
    ( v! R4 _, y# \5 ^7 @0 h- c; K2 ~& \
    * ^, Q7 i$ G3 q0 K/ J: x
    Beautiful Soup 是一个可以从HTML或XML文件中提取数据的Python库.它能够通过你喜欢的转换器实现惯用的文档导航,查找,修改文档的方式.Beautiful Soup会帮你节省数小时甚至数天的工作时间.
    ) D" f/ K4 }! u4 Y0 ]3 J! L2 O, u4 _1 ~, d, Q
    : w/ Z, K  ^2 R( f
    , h5 |/ s  u: D* @2 H8 D. i  t
    9 p( I; k( t: q0 S0 y2 [* W

    2 D! K7 |1 ?- E5 a% G% ]  t
    . A: Q0 ]3 y5 ^  a
    文章目录
    0 r$ [3 Q4 I& K% j! ]requests' t: F. u  S: e$ o/ l9 G/ o& V" U
    requests基础
    2 M" k4 ?+ G' ?% _$ Vrequests模块发送get请求
    7 w8 s' S' V! E6 I+ Nresponse响应对象- Y, P4 Z' }7 y7 y  q+ w! N
    response.text 和response.content的区别
    , z. J' u, h# q. |. b$ K+ J解决中文乱码. l7 ]1 F7 v) ~; ?
    response响应对象的其它常用属性或方法
    9 j; F3 s. s. ?9 jrequests实操
    & g6 L- N5 [+ a& T; ]requests模块发送请求0 i' n/ u8 V, X- z1 y# C
    发送带参数的请求) d, L. ?- ^: L1 F3 `
    超时参数timeout的使用  x, `& _8 A, c& p* b
    requests发送post请求的方法
    ) H, r% K+ |+ Z: U9 [BeautifulSoup* r: j3 a3 h, v- i% n
    常见解释器的优缺点
    ! X& g. W7 P( t, r常用操作
    & z, e9 _2 i& m2 {几个简单的浏览结构化数据的方法' v- \) B1 w* Z' g8 t
    从文档中找到所有的< a>标签的链接
    % R% T8 E1 g% `1 y/ J* M9 k9 P在文档中获取所有的文字内容" T" u  {# @& m* ]
    通过标签和属性获取
    ! K" ^2 `' u; g1 @+ \& O% }Name属性) N4 z. e: l2 D4 ^
    多个属性& @' g5 l" q  x' E5 x
    多值属性
    ) ^' \  Z8 I4 z) U2 s! E; f8 x可以遍历的字符串0 D7 V, X' y; i& j" S" x
    注释及特殊字符串, w( f9 i$ \2 O. G$ V
    遍历文档树4 a0 m$ l- @: S
    子节点; _! u* h; O7 ~- l, ]
    find_all方法
    - q: x# t' f& S# V2 }' N2 z( j.contents和.children. H7 S1 w6 a2 u& b
    selenium
    ; p+ T% o0 j' g. n+ l) W! ~selenium介绍! L) x  V  y* ^8 S5 z4 P* M: M
    chrome浏览器的运行效果' `2 ]* y4 R& ^% a
    phantomjs无界面浏览器的运行效果
    9 \1 y2 h! w, S1 ~- \selenium的作用和工作原理
    $ k) A5 r7 C! J# Cselenium的安装以及简单使用
    2 K% a% [# d" s( G8 Z6 Oselenium的简单使用
    $ @" ?; i# k: w, [4 slxml9 l$ {2 r6 }$ Y8 \) B. w+ z
    常见解释器的优缺点
    3 T* p! H3 s3 ?# ^8 h; J5 F( @
    5 ~/ s2 b7 v* r0 A4 U) A

    0 n2 Y! |1 m+ @/ Y  b! I& O
    ( o% Q/ w! i( C

    / A% L# o& f" k& _) [% V, _& {常用操作
    ) j  j+ u7 _3 U' r' x安装方法
    , A4 v- F3 C3 N+ d& ~. G% ^* V' H! N% o5 @) S. g: u  W5 W
    " S4 B+ a# ~3 p- l' O; L& |
    pip3 install -i https://pypi.tuna.tsinghua.edu.cn/simple beautifulsoup4  1 N2 ?" L8 X6 v# g! a" V. \
    1
    : Q. }3 ~) a9 H* X7 W% y导入即可
    2 n# G, Z% ~8 \0 w1 l: x
    4 ^5 |- ^, u* H2 A; R* u1 n4 m

    & L" |& }: \* Q' O; i4 |from bs4 import BeautifulSoup
    . Z" g; P2 D- a+ r. `1: f: J" @$ _$ {: ^, D. W
    html_doc = """
      d. F3 `% ^) {4 D) f( k4 c7 g0 n6 h<html><head><title>The Dormouse's story</title></head>
    . d4 H9 L: c  W/ [6 _+ E7 X* t<body>  Z/ Q* t+ k; P* W& d5 n: b3 F3 X
    <p class="title"><b>The Dormouse's story</b></p>
    0 {+ k; @/ C. J, L
    8 w! D0 ^; J2 {& ^2 d
    5 a. P* I, A# J5 @0 `: a+ l7 S
    <p class="story">Once upon a time there were three little sisters; and their names were
    * k6 @3 S5 Q2 Z3 i( x<a  class="sister" id="link1">Elsie</a>,9 u% p- Y  A- {6 |# ~  ?/ g
    <a  class="sister" id="link2">Lacie</a> and
    : g  s2 X1 p, ^, \<a  class="sister" id="link3">Tillie</a>;; R) y4 v3 Q( o* o8 P& u: a+ R
    and they lived at the bottom of a well.</p>
      V9 m3 S* o9 `4 p
    ; _" ]; [+ w0 \& Q1 w& P  ?

    # p- |/ ^! m5 m! {6 k7 H<p class="story">...</p>
    8 d5 C/ Z+ M8 U4 O"""5 M/ |( |  H+ G
    1
    & x' o5 y/ @3 K8 {1 j) F+ X8 B. c2# P  U3 D$ Y. M2 i: D) p
    3$ C4 S9 w  U$ V! e3 o
    4# K' \9 n- F" w3 s7 T' w
    5. }3 I& o) r1 y9 S. o- A
    6" x! _" f9 A+ p5 ?+ M' \
    7
    - I4 \1 r0 d' [% K$ p: T+ N# z8
    3 W2 x1 O# l9 R# ^: s9
    & ?3 O3 a# {' Q8 q- @4 g- w10
    . `4 x' H: v+ f+ U11
    , ?& @. A; ~$ E12. D( l0 l; B, z" j" c( B) \9 T
    13
    3 L. v; `% T" g& \$ a) o1 U7 y# Esoup = BeautifulSoup(html_doc,"lxml")4 x7 A6 |. f1 r, t; C1 c# Q
    1/ W, ~5 k* i$ n/ q+ W% V5 J
    几个简单的浏览结构化数据的方法" U' K4 `2 \" a) l: B
    soup.title
    / G* \3 q/ |0 O* k) \; Q- O5 R" o* ?1, R+ k% q- ?9 q" E! G# J
    <title>The Dormouse's story</title>
    & J7 d; Q0 |9 T1 e. a0 Q: j1
    6 F- D" z) ^8 T" h0 c/ K4 Nsoup.title.name
    ! }. z; C) p* C" n1
    # s2 u* n/ F! y  C0 A. D'title'2 I. C5 L5 {- s9 N. v
    1! h! H5 S: F! ~4 K/ a, ?
    soup.title.string
    7 @% W( p6 _- C# O# B* z) F! h1; s* h- A/ a; ?
    "The Dormouse's story"% d1 i. u  C) Z: w) t
    11 F4 @0 d9 T+ n8 T; R) [
    soup.title.text
    ! M5 O2 M8 \9 H1 `10 P1 P7 U) z1 p, ~. @
    "The Dormouse's story"2 J( m# l5 v+ t$ L4 N, r/ Y
    14 \4 E# d4 J1 `3 m
    soup.title.parent.name, Y+ l8 \7 B, I, p2 ^
    1$ }+ ~) O7 V# [1 k
    'head'
    : l1 [( A$ _/ }) k15 [* V+ \6 W! d
    soup.p0 W  a9 {/ j% ?
    1
    1 s# _8 T$ M" u7 i6 `" Q3 F6 q2 G<p class="title"><b>The Dormouse's story</b></p>4 h8 e0 O: a, c& r8 M
    1* |4 X/ F+ P9 e+ a: r$ Z
    soup.p.name
    ( P& P3 P+ P" z  q1" @) J) N* b5 @4 j* @! ~' C
    'p'
    8 R( q  r' Q# a% ^1 J* x0 _17 J+ `' t; [+ l( ]9 R9 Q
    soup.p["class"]
    5 s- f! f2 n% R  P1; ]" z" U, ^" j0 L+ X; K& w2 q
    ['title']. O$ z: f  l; R5 v/ A
    1! S6 F' b/ X$ b1 o/ c
    soup.a. R, Y7 q9 G$ }) W8 p
    17 I$ M. e! Q- C) d& m' E
    <a class="sister"  id="link1">Elsie</a>4 a+ @. V  }4 a  L6 L
    1
    ) l& k1 e! w- `0 W! ?$ o# usoup.find("a")
    7 I, t1 e* Z% H1 [0 M1
    ! r4 v& m% `( M3 F<a class="sister"  id="link1">Elsie</a>' J; B) b5 L: J& Y. r* v. y0 v
    1
    / r: v) u3 E. u. ]8 p1 `2 m2 @soup.find_all("a")4 B) S, y" l" t$ R' V
    1. s- x! ]' m: `" o
    [<a class="sister"  id="link1">Elsie</a>,. M* R& `( u7 ~7 r9 D; w
    <a class="sister"  id="link2">Lacie</a>,
    4 P- l9 Q: v' }1 M <a class="sister"  id="link3">Tillie</a>]: X2 k1 M8 o- l& w  `
    1
    : R+ s7 z2 ?5 k3 f# {! r5 P2) C3 R# ~" N. O# k5 N
    3  D% x" A  K. c
    从文档中找到所有的< a>标签的链接
    ! a. X! z% a7 E9 N# s% ^; W$ q6 gfor link in soup.find_all("a"):- h3 h1 v: m+ y- x: Y
        print(link.get("href"))" \* Y8 }( F" t( B
    1
    4 A8 J( |4 b1 D: A3 T' K+ |20 x2 T) T4 u& h/ J# w! U0 c
    http://example.com/elsie
    . c9 X5 k: x" r% shttp://example.com/lacie, ^- O; ^* V# O; N% g$ k
    http://example.com/tillie; a8 _# b6 |) u3 w4 ?- q4 i
    1
    - f3 S% p. F0 w2( A: `# O7 T7 O  g" x9 n1 w% [
    30 ^  I. O; U. y
    在文档中获取所有的文字内容/ x1 r6 U! |9 k1 `) C& o" b
    print(soup.get_text())
    1 c* _( K5 o4 l3 G, C7 r; _1
    9 p8 i. {2 y4 l, P4 yThe Dormouse's story
    & `% v, }% J2 a
    % K2 u/ `+ J# c8 G7 q  J9 l

    3 ~" T: b2 ^+ v: h; T7 @( B) qThe Dormouse's story
    1 w5 |' G& e1 w3 |+ qOnce upon a time there were three little sisters; and their names were
    % |1 u) J; X- ZElsie,
    ! g2 P0 O9 |! ^Lacie and; p3 t. d, h7 s" g/ D  P3 U1 Z
    Tillie;/ ~. T( V" J- v/ S7 y! p' G
    and they lived at the bottom of a well.# W( _) \7 m: {7 L9 l6 q- `
    ...
    0 ^5 y! Q3 }0 O$ L1
    : T' M7 [  i, q. o; h; G29 L2 k; |/ X8 c9 X1 G1 R4 E5 K
    39 B# J6 _" L4 p$ K% m
    4, ~. x. v  |& a2 A. E* f! C
    5
    ; e' s# ^0 R8 ], E. _# F  {6
      Y& f- j! s3 \- r5 m; \5 m7
    9 Y6 \3 X; U( x) I8# ?5 D) r8 k; W& u- S  }
    9; D" ~, l: x# ?7 ?7 j) ^9 Q
    3 M) T. g# g6 b. q1 p
    % f( G7 u: t4 E

    ' ?( q* q. R% O  {+ l7 f通过标签和属性获取5 N' c$ \8 B; A3 `( R
    Tag有很多方法和属性,在 遍历文档树 和 搜索文档树 中有详细解释.现在介绍一下tag中最重要的属性: name和attributes
    $ g- o% Y- p3 l5 x) a* msoup = BeautifulSoup('<b class="boldest">Extremely bold</b>'), E: c& O/ {1 j! ]" E
    tag  = soup.b4 j( n8 H2 f' W  b
    tag( b) t% u6 {; G! O( K
    1
    ' r/ ~, _: s' D7 o2
    8 O+ i+ h$ o) n" o& k3
    # y, o, N% \4 E! ]) w' V! a<b class="boldest">Extremely bold</b>" d( g# j& {2 S( ?7 g3 N3 a
    1
      M) k+ h6 O: A2 T. }' m2 ~& [type(tag)& K/ g2 T7 k& S6 }  Z  Z
    1$ ^! M: @5 L( C0 `
    bs4.element.Tag3 n9 K' C: @: u& m
    12 c1 Y7 |4 L+ }4 d- N
    Name属性
    $ n0 }  v1 L8 `每个tag都有自己的名字,通过 .name 来获取:
    ; l3 k5 R1 _" g; |, c% h+ D! Ltag.name
    2 j! \2 c5 W4 }9 }/ h; E  H2 P) k1
    * E9 w% i# y9 w" X'b'
    / G. h6 @% q+ b+ e, g1  h8 E: R4 }: D; a0 P
    如果改变了tag的name,那将影响所有通过当前Beautiful Soup对象生成的HTML文档
    6 o! j! w( l6 K( @/ k: S9 Ytag.name = "blockquote"3 R$ E: ^! l6 g5 s- a
    tag
    & y2 d+ @8 ~' F& m. v1
    , P: ?! w3 o. R% b: S20 |, k- m- e' j$ w) W! i
    <blockquote class="boldest">Extremely bold</blockquote>8 l% l6 p2 n' M1 `' P, {
    1
    ' I" f- Y' F5 F. G' i7 p多个属性
    - v' z* d3 v) a  Z( f3 M一个tag可能有很多个属性.tag 有一个 “class” 的属性,值为 “boldest” . tag的属性的操作方法与字典相同:
    7 b0 N( ^; z% h% |# O: Ptag["class"]: g1 u) A/ U/ F3 _1 M8 x
    1/ Z5 ]' ~# f3 \2 h/ A' W
    ['boldest']9 f8 s) Z" V* u7 A1 i$ q  i' _
    1
    + \  F, q0 [$ l1 C! S& ?& V* Ctag.attrs
    , r4 \- h9 O; o6 B' j$ a1& K% |  v. @1 u3 C/ w5 I  k
    {'class': ['boldest']}' K5 r% s8 [  @+ [% c6 t" w
    1
    # Z7 [3 Z  ?4 h& Q' b- ]tag的属性可以被添加,删除或修改. 再说一次, tag的属性操作方法与字典一样& \0 }, W( `; M0 x+ n5 ~) w1 ]
    tag["class"] = "verybold"4 ?4 m$ ^9 @* q
    tag["id"] = 1$ p& [- D- c9 r! d+ m5 z
    tag2 ~  O& V4 [; f' Q' m! @7 w
    1: D" g3 M/ |/ V
    2
    ; C8 i- ?6 q6 Z- @+ x- S3
    . }& L: A/ k# Z- r3 G5 u0 K6 s<blockquote class="verybold" id="1">Extremely bold</blockquote>4 r/ }4 ^/ G4 h! j) n. ?. P
    1% W0 z4 J- A/ q3 v4 Q2 S) ^2 h$ f! h
    del tag["class"]
    ; ]: S8 J( g3 f# Itag
    0 _( y9 }9 a0 i$ Y8 i5 Z1
    ! h1 B5 ]: W, J. V; A0 m' g2 D2
    4 q) p2 c/ P& A# y( k<blockquote id="1">Extremely bold</blockquote>
    , W5 {* f, W! `% _4 a# B& D1+ L( S8 R$ {1 j) [+ I8 J" E
    多值属性
    " G  K2 q7 z' ?6 Zcss_soup = BeautifulSoup('<p class="body strikeout"></p>')
    ) F; B% A2 V: e- X& S# t, Ecss_soup.p['class']+ `  Z# b: U' _/ N: }9 r! d9 C
    1, M; E' \/ {% t! O  q! Q* Q+ T+ _
    2& V" b0 x7 w& J3 {4 Z" Y3 k3 D0 u
    ['body', 'strikeout']
    2 {. j/ ]) |/ k# E- \1
    ( S  K6 |; x8 a$ |5 ]* E% h. Wcss_soup = BeautifulSoup('<p class="body"></p>'): F  K' }+ U4 v' E; \
    css_soup.p['class']" ]- x) S) d2 _' \3 D8 ~& U
    1
    ) W8 Z% u1 c5 ^' a. J1 i% o) F1 G9 b28 m7 ]* w; l4 D
    ['body']8 c! T1 y# O: r5 G2 L
    1" V. ]1 \- y' r2 B$ p; L
    可以遍历的字符串" F' K: H) M0 x8 t9 M  V) o) R
    字符串常被包含在tag内.Beautiful Soup用 NavigableString 类来包装tag中的字符串:
    : C6 k! I2 h! O1 _' Ptag.string4 Y2 S2 k6 F4 f
    18 L  @0 G& n% e! R) r8 }/ T3 \: a
    'Extremely bold'
    2 a# m. ]# N3 T0 J2 g1
    / E, j/ M3 Z7 rtype(tag.string)% t2 Q2 Y! T0 L4 P# t* |
    1& }# o0 d% z- G+ K
    bs4.element.NavigableString( U( g# d# M; W" i8 {& O' N
    1' R* S4 ]( Q- V5 L7 g
    一个 NavigableString 字符串与Python中的Unicode字符串相同,- X3 D3 i- d$ W& ]1 }7 q% \1 u
    并且还支持包含在遍历文档树 和 搜索文档树 中的一些特性.
    3 m5 j: E' e" m+ U' O通过 unicode() 方法可以直接将 NavigableString 对象转换成Unicode字符串:
    0 Q: ~( i, w' A0 L  C- }3 [. V
    7 F4 _3 M( d8 e, Q
    - X0 G$ n- P( J0 _
    tag中包含的字符串不能编辑,但是可以被替换成其他的字符串,用replace_with()方法8 m5 X4 C  I2 |* t2 p; \* |' J0 x

    : F% X# e' d6 R$ n0 z2 }; U- ^

    6 y5 ~1 [# J6 ?, k+ K' Q: \tag.string.replace_with("No longer bold")! L: E+ U1 k  f3 I8 D' \
    tag: ]7 U. y, a' Z; z  Z
    1, ^; d7 A, t' k+ M; K" t" [
    2& \) L! S! e, N4 H: N' j! |: U' Z
    <blockquote id="1">No longer bold</blockquote>
    , m& U) M6 b3 U8 a0 c1
    5 {9 u7 D7 E7 J注释及特殊字符串% p3 v9 b0 I! b( k, v# |1 X( N
    文档的注释部分
    * c* ~# n# d/ Q6 I3 N* \markup = "<b><!--Hey, buddy. Want to buy a used parser?--></b>"
    $ T! E( f8 [# Y% hsoup = BeautifulSoup(markup)# W: ^# f) Z# n- n1 m" y- |
    comment = soup.b.string' q* h+ \. k% W2 d* x
    comment7 S# l1 @4 W. V/ \! Y, F; V) ?
    1
    1 ^9 V% q( d9 B5 k( ~9 F& p2( D3 Z0 d$ D* {& a# P3 R6 _3 ^( V$ a
    3! G3 w. H- |2 g$ H8 g1 U9 q8 Z4 z
    4
    - K, {6 O' k. z4 f'Hey, buddy. Want to buy a used parser?'& ?8 Q" }* J7 o+ C2 k/ ?
    1
    - ^) r) e0 O, _3 r9 a2 |. ?* u" F6 ?type(comment)/ @) J4 x- m; T5 z: e
    1
    $ B  O2 W6 F8 m" q8 zbs4.element.Comment( |  b4 U8 F: r' j$ G  V. e" e. o
    14 [1 j- A; p# U+ {& Y7 g* L
    Comment 对象是一个特殊类型的 NavigableString 对象:- X) J  A: V% g3 V- U1 d! W
    comment# Y$ E' _! b+ z8 f! ~
    1
    ' P( B4 c2 j" ~'Hey, buddy. Want to buy a used parser?'. }+ R) m' |. P! h$ [; u
    1$ y, u0 E& y2 j+ H9 v7 u+ ^
    但是当它出现在HTML文档中时, Comment 对象会使用特殊的格式输出:) D( D( v/ w* m. \
    1 b  v! L$ y6 ^& I- X5 W

    7 s4 W3 @8 F% @2 Cprint(soup.prettify())4 O9 P) R) U/ V; i. {
    1
    , H6 m3 g  n- T! o) C4 G<html>2 f+ y+ p. h5 w- _% ~- {
    <body>" V- Y; \* I2 a7 }9 [
      <b>( m$ P( B- n  Q
       <!--Hey, buddy. Want to buy a used parser?-->
    $ {) H& I8 c& ^; {  </b>
    / ~0 x8 P/ t% }! D9 ^/ x </body>
    8 G2 |" B  r0 M3 p! |/ Z  L</html>
    - F6 L, L5 g6 T4 q4 z- `8 R1
    * x6 G# M; ^9 [: t3 Y' W6 A2 x% S: [26 o7 q: C- o3 e% A
    3
    & x4 x; g0 o) k8 B46 m8 R' x' K: h) K
    59 P1 ~! _/ w8 X1 Q- ?" i
    6
    & Z" h. F2 `1 \+ ^  V, |1 j8 K0 K7* c, Q; X! ~( h2 t" P& J6 U! q, u0 Y
    from bs4 import CData1 \' g5 q5 C/ u6 ]5 T
    cdata = CData("A CDATA block")
    / L) j- i/ Y9 Q0 t( L3 acomment.replace_with(cdata). ^9 r- \/ |. \4 U8 f7 S
    print(soup.b.prettify())/ A- Q' i$ O) C2 b% E6 |8 u/ K% _
    1* L% }' ~; `. k2 T
    2
    $ k# i6 @& ?; m39 O$ o" ?) r. ^+ @" ?9 T
    4
    + K5 }* f7 s, {2 l2 N$ F<b>$ d, M7 L* k: M8 ^% l2 r
    <![CDATA[A CDATA block]]>% D* l" n* {, d! H# |$ B
    </b>
    5 E( p" q6 V# R! a4 |. C6 {1
    + t2 _. S2 m0 @8 ]$ O% P7 o; G  Z2
    9 w& [8 x+ k" h% _3
    ) ]: l/ R/ r. r遍历文档树& w/ d7 \/ p  a) v7 {
    html_doc = """4 w2 h6 N2 N& _4 R! k
    <html><head><title>The Dormouse's story</title></head>
    7 _# Z$ n5 `# D7 N9 N# U+ L+ Z    <body>
    % h# B* K3 [  a<p class="title"><b>The Dormouse's story</b></p>
      R% D0 x& ^, i- N2 }, Q
      u* {. m8 J! D7 S: q0 V0 L
    ) [  Z1 p2 Y6 x: m6 h
    <p class="story">Once upon a time there were three little sisters; and their names were& i1 w. f2 E% T9 a- ]6 J
    <a  class="sister" id="link1">Elsie</a>,
    : \1 {& t* u* v5 j3 s7 S' J<a  class="sister" id="link2">Lacie</a> and
    ; \. x1 k8 I- q6 B$ Y! X9 l<a  class="sister" id="link3">Tillie</a>;$ a. R) }# U2 G" |! @
    and they lived at the bottom of a well.</p>
    8 T2 U  ]! t) J! L9 V3 u2 y) a' n, C

      p4 |5 r2 S; ?5 R/ V8 Q( g<p class="story">...</p>
    1 c( R# t  ]  Z) b5 j"""
    5 G& b1 ]2 ]' e1
    ) c" H# ^% _2 ]  x3 z2( M: j( ?' S5 E" @& m0 o9 H* P
    3
    9 Z4 n1 M: ~. H$ q" u4
    ) |! T' v/ V# r# h7 w5
    ! Q6 u9 }/ Z: g" ]8 a6
    - F4 d: }  _/ s, a7
    6 m, i/ O  r. g% M# ~8
    # P5 F8 D  W3 K( c6 j; Y* U# c1 M. L7 }9
    1 ]% g1 J$ \2 [# d10
    % ^( D; F3 m9 k3 r, X11, u. y3 u5 B- [! N/ d9 x8 n
    12* ^) {* l% r0 b  l' O
    13
    ) b# c3 ]" x, ufrom bs4 import BeautifulSoup
    , |, F. M/ ?3 l" F4 X1 Q1
    7 q! x) v4 I3 t* R; Qsoup = BeautifulSoup(html_doc,"html.parser")8 Z6 G0 ~9 f5 U: h6 w
    1
    ' q# ~+ B0 v: ]$ M, ~! t) Z6 L$ ?$ L子节点
    3 q: l. V: N  k- K一个Tag可能包含多个字符串或其它的Tag,这些都是这个Tag的子节点.Beautiful Soup提供了许多操作和遍历子节点的属性.
    9 R% |" T7 S- g. ~7 L0 O5 j7 h. }) d) a8 D( k" Y# `# l
    4 e& i* D4 v$ q: r% n
    soup.head
    5 S. `5 p" p% V" ?4 G1
    ( l/ ]' y/ r8 O/ n; z<head><title>The Dormouse's story</title></head>
    5 m. O$ r. ]% I! }% _; a7 a. u- e1
    $ k  S  e2 i6 hsoup.title
    0 |" P! D9 m% S% E/ [1
    % v3 q/ k: z' m4 ]1 A<title>The Dormouse's story</title>
    - e1 f7 Z$ S( |/ w1' |1 G9 c. e3 v* `2 D; g/ E
    这是个获取tag的小窍门,可以在文档树的tag中多次调用这个方法.下面的代码可以获取标签中的第一个标签:
    8 d# v% D% _+ U+ m; e0 c  }+ ]) S& P* F

    0 T0 `6 S5 z0 Q" C/ vsoup.body.b5 P7 v: ~8 l8 i) j3 J" v7 ]  B
    1
    / o: i6 ^4 [5 l8 B' g, W! g<b>The Dormouse's story</b>. R# Y) @$ m: ?0 d+ V& o4 ]) Y4 \
    1
    6 G" q5 U: V! a% m7 {% d' T通过点取属性的方式只能获得当前名字的第一个tag:) e$ |, |4 F$ G  _

    $ n# U4 f  A8 h( k0 _, h$ |
    ( D1 w# }! a; z, S% V+ M3 v
    soup.a( s" m' E" E4 I# n: C+ [
    1# t3 f6 T* ^* ~5 p8 l2 }' R' N
    <a class="sister"  id="link1">Elsie</a>" o* L) t, {- ^9 l. V! [. ?
    11 b$ _( l6 i4 c
    find_all方法% }7 ^( h% W+ R
    如果想要得到所有的标签,或是通过名字得到比一个tag更多的内容的时候,就需要用到 Searching the tree 中描述的方法,比如: find_all()# R1 O& U" y$ Q5 t% D2 e2 T

    4 A  L# B' ~$ L, t1 S2 m
    , K6 c8 z5 J* i3 y$ U3 @
    soup.find_all("a")) c9 b& C( K$ P# q
    1
    7 @1 Q: }' ^( X- l5 K1 |[<a class="sister"  id="link1">Elsie</a>,/ R. A' ]. Y$ o* @- w9 s1 O4 v) H
    <a class="sister"  id="link2">Lacie</a>,
    5 Z1 \5 o! I. b1 B; o <a class="sister"  id="link3">Tillie</a>]
    ! I! n: ^- F7 J4 U4 T1
    8 k3 k: e& `) l0 `5 C. a$ N/ Y2$ d4 C. o3 V# ~/ s
    3
    ( [+ [4 N2 G8 M3 C.contents和.children
    * @  ^* F/ p/ k) C6 }' z5 H* lhead_tag = soup.head/ t0 V# |4 F& F- _7 m! O
    head_tag& {7 A6 u+ {/ [; K8 }- O* }7 L
    1* n2 ]" L# Z! [; o) I
    2" F6 D# Q( B6 f3 t
    <head><title>The Dormouse's story</title></head>- h, Z, F  [5 l8 Y% m* e  E. Q
    1( H, m4 a) m/ v
    head_tag.contents
    * [  ^6 N7 a& q% e4 ]" R1
    ; B- w/ }. H5 l) _[<title>The Dormouse's story</title>]0 V% m0 ?, H4 d; j; a8 K& s
    1
    & M6 }, l$ q, r6 z0 }* Y: hhead_tag.contents[0]3 X4 C  m+ I8 o$ c
    1
    5 c- g# l" h5 v: |' r- g2 |<title>The Dormouse's story</title>) Z5 C# r0 ?9 }
    1
    / t# }4 @+ e2 X" b) N) whead_tag.contents[0].contents8 {2 [6 w, X2 D8 p/ |9 e- I
    17 V" F% L2 z' ~+ Z$ y( T5 B
    ["The Dormouse's story"]
    + J/ @" J/ e0 i' H& x0 p  a) N1( |8 X6 B1 m9 t
    selenium
    , a4 b% h: V0 a+ Y7 }9 b# @2 u2 @, Y" u
    7 {# v+ ^7 [" e9 N) g
    ) J! q& A1 e* C: }

    # }9 ?! D# U' |% U' q2 eselenium官方文档 https://www.selenium.dev/selenium/docs/api/py/api.html
    2 }3 S) t$ }1 D0 h+ D% ^7 J$ p. ]
    ! r; b4 R- m2 I! X/ V
    selenium介绍
    - h8 Z" X- x, _7 |. Mchrome浏览器的运行效果# v$ q$ k+ d' ^/ ^9 t# I7 V
    在下载好chromedriver以及安装好selenium模块后,执行下列代码并观察运行的过程8 f4 h, Y0 ~% E
    & D$ `, x% H  g# Q$ Y

      u. ^- Q/ a- e  x$ [" z( _from selenium import webdriver 3 K3 v9 E, @; H- q
    ( p7 ]& p5 p7 j

      o& m- c& L7 w4 k+ w7 F3 d# 如果driver没有添加到了环境变量,则需要将driver的绝对路径赋值给executable_path参数4 Q+ P5 |, b1 s0 C* d. ^
    # driver = webdriver.Chrome(executable_path='/home/worker/Desktop/driver/chromedriver')
    / P+ n6 U9 J; _! Q6 k
    3 h, M# z% b3 C% A& S; R9 s

    4 [& W1 o7 P5 [3 y2 Y" J( X+ [# 如果driver添加了环境变量则不需要设置executable_path- S. H5 W1 j2 c( j) g7 w* }" c
    driver = webdriver.Chrome()# q7 I9 ^. v. o; s1 y# w

    , Z! Q/ R/ ~( H& a; {

    " h: a* P; U. A9 F7 N# G# 向一个url发起请求
    : T( U/ t5 G/ ]0 x8 E% Idriver.get("http://www.itcast.cn/")2 K' t; k% F! ^/ Z

    + F; v. E4 G8 ~5 k9 c$ A, z

    6 c$ g! c0 |! M9 b# 把网页保存为图片,69版本以上的谷歌浏览器将无法使用截图功能8 @) s  \! g/ A" O9 ]7 s& B
    # driver.save_screenshot("itcast.png")
    ! U( K9 w3 i) _0 G8 ~! E
    ! P% |+ c8 W0 B. N: E

    # g5 l3 j9 _7 ?; P& v4 |$ j$ fprint(driver.title) # 打印页面的标题, R0 e  a. L* l! o2 E, v: i

    ( _) s6 y% ?( M" e; ]% r6 U7 c3 @

    ; [2 [! N4 \( Q6 D# c# 退出模拟浏览器
    : s  Z% S; x" U( d. Y) Ddriver.quit() # 一定要退出!不退出会有残留进程!
    5 {4 \9 k% g& j8 `. t1. W! d3 d, ~: |9 S7 ?' ^) _: D
    2' j- p" I( ]' o/ Y+ W& `6 z5 s2 q; E
    3+ s) J  A5 v/ h6 i
    4
    : c2 R1 n2 c# j. P5
    : l' Q( |# ]3 v6
    4 C4 ]% C9 ~- F0 \7" `, n, e4 b* i6 p. N
    89 ^; g' ]9 `7 f) r- `! v
    9
    , M, Z0 X! [+ z10# ]# _3 A/ g+ E4 z) |( h% ]( `
    11
    - j7 S0 v6 T, O7 J0 P: t% c12
    ' Y* ?4 i- |+ e8 g7 }5 }5 ^1 s13  t5 I  N) X( N. s/ Q% \/ l
    14
    7 G. z9 H3 z+ [1 \: t15
    0 L5 p; k* a* x# t5 ~16* J7 y$ @( G* t( Z* ]' f
    17
    ' a" [1 M* {: H& }" ]' r# F" y3 l18: f* z( Q8 |" R* b: o
    phantomjs无界面浏览器的运行效果
    # H( ?2 V3 l+ U0 dPhantomJS 是一个基于Webkit的“无界面”(headless)浏览器,它会把网站加载到内存并执行页面上的 JavaScript。下载地址:http://phantomjs.org/download.html
    7 V9 }' ^. U& C" P& t9 t+ U3 {0 Q0 Y* D! M/ q/ \1 P

    & H9 T, y4 k6 [# ]/ L! |8 U& |* b. efrom selenium import webdriver ) r0 {: d5 B1 Y

    + r' {2 z$ J! c4 k

    , F2 ?; y4 n1 }1 C, @# 指定driver的绝对路径
    * N! ?0 g) D2 t1 t  u. g6 Edriver = webdriver.PhantomJS(executable_path='/home/worker/Desktop/driver/phantomjs') " w0 M+ ?. r. e. x+ l/ P. t6 x
    # driver = webdriver.Chrome(executable_path='/home/worker/Desktop/driver/chromedriver')
    9 Q) A9 s* M7 x# t9 k
      e  R# H7 z) W/ Z7 C9 W
    8 x5 \1 d& {5 @& d) L) I) P
    # 向一个url发起请求
    , A9 _- c+ D: g6 u; S& P9 ?& c: {& Xdriver.get("http://www.itcast.cn/")- b8 O/ k; Z, w  e8 I& ~& Z% N; J

    + f: y; f2 A) O" B- {
      ?2 W2 c+ p' N/ r2 ]
    # 把网页保存为图片
    ( ^( _, h4 {6 c; Fdriver.save_screenshot("itcast.png")8 R! i: u" e& ~  J+ w3 @
    9 ^1 _2 c1 O: {0 g+ {/ a6 {

    1 y& ]! V2 w* U2 O( {$ b# 退出模拟浏览器) Z8 g8 D& N/ f) z$ X: {6 u1 B9 ~/ n+ Z
    driver.quit() # 一定要退出!不退出会有残留进程!% ]" O7 Z' i. a* g' b. n
    1
    $ U7 B0 a5 |0 U6 L  ]2
    / M  I6 S2 @1 ~6 ?6 N+ p# r0 l3) U( m; f2 F" P# \" c- c: H
    43 p1 i. F- Q0 v2 o7 m4 A
    57 g  R- [& o5 v5 f/ u6 g  f
    60 t$ ?3 i8 ^( \, K7 {8 H+ U. W
    74 w/ E' U  l  |, ~8 u$ @% s% L; q
    8
    $ A4 Q6 M) r' H/ D+ U9
    $ x* c, y( `* ?1 [0 k6 ]! d103 i; f0 Q+ T1 p; w5 G, S
    11
    9 i5 ?1 @% O4 _( Z  M12$ S& J. J' o. e. }
    13) m5 S; s) l7 m
    148 v9 g. L+ x9 a, a8 A0 L% _" q
    无头浏览器与有头浏览器的使用场景8 l8 @) r0 O# v  c5 J, R
    9 f0 F. B" [" r. _( V, u' {1 u

    " z6 q( [6 ]- q* w# m, g) v通常在开发过程中我们需要查看运行过程中的各种情况所以通常使用有头浏览器
    9 C# p  _4 u8 \: @$ g5 w在项目完成进行部署的时候,通常平台采用的系统都是服务器版的操作系统,服务器版的操作系统必须使用无头浏览器才能正常运行
    7 p, G! U" m/ e' aselenium的作用和工作原理2 k# t5 Y6 X4 G* M' I- G% O
    利用浏览器原生的API,封装成一套更加面向对象的Selenium WebDriver API,直接操作浏览器页面里的元素,甚至操作浏览器本身(截屏,窗口大小,启动,关闭,安装插件,配置证书之类的)8 l* D* U. b1 l
    8 [3 _, q2 Y3 v# V# ]
    5 M( f6 r! N6 F5 J; H
    selenium的安装以及简单使用, _3 j$ F  ]4 a4 Q9 n. K- L6 ]/ c3 j
    以edge浏览器为例 参见这个blog哦,驱动chrome浏览器同理4 G5 @% n7 e1 f
    selenium驱动edge浏览器
    . W% m4 k/ A, W% A' d1 s
    * C" K8 D. `+ j  @' w+ x
    . W& D1 d$ Q. h. j* @
    chromedriver环境的配置
    ( s7 C3 w+ j* `! u) k; j, o+ J% Y& twindows环境下需要将 chromedriver.exe 所在的目录设置为path环境变量中的路径4 g! a/ M" K5 R( U
    linux/mac环境下,将 chromedriver 所在的目录设置到系统的PATH环境值中1 n3 z, Q: [) h* g6 r
    selenium的简单使用
    , D( X$ n1 `! s4 w6 ~接下来我们就通过代码来模拟百度搜索
    ; R, n2 j$ ~4 x3 [9 [# Y
    3 \+ R/ r7 c& |1 G

    ) ?- J' G* c* I/ o  [  v# Cimport time* M* J2 K* U' @  Q  N" n
    from selenium import webdriver
    # A. \  |% q* r5 t; p& R( a1 R. C2 f. ^' p" p- U6 P

    # W/ F/ ~1 P  @2 C  [% e# 通过指定chromedriver的路径来实例化driver对象,chromedriver放在当前目录。7 r/ [( e$ q1 g% e& m3 x
    # driver = webdriver.Chrome(executable_path='./chromedriver')2 n9 E5 S( y7 e* a  n
    # chromedriver已经添加环境变量
    8 z. Q2 e. _  {driver = webdriver.Chrome(), \7 v! A9 E1 n8 I" n
    - r- v+ _. m/ j  {  |% X" ]
    ' W' F& P+ o  ]' f; u; J
    # 控制浏览器访问url地址
    ( F, _8 n' u& H# Y, ?/ gdriver.get("https://www.baidu.com/")
    8 `/ S9 S6 d) H" U
    & Q2 S( }# @8 E1 }; @

    % @% d; }% O3 y4 v; h) {# 在百度搜索框中搜索'python'
    7 d. T' K" y$ _0 }driver.find_element_by_id('kw').send_keys('python')7 f- t, |% r7 ]; @6 e1 m+ J6 n
    # 点击'百度搜索'
    & [# e4 s6 B1 Tdriver.find_element_by_id('su').click()2 w+ b7 u3 l8 h5 d$ ?
    3 g, P6 O0 C, o3 v4 [

    : q  Q4 x7 U/ ?7 Utime.sleep(6)
    ! P! O9 N- q2 E- i# 退出浏览器
    7 ^3 D9 ^2 a) `% f  t& x4 ddriver.quit()
    ; l( p# f3 z* H  c11 E1 c7 L& ~- F$ n
    2
    $ T# v( m% i) Y3
    ! @( L$ y: B8 p4
    ! F/ S# K; Q& F8 L# J( Q( x+ ~& h51 x" m7 g7 U  h) x- [8 p) _% l! I
    6
    0 h" p- q& f8 y) P7
    + n: B$ N0 z5 t+ n' b: w/ n8# T  }( J  x% \! v1 b8 f
    9# e1 R* A2 {8 B7 L" Q
    108 |1 O- w) l2 }6 d1 O+ ^
    11
    + c9 T% e8 L; G( [* C+ [12* _% H/ _) X5 }* |
    13; y/ w& C) ~2 b1 g" ~
    14
    7 r# V$ N$ L. G1 p1 e+ ?% X# f15
    * D* ?& ^5 U; i) j9 A9 \# C16
    $ A3 n7 u7 x3 k8 ?/ k1 a5 v17
    " H- B, L( r, ]' J5 {: c% R18
    0 p' m( Z" K  x" S/ H) h: R19
    ; O$ B: \7 h' a1 ]4 p, dwebdriver.Chrome(executable_path='./chromedriver')中executable参数指定的是下载好的chromedriver文件的路径9 h& ?! [" @4 T* o
    driver.find_element_by_id('kw').send_keys('python')定位id属性值是’kw’的标签,并向其中输入字符串’python’
    1 f) n- r. n( z! a* |driver.find_element_by_id('su').click()定位id属性值是su的标签,并点击. y6 D/ [. m  ~" m$ T% w
    click函数作用是:触发标签的js的click事件
    2 |1 S+ ]) b3 ^7 h7 q: a- C: B" ?值是’kw’的标签,并向其中输入字符串’python’
    7 f, b/ a: @8 V" y& i5 A' j" p% {0 y

    ' V( h. y  b5 T, R9 N: \: c' ydriver.find_element_by_id('su').click()定位id属性值是su的标签,并点击
    ( j( m  a$ V, I8 \1 |6 j' pclick函数作用是:触发标签的js的click事件$ l; d. j1 u/ q$ O2 I
    使用xpath来提取数据,爬取数据的简单语法。( C- y5 l9 F. `) Z. W

    . R: I* y/ Y& ~& S, ^

    - Z6 k+ Z* Q! c% slxml
    2 }) M0 p) }& t# H5 v% _
    3 A" ^8 t0 T7 u/ k+ o4 C; Z) n, K

    ' Y: Y4 M9 e9 b. Z6 z$ t0 o& A* t9 B8 Z

    ! _* L0 }& m- G0 drequests官方文档 https://lxml.de/
    2 C0 X. b! p0 i5 p0 ^1 c9 Z  c1 f; T4 A2 ?6 A" a; r8 W
    - `: j& ]& z1 P! [) L0 e& Z4 L
    pip install lxml0 c# z7 s( P( F, @( Y" }
    14 n" c) I7 ^2 `, V# X, p4 R
    导入模块1 h. o6 q$ A" s
    from lxml import etree
    ( x- V- h  ]: S- b& J1: O( a; |( g3 s* S( d7 C
    利用xpath获取text或者href内容: H+ [$ v  }) t
    /li/a/@href 这样取的应该是href的内容
    ; T/ e1 ~2 d  x# O6 ]+ g/li/a/text() 这样取得是text内容
    3 B* V- e- u4 `% q12 n* Z. i5 h$ w- d( ?5 X- {7 O( s7 O4 R# a
    2  c, V6 n, Z0 f8 ^  \" U
    etree的使用- t3 a  v% O3 f% B& }7 |5 H; Q& k
    h=etree.HTML(response.text)#response.text是网页的源码5 i2 A0 ~! ?8 G: x9 p5 N
    h.xpath('//img')  #寻找所有的img结点,9 e5 ?4 F, H$ b
    h.xpath('//div').xpath('.//img')#寻找所有div下的所有img结点+ y" B. n# V8 W. g5 H/ b7 F
    1
    7 h, Q/ Z# [1 U5 Z6 s2
    + y2 R7 e7 R% J# O$ p% b39 j- c0 u9 ]# A& @; d' P- L
    xpath的语法, Q7 t5 S% E: I& G) i+ K, h, t
    符号8 l+ Y/ |' v7 D7 h9 T9 S
    XPath 使用路径表达式在 XML 文档中选取节点。节点是通过沿着路径或者 step 来选取的。
    2 h1 @: }; k" C. m& |4 j9 F4 @6 C6 k4 Y' M7 L' B* b4 g
    7 E& S7 l5 N: k4 P$ \
    表达式        描述
    ; g3 g( N* I# L1 j9 j8 |. F: i* z/        从根节点选取. Y! e, a9 ~+ B) z/ g
    //        从匹配选择的当前节点选择文档中的节点,而不考虑它们的位置。
    2 h  ]- M5 Q9 H9 H.        选取当前节点。* X; e1 l. n# h! [
    . .        选取当前节点的父节点。( G. v, S( @, _: d6 p8 j
    @        选取属性。# S2 e9 E5 a' E. G3 I1 y
    |        在两个中结点中选择
    9 P2 `1 T6 z* Z7 m()        用()来包含|
    $ r; B  ]. s" W: s/ B*        包含所有元素
    , {8 O+ x1 C: A& ~1 L# z0 [not        取反
    " j5 l  B+ Y/ [1 p% ^* {实例3 @5 ?- h8 G/ P0 r# Q
    $ G5 ~3 m9 t4 A5 A! c1 D1 F

    # I: Z( v2 e# ]# {; c路径表达式        结果( b) W  P, ?8 @# F% W/ C$ j
    bookstore        选取 bookstore 元素的所有子节点。+ ]* F& R8 w  Q( Z' N
    /bookstore        选取根元素 bookstore。注释:假如路径起始于正斜杠( / ),则此路径始终代表到某元素的绝对路径!8 C! ]" U$ `" n( b& ^- s- t7 V
    bookstore/book        选取属于 bookstore 的子元素的所有 book 元素。0 j  V* s$ R; ~- o# Y$ x/ U. {
    //book        选取所有 book 子元素,而不管它们在文档中的位置。
    8 T5 @7 F! [3 Tbookstore//book        选择属于 bookstore 元素的后代的所有 book 元素,而不管它们位于 bookstore 之下的什么位置。
    ) }3 K& J' N7 s. g' V6 G' Z9 O//@lang        选取名为 lang 的所有属性。
    9 d' \: [" [% x  \# ]9 W6 W//*[@class]        选取带有class属性的所有元素$ w6 C% Q4 |4 S7 _- G
    //div[@*]        匹配任意属性的div元素) P# L- }1 ?, O4 \; l8 M
    //a[not(@class)]        匹配没有class属性的a元素% H* e& ~% `3 W' W! M. h
    谓语
    5 u' D, H3 W  e" P& O: A; Z" T带谓语的路径表达式& ]3 B% Z" u6 c9 H

      R: c; s9 W  v9 N

    & s9 X7 \: h: A, H7 G路径表达式        结果1 F' M. h% A; q/ U
    /bookstore/book[1]        选取属于 bookstore 子元素的第一个 book 元素。' E& Q/ \3 m! s; }1 w1 }* P
    /bookstore/book[last()]        选取属于 bookstore 子元素的最后一个 book 元素。4 y& H! C+ U* w$ }: O5 L5 o
    /bookstore/book[last()-1]        选取属于 bookstore 子元素的倒数第二个 book 元素。, |  x" p: F) ?( C2 H9 S7 a5 y  v
    /bookstore/book[position()< 3]        选取最前面的两个属于 bookstore 元素的子元素的 book 元素。; o- {" g. N; D
    //title[@lang]        选取所有拥有名为 lang 的属性的 title 元素。
    , p+ E0 m- C; n2 f1 Z//title[@lang=‘eng’]        选取所有 title 元素,且这些元素拥有值为 eng 的 lang 属性。- E# m2 T5 U# f. k
    /bookstore/book[price>35.00]        选取 bookstore 元素的所有 book 元素,且其中的 price 元素的值须大于 35.00。9 p% Q0 d# }; t! R
    /bookstore/book[price>35.00]/title        选取 bookstore 元素中的 book 元素的所有 title 元素,且其中的 price 元素的值须大于 35.00。2 F* ^1 I6 J$ L+ G6 E  Q: u9 X
    ————————————————5 b" f3 \. U% N! O8 `
    版权声明:本文为CSDN博主「北山啦」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。; R4 i, q" ]: ~
    原文链接:https://blog.csdn.net/qq_45176548/article/details/1181870680 o+ Q) Z7 u% k, V- O

    : Q. c6 D. D0 D% ~% }- R+ B, m+ H, n! o) L
    zan
    转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信

    0

    主题

    10

    听众

    299

    积分

    升级  99.5%

  • TA的每日心情
    开心
    2023-10-14 10:28
  • 签到天数: 28 天

    [LV.4]偶尔看看III

    回复

    使用道具 举报

    0

    主题

    10

    听众

    299

    积分

    升级  99.5%

  • TA的每日心情
    开心
    2023-10-14 10:28
  • 签到天数: 28 天

    [LV.4]偶尔看看III

    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-4-10 15:28 , Processed in 0.478933 second(s), 61 queries .

    回顶部