看示例,模拟12306登陆(需要用到自己的12306账号密码,和自己的超级鹰账号密码和软件id) selenium模块可以帮我我们便捷的爬取到页面中动态加载出来的数据
# 为了模拟人为登陆,需要阻塞程序 import time # 使预览器自动化的模块 from selenium import webdriver # 预览器自动化的动作链模块 from selenium.webdriver import ActionChains # 截取图片的模块(pip install Pillow) from PIL import Image # 引入超级鹰的软件,用于提交验证码,不用动 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 # 898175 self.base_params = { 'user': self.username, 'pass2': self.password, 'softid': self.soft_id, # 898175 } 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=r'./chromedriver.exe') # 打开的浏览器窗口最大化 bro.maximize_window() # 访问12306登陆页面 bro.get('https://kyfw.12306.cn/otn/resources/login.html') time.sleep(2) # 因为这个页面是扫码登陆,所以我们自动化点击账号登陆 bro.find_element_by_xpath('/html/body/div[2]/div[2]/ul/li[2]/a').click() # 定位到验证码图片的位置 yanzheng_addr = bro.find_element_by_xpath('/html/body/div[2]/div[2]/div[1]/div[2]/div[3]') # 得到验证码图片的坐标{'x': 0, 'y': 0} location = yanzheng_addr.location print('起始坐标location',location) # 得到验证码图片的尺寸,宽和高{'height': 188, 'width': 300} size = yanzheng_addr.size print('图片尺寸size',size) # 构建一个元组数据结构,截取验证图片的起始坐标,结束坐标.(图片左上角坐标和右下角坐标) tup = (location['x'], location['y'], location['x']+size['width'], location['y']+size['height']) # 截取这个预览器打开网页的整张图片,并给一个路径+文件名 bro.save_screenshot('./all.png') # 打开这个整张页面的图片 i = Image.open('./all.png') # 在整张页面中通过坐标(构建的tup元组数据)截取验证码图片的数据,并把验证码图片保存 img = i.crop(tup) img.save('./code.png') # 固定写法,实例化超级鹰对象,参数分别是超级鹰的(!!!用户名,密码,自己创建软件的ID(同时会生产密钥)) chaojiying = Chaojiying_Client('xuebaohua', '123456xbh', '898175') im = open('./code.png', 'rb').read() result = chaojiying.PostPic(im, 9004) # 9004验证码类型() print("返回的数据是一个字典result", result) # 取出['pic_str']的数据,是'78,104|212,114',就是符合验证码规则的坐标 result = result['pic_str'] # 准备一个列表,用来构建数据结构[[x1,y1],[x2,y2],[x3,y3]],里面存放的就是验证码符合条件的坐标 lis =[] # 判断符合条件的数量,最少有两个坐标的话,会有'|' if '|' in result: lis1 = result.split('|') for i in lis1: lis2 = i.split(',') lis.append(lis2) # 如果符合条件的只有一个的话(验证码只有一个符合条件),打印构建的lis数据 else: lis.append(result.split(',')) print(lis) # 得到要点击的坐标,然后以验证码图片位置()为全局坐标,依次点击符合条件的地方 for i in lis: # 得到了坐标,可以创建动作链,用来点击验证码符合条件的地方(这一步必须在列表循环的里面) action = ActionChains(bro) x = i[0] y = i[1] # 以验证图片为全局坐标,以x,y坐标为目标位置,点击该位置.必须用下面的方法. # perform()立即执行动作链 action.move_to_element_with_offset(yanzheng_addr,x,y).click().perform() time.sleep(2) # 点击完成之后,在标签id="J-userName"的input框中输入用户名 bro.find_element_by_id('J-userName').send_keys('!!!自己12306登录名') time.sleep(2) # 在标签id="J-password"的input框中输入用户名 bro.find_element_by_id('J-password').send_keys('!!!自己12306密码') time.sleep(2) # 点击登陆按钮 bro.find_element_by_xpath('//*[@id="J-login"]').click() # time.sleep(2) # bro.quit()
处理无头浏览器的代码
from selenium import webdriverfrom lxml import etreeimport time # 无头浏览器,不会显示浏览器from selenium.webdriver.chrome.options import Optionschrome_options = Options()chrome_options.add_argument('--headless')chrome_options.add_argument('--disable-gpu')bro = webdriver.Chrome(executable_path='./chromedriver.exe',chrome_options=chrome_options)#让浏览器对指定url发起访问bro.get('http://125.35.6.84:81/xk/')#获取浏览器当前打开页面的页面源码数据(可见即可得)page_text = bro.page_sourcetime.sleep(2)tree = etree.HTML(page_text)# 药监局动态加载的公司名称,取第一个,打印一下name = tree.xpath('//*[@id="gzlist"]/li[1]/dl/a/text()')[0]print(name)time.sleep(3)bro.quit()