Selenium模块
selenium模块和爬虫之间有什么关联?
- 便捷的获取网站中动态加载的数据
- 便捷实现模拟登陆
什么是selenium模块?
- 基于浏览器自动化的一个模块
selenium使用流程:
- 环境安装: pip install selenium
- 下载一个浏览器的驱动程序(谷歌):
- 下载路径:http://chromedriver.storage.googleapis.com/index.html
- 驱动程序和浏览器版本的对应关系:http://blog.csdn.net/huilan_same/article/details/51896672
- 实例化一个浏览器对象
- 编写基于浏览器自动化的操作代码
- 发起请求:get(url)
- 标签定位:find系列的方法
- 标签交互:send_key('xxx')
- 执行js程序:excute_script('jsCode')
- 前进,后退:back(),forward()
- 关闭浏览器:quit()
# -*- coding: utf-8 -*-
from selenium import webdriver
from time import sleep
bro = webdriver.Chrome(executable_path='./chromedriver')
#.get()相当于是让浏览器输入一个网址
bro.get('https://www.taobao.com')
#标签定位
#找到搜索框标签(在一个id为q的div中)
search_input = bro.find_element_by_id('q')
#标签交互(相当于在搜索框中输入要搜索的东西)
search_input.send_keys('Iphone')
#执行一组js代码,让浏览器自动滚动页面
bro.execute_script('window.scrollTo(0,document.body.scrollHeight)')
#document.body.scrollHeight--滚动屏幕大小的高度
sleep(2)
#点击搜索按钮
btn = bro.find_element_by_css_selector('.btn-search')#根据选择器找
btn.click()#点击
bro.get('https://www.baidu.com')
sleep(2)
#回退
bro.back()
sleep(2)
#前进
bro.forward()
sleep(5)
- selenium处理iframe(嵌套子窗口)
- 如果定位的标签存在于iframe标签中,则必须使用swith_to.frame(id)
- 动作链(拖动):from selenium.webdriver import ActionChains
- 实例化一个动作链对象:action = ActionChains(bro)
- click_and_hold(div) : 长按且点击操作
- move_by_offset(x,y)
- perform()让动作链立即执行
- action.release()释放动作链对象
# -*- coding: utf-8 -*-
from time import sleep
from selenium import webdriver
from selenium.webdriver import ActionChains#动作链
bro = webdriver.Chrome(executable_path='./chromedriver')
bro.get('http://www.runoob.com/try/try.php?filename=jqueryui-api-droppable')
#如果定位的标签是存在于iframe标签之中的必须通过如下操作进行标签定位
bro.swithch_to.frame('iframeResult')#切换到对应iframe标签定位的作用域
div = bro.find_element_by_id('draggable')
#动作链
action = ActionChains(bro)
#点击长按指定的标签(click_and_hold)
action.click_and_hold(div)
for i in range(5):
#perform()--立即执行动作链操作
#move_by_offset(x,y):x-水平方向,y-竖直方向
action.move_by_offset(17,0).perform()
sleep(0.5)#慢慢拖动
#释放动作链
action.release()
bro.quit()
模拟登陆案例:
# -*- coding: utf-8 -*-
from selenium import webdriver
from time import sleep
bro = webdriver.Chrome(executable_path='./chromedriver')
bro.get('https://qzone.qq.com/')
#由于刚弹出来的是扫码登陆,所以要点击一下账号密码登录才可以跳转
bro.switch_to_frame('login_frame')
#‘账号密码登录按钮’
a_tag = bro.find_element_by_id("switcher_plogin")
a_tag.click()
sleep(2)
#输入账号密码
userName_tag = bro.find_element_by_id('u')
passName_tag = bro.find_element_by_id('p')
userName_tag.send_keys('154919633')
sleep(2)
passName_tag.send_keys('*******')
#点击登录按钮
btn = bro.find_element_by_id('login_button')
btn.click()
sleep(3)
btn.quit()
无头浏览器:让谷歌浏览器不可视化(驱动不会弹窗口)
# -*- coding: utf-8 -*-
from selenium import webdriver
from time import sleep
#实现无可视化界面
from selenium.webdriver.chrome.options import Options
#实现规避检测
from selenium.webdriver import ChromeOptions
#实现无可视化界面的操作
chrome_options = Options()
chrome_options.add_argument('--headless')
chrome_options.add_argument('--disable-gpu')
#实现规避检测
option = ChromeOptions()
option.add_experimental_option('excludeSwitches',['enable-auto-automation'])
#如何实现让selenium规避被检测的风险,且无头化
bro = webdriver.Chrome(executable_path='./chromedriver',chrome_options=chrome_options,options=option)
bro.get('https://www.baidu.com')
print(bro.page_source)
sleep(2)
bro.quite()
模拟12306登录案例(点击型验证码)
from selenium import webdriver
from PIL import Image#图片工具类
from time import sleep
from selenium.webdriver import ActionChains#动作链
import requests
from hashlib import md5
class Chaojiying_Client(object):
def __init__(self, username, password, soft_id):
self.username = username
password = password.encode('utf8')
self.password = md5(password).hexdigest()
self.soft_id = soft_id
self.base_params = {
'user': self.username,
'pass2': self.password,
'softid': self.soft_id,
}
self.headers = {
'Connection': 'Keep-Alive',
'User-Agent': 'Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.1; Trident/4.0)',
}
def PostPic(self, im, codetype):
"""
im: 图片字节
codetype: 题目类型 参考 http://www.chaojiying.com/price.html
"""
params = {
'codetype': codetype,
}
params.update(self.base_params)
files = {
'userfile': ('ccc.jpg', im)}
r = requests.post('http://upload.chaojiying.net/Upload/Processing.php', data=params, files=files, headers=self.headers)
return r.json()
def ReportError(self, im_id):
"""
im_id:报错题目的图片ID
"""
params = {
'id': im_id,
}
params.update(self.base_params)
r = requests.post('http://upload.chaojiying.net/Upload/ReportError.php', data=params, headers=self.headers)
return r.json()
bro = webdriver.Chrome(executable_path='./chromedriver')
bro.get('https://kyfw.12306.cn/otn/login/init')
sleep(1)
'''
经过观察,网站的验证码是点击图片的,且每次请求都会刷新
因此不能直接获取src请求图片。
思路:将整张页面截图下来,在通过标签找到验证码图片位置,分割下来识别
'''
#save_screenshot就是将当前页面进行截图并保存
bro.save_screenshot('aa.png')
#确定验证码图片对应的左上角和右下角坐标,从而确定裁剪区域
#按路径拿到验证码图片
code_img_ele = bro.find_element_by_xpath('//*[@id="loginForm"]/div/ul[2]/li[4]/div/div/div[3]/img')
location = code_img_ele.location #返回验证码图片左上方的坐标 x,y
print('location:',location)
size = code_img_ele.size #返回验证码图片长,宽
#定义一个点,将左上角和右下角的坐标存进去
rangle = (int(location['x']),int(location['y']),int(location['x']+size['width']),int(location['y']+size['height']))
#至此,已经定义好了验证码区域
i = Image.open('./aa.png')
code_img_name = './code.png'
#crop--根据指定区域进行图片裁剪
frame = i.crop(rangle)
frame.save(code_img_name)
chaojiying = Chaojiying_Client('DIAJEY', 'abc1314', '910539') #用户中心>>软件ID 生成一个替换 96001
im = open('code.png', 'rb').read() #本地图片文件路径 来替换 a.jpg 有时WIN系统须要//
print(chaojiying.PostPic(im, 9004)['pic_str'])
sleep(3)
#返回了坐标后,创建动作链去点击
#首先先将点获取出来,存到列表中
result = chaojiying.PostPic(im, 9004)['pic_str']
all_list = []
if '|' in result:#如果有|表示返回多个坐标,分割后再存进去
list_l = result.split('|')
count_l = len(list_l)
for i in range(count_l):
xy_list = []
x = int(list_l[i].split(',')[0])#以,为分割,左边为x,右边为y
y = int(list_l[i].split(',')[1])
xy_list.append(x)
xy_list.append(y)
all_list.append(xy_list)
else:
x = int(result.split(',')[0])#以,为分割,左边为x,右边为y
y = int(result.split(',')[1])
xy_list = []
xy_list.append(x)
xy_list.append(y)
all_list.append(xy_list)
#遍历坐标列表,通过动作链对每一个列表元素对应的坐标进行点击操作
for l in all_list:
x = l[0]
y = l[1]
ActionChains(bro).move_to_element_with_offset(code_img_ele,x,y).click().perform()
#因为x,y是相对于验证码的位置,所以动作链要先移动到验证码区域再开始点击操作
sleep(0.5)
#验证码识别成功,输入账号密码,然后点击登录
bro.find_element_by_id('username').send_keys('xxxxxx')
sleep(2)
bro.find_element_by_id('password').send_keys('xxxxxx')
sleep(2)
sleep(10)
bro.quit()