用python给自己做一款小说阅读器过程详解


Posted in Python onJuly 11, 2019

前言

前一段时间书荒的时候,在喜马拉雅APP发现一个主播播讲的小说-大王饶命。听起来感觉很好笑,挺有意思的,但是只有前200张是免费的,后面就要收费。一章两毛钱,本来是想要买一下,发现说的进度比较慢而且整本书要1300多张,算了一下,需要200大洋才行,而且等他说完,还不知道要到什么时候去。

所以就找文字版的来读,文字版又有它的缺点,你必须手眼联动才行。如果要忙别的事情,但是又抑制不住想看的冲动,就很纠结了。在网上找了一圈,没有其他的音频。而且以前用的那些有阅读功能的软件,比如微信阅读、追书神器也都开始收费了。

那怎么办呢?这能难倒一个学Python的程序员吗?必须滴、坚决滴不能。我用的可是世界上最好的编程语言-Python,很多伙伴也学了不少Python编程学习教程,不妨这里一起操练一下?

用python给自己做一款小说阅读器过程详解

自己动手丰衣足食,接下来就用我们学过的Python编程学习教程实现自己的小说阅读器吧。

语音合成选择

要想读文字,就必须要用到语音合成。现在这种语音合成的软件有很多,其中讯飞和百度是比较好的两种,我们这里就使用百度语音合成API来实现。

创建语音合成应用

首先注册百度账号,然后登录到百度AI开放平台,创建一个应用

用python给自己做一款小说阅读器过程详解

用python给自己做一款小说阅读器过程详解

填写应用名和描述信息提交

用python给自己做一款小说阅读器过程详解

记住AppID、API Key、Secret Key,在使用API的时候会用到,查看一下技术文档

),使用pip install baidu-aip安装完API,文档内有详细的示例代码,很容易就上手了。里面有各种参数解释,比如音量、语调、语速、发声人等。现在语音合成已经有了,已经有了阅读的前提,下面就是获取小说内容了。

获取小说内容

小说内容的获取我们从笔趣阁网站上获取,一方面免费,另一方面没有反爬,找到网站首页

https://www.biquge.info/40_40289/,使用requests大法就可以了。简单分析一下页面

用python给自己做一款小说阅读器过程详解

所有章节信息都在dd元素下,而且链接也是很有规律的,直接用xpath获取所有章节列表信息。

def get_chapters(self):
 url = "https://www.biquge.info/40_40289/"
 r = self.session.get(url)
 r.encoding = chardet.detect(r.content).get("encoding", "utf-8")
 html = etree.HTML(r.text)
 for item in html.xpath("//dl/dd/a"):
 yield item.attrib["title"], url + item.attrib["href"]

章节内容获取也非常简单,就不分析了

def get_content(self, url):
 r = self.session.get(url)
 r.encoding = chardet.detect(r.content).get("encoding", "utf-8")
 html = etree.HTML(r.text)
 title = html.xpath(r'//*[@class="bookname"]/h1')[0].text
 for info in html.xpath("//div[@id='content']"):
 text = info.xpath("string(.)")

这里有一点要注意的,获取的章节内容中有html元素,xpath为我们提供了string(.),提取多个子节点的文本,非常好用。

合成存储

小说内容获取成功了,与语音合成结合一下,小说阅读器的雏形就有了。简单实现如下:

import chardet
import requests
from lxml import etree
from aip import AipSpeech
class CollectNovels:
 def __init__(self):
 self.session = requests.session()
 self.session.headers["user-agent"] = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.169 Safari/537.36"
 """ 你的 APPID AK SK """
 APP_ID = '16416498'
 API_KEY = 'oEWGafQkaUGqmsmPbfkE5OMx'
 SECRET_KEY = '6jdsUcH0PXz5TYoELU47u58W5vPV9lwf'
 self.client = AipSpeech(APP_ID, API_KEY, SECRET_KEY)
 def get_chapters(self, url):
 r = self.session.get(url)
 r.encoding = chardet.detect(r.content).get("encoding", "utf-8")
 html = etree.HTML(r.text)
 for item in html.xpath("//dl/dd/a"):
 yield item.attrib["title"], url + item.attrib["href"]
 def get_content(self, url):
 r = self.session.get(url)
 r.encoding = chardet.detect(r.content).get("encoding", "utf-8")
 html = etree.HTML(r.text)
 for info in html.xpath("//div[@id='content']"):
 text = info.xpath("string(.)")
 for line in text.split("。"):
 content = self.client.synthesis(line, 'zh', 1, {"per": 0})
 with open("auido.mp3", "rb") as fp:
 fp.write(content)
if __name__ == '__main__':
 novel = CollectNovels()
 home_url = "https://www.biquge.info/40_40289/"
 for title, url in novel.get_chapters(home_url):
 novel.get_content(url)

这里是生成了mp3文件,按行生成以后,再使用合成软件合成后,我们就可以放在任意地方去听了。但是这样也有缺陷,必须提前生成,然后才能使用播放器听,这样不是很方便。如果可以边生成边播放是不是更好呢?

播放合成语音

我们可以使用python的pygame库,其他的好几个库都不太好用,有些已经年久失修了,所以就不用了。

import time
import pygame
from io import BytesIO
pygame_mixer = pygame.mixer
pygame_mixer.init(frequency=frequency)
byte_obj = BytesIO()
byte_obj.write(content)
byte_obj.seek(0, 0)
pygame_mixer.music.load(byte_obj)
pygame_mixer.music.play()
while pygame_mixer.music.get_busy():
 time.sleep(0.1)
pygame_mixer.stop()

这里使用BytesIO将语音合成的二进制文件存储在内存中,就不需要再保存成本地mp3了。

有一个需要注意的地方pygame_mixer.init(frequency=frequency),这个frequency参数是音频频率,如果不设置的话默认是22050,播放出来的声音和mp3播放相差太大了,一直以为是这个库有问题,换了好几个,有的是用不了,有的有问题,后来我才发现需要设置这个参数,那么这个参数从哪里来呢?查看之前生成的mp3文件属性

用python给自己做一款小说阅读器过程详解

然后将频率设置为16000就可以了。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
windows下python连接oracle数据库
Jun 07 Python
Python实现简易Web爬虫详解
Jan 03 Python
Python callable()函数用法实例分析
Mar 17 Python
python文件操作之批量修改文件后缀名的方法
Aug 10 Python
Python实现简易过滤删除数字的方法小结
Jan 09 Python
django-rest-framework 自定义swagger过程详解
Jul 18 Python
K最近邻算法(KNN)---sklearn+python实现方式
Feb 24 Python
Pytorch中的自动求梯度机制和Variable类实例
Feb 29 Python
Pytorch十九种损失函数的使用详解
Apr 29 Python
Python实现动态循环输出文字功能
May 07 Python
python进度条显示之tqmd模块
Aug 22 Python
python中openpyxl和xlsxwriter对Excel的操作方法
Mar 01 Python
Python 200行代码实现一个滑动验证码过程详解
Jul 11 #Python
ML神器:sklearn的快速使用及入门
Jul 11 #Python
python 随机森林算法及其优化详解
Jul 11 #Python
python实现从本地摄像头和网络摄像头截取图片功能
Jul 11 #Python
python常用库之NumPy和sklearn入门
Jul 11 #Python
python在新的图片窗口显示图片(图像)的方法
Jul 11 #Python
Python实现K折交叉验证法的方法步骤
Jul 11 #Python
You might like
制作美丽的拉花
2021/03/03 冲泡冲煮
[FAQ]PHP中的一些常识:类篇
2006/10/09 PHP
在PHP中执行系统外部命令
2006/10/09 PHP
php解析html类库simple_html_dom(详细介绍)
2013/07/05 PHP
php防止CC攻击代码 php防止网页频繁刷新
2015/12/21 PHP
Nigma vs Alliance BO5 第三场2.14
2021/03/10 DOTA
导航跟随滚动条置顶移动示例代码
2013/09/11 Javascript
巧用jquery解决下拉菜单被Div遮挡的相关问题
2014/02/13 Javascript
jQuery实现异步获取json数据的2种方式
2014/08/29 Javascript
JavaScript使用concat连接数组的方法
2015/04/06 Javascript
jQuery中next方法用法实例
2015/04/24 Javascript
JavaScript实现图片DIV竖向滑动的方法
2015/04/25 Javascript
分享使用AngularJS创建应用的5个框架
2015/12/05 Javascript
js实现获取两个日期之间所有日期的方法
2016/06/17 Javascript
Vue.js每天必学之组件与组件间的通信
2016/09/08 Javascript
vue1.0和vue2.0的watch监听事件写法详解
2018/09/11 Javascript
Mint UI实现A-Z字母排序的城市选择列表
2018/12/28 Javascript
Angular8 Http拦截器简单使用教程
2019/08/20 Javascript
swiper实现异形轮播效果
2019/11/28 Javascript
解决Vue的文本编辑器 vue-quill-editor 小图标样式排布错乱问题
2020/08/03 Javascript
js实现移动端轮播图滑动切换
2020/12/21 Javascript
Python中实现结构相似的函数调用方法
2015/03/10 Python
python 读入多行数据的实例
2018/04/19 Python
python 查找文件名包含指定字符串的方法
2018/06/05 Python
Python多进程方式抓取基金网站内容的方法分析
2019/06/03 Python
用Python实现二叉树、二叉树非递归遍历及绘制的例子
2019/08/09 Python
Pandas将列表(List)转换为数据框(Dataframe)
2020/04/24 Python
国际化的太阳镜及太阳镜配件零售商:Sunglass Hut
2016/07/26 全球购物
新加坡第一大健康与美容零售商:屈臣氏新加坡(Watsons Singapore)
2020/12/11 全球购物
卫生标语大全
2014/06/21 职场文书
2014年扫黄打非工作总结
2014/12/03 职场文书
营运督导岗位职责
2015/04/10 职场文书
公司晚会主持词
2019/04/17 职场文书
Python Pandas pandas.read_sql函数实例用法
2021/06/21 Python
Python中22个万用公式的小结
2021/07/21 Python
CSS Transition通过改变Height实现展开收起元素
2021/08/07 HTML / CSS