QQ音乐MP3下载
qq音乐mp3下载
本篇文章涉及内容包括:python,爬虫,json解析,request 库的使用
缘起
前几天刷b站无意中又刷到了一首神曲,“i want my tears back”,挺好听的。听了几遍后便寻思着能不能把这歌给下到手机上拿来当闹钟的,听过的同学应该知道这歌有多提神,哈哈哈~~~
没听过的同学可以点击文章上方的音乐,感受一下下。
动手分析
接下来,当然要选择一下从哪个平台下搞这首歌回来啦。网易云音乐和qq音乐,选择哪个?那就从网易云入手吧,毕竟用的多些,接着便去网易云一顿操作,此处省略1000字描述。发现,哎呦,这网易云……不好搞呀。所以先不管了,看一下qq音乐的情况先。
f12大法一开,qq音乐就先给我来了个惊喜,大大符号图标倾情相送。对比云村的就没有啦。
撇开这些不关键的东西不说了,接下来就是搜索一首歌。
这个时候,先把控制台切到network栏,这个时候,你会发现左下角有很多请求链接。请求各种各样的内容,这个时候,这些东西对我们都是没用的,是吧。我们要的是我们请求搜索那一瞬间他发出去的数据。所以,为了避免无关数据的干扰,建议点击左上角的清空把记录清空先。然后右上角勾上disable cache,避免缓存使我们看不到我们要的数据。
ok,这个时候我们输入我们要查的歌,i want my tears back。我们已经可以看到有东西已经发出去并返回回来了。就是这个链接
我们观察一下,这里面的参数,key就有i want my tears back这个歌名了,再看一下返回数据。是一个json数据,有很多内,我们可以很容易看出了数据包含了 专辑、mv、歌手、歌曲 这四大区域。我们关注歌曲这个栏,在count 中已经表明找到了两首符合度最高的歌曲。一首是nightwish的一首是hok-key的。因为我想要的就是 夜愿 这个版本的,所以我们就默认第一首出现的就是我们要下的歌曲哈。
这个时候,拿到这些数据有什么用呢?歌曲链接还没不知道怎么构造呀。。。
先别急,我们看看歌曲链接长啥样。然后…………
好尴尬呀,有木有,这,这谁忍得住呀?所以,机智的我打开了另外一首歌……
瞬间右边记录出来了一大堆,没错是一大堆东西……但是,控制台有分类呀,不怂。选择media分类,就给我们过滤出了那几个链接。
没办法了,逐个点击。1分钟过后,i got it.最后那个链接就是歌曲链接。
可以清楚看到有几个关键参数。guid、vkey、uin、fromtag,我们先暂且不管那个参数可以省略好吧。先看看可以从哪里找到这几个参数……又得看链接的返回值了(此处说明,可以在控制台点击response标签查看返回值,你只要按↓键就ok了)。在漫长的遍历过程中,我注意到了一个链接的内容十分不一样的,不相同的,很多很多。直觉告诉我,就是这个链接了。
拿去做一下url解码后,咋一看一头雾水。不信你看一下下?
接合歌曲链接,这个请求链接,搜索请求结果,这三个内容。其实不难发现,songmid 这个参数是关键。
所以,我们这个时候基本就以及厘清下载音乐的步骤以及思路了。首先,请求搜索某一首歌,然后或者到song mid 接着song mid 去请求服务器拿到这首歌的播放链接。接着用request请求数据回来再用二进制保存就ok啦。
代码写起来
导入需要用到的json库和requests库。
构造一个类,downloadmusic,初始化一个headers,做个最简单的反爬。
写run方法,将上面的思路实现出来。请求回数据后就用json.loads方法加载json数据。逐步逐步请求服务器,填充需要填的数据。
最后拿到链接的url后,就easy了,用requests请求资源回来后。用.content,不要用.text。又人问过我这两个方法的区别。
简单来说,content拿到的数据是字节,调试打印出来你会发现数据前面会有个b',text的话,就是一个字符串了。
因为我们要保存歌曲,就肯定要用字节保存好,用wb方式打开,然后写进去后便关闭即可。代码贴在下面。
至此写完收工,可以美滋滋地下载我们要的歌了。有了它还怕什么铃声找不到自己喜欢的问题吗?
# -*- coding: utf-8 -*- # @time : 10/10/2018 9:31 pm # @author : marx·cbr # @file : __init__.py import requests import json class downloadmusic: def __init__(self): self.headers = { 'accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8', 'accept-encoding': 'gzip, deflate', 'accept-language': 'zh-cn,zh;q=0.8,en-us;q=0.5,en;q=0.3', 'upgrade-insecure-requests': '1', 'user-agent': 'mozilla/5.0 (windows nt 10.0; win64; x64; rv:55.0) gecko/20100101 firefox/55.0', } self.name='i want my tears back' def run(self,sn): self.name=sn session=requests.session() firstjsonurl='https://c.y.qq.com/splcloud/fcgi-bin/smartbox_new.fcg?is_xml=0&format=jsonp&key={}&g_tk=5381&jsonpcallback=smartboxkeyscallbackmod_top_search1467&loginuin=0&hostuin=0&format=jsonp&incharset=utf8&outcharset=utf-8¬ice=0&platform=yqq&neednewcode=0'.format(self.name) r=session.get(firstjsonurl).text print(type(r)) print(r[39:-1:]) myjson=json.loads(r[39:-1:]) mid=myjson['data']['song']['itemlist'][0]['mid'] print(mid) searchurl='''https://u.y.qq.com/cgi-bin/musicu.fcg?callback=getplaysongvkey2236996910208997&g_tk=5381&jsonpcallback=getplaysongvkey2236996910208997&loginuin=0&hostuin=0&format=jsonp&incharset=utf8&outcharset=utf-8¬ice=0&platform=yqq&neednewcode=0&data={"req":{"module":"cdn.srfcdndispatchserver","method":"getcdndispatch","param":{"guid":"8665097290","calltype":0,"userip":""}},"req_0":{"module":"vkey.getvkeyserver","method":"cgigetvkey","param":{"guid":"8665097290","songmid":["'''+mid+'''"],"songtype":[0],"uin":"0","loginflag":1,"platform":"20"}},"comm":{"uin":0,"format":"json","ct":20,"cv":0}}''' r=session.get(searchurl).text print(r) songjson=json.loads(r[32:-1:]) print(songjson) header=songjson['req_0']['data']['sip'][0] two=songjson['req_0']['data']['midurlinfo'][0]['purl'] songurl=header+two with open("{}.mp3".format(self.name),'wb') as ms: print(songurl) raw = session.get(songurl, headers=self.headers) content=raw.content if len(content) >500: ms.write(content) print("下载成功") else: print("下载失败") app=downloadmusic() while 1: songname=input("请输入歌曲名字") app.run(songname)
欢迎各位关注我的微信公众号:叙之亭
上一篇: [20181130]hash冲突导致查询缓慢.txt
下一篇: DI是实现面向切面和面向抽象的前提