Python爬虫-QQ音乐下载(详解)

前言

一直想下点歌,今天就对QQ音乐下手了,分析了半天的地址,发现了规律,但就是找不到有关信息,所以只能请教大神,在大神那我找到了歌曲的API地址,一下解决了我的难题,所以我这次的成功也是站在巨人的肩膀上。

API奉上:

https://u.y.qq.com/cgi-bin/musicu.fcg?format=json&data=%7B%22req_0%22:%7B%22module%22:%22vkey.GetVkeyServer%22,%22method%22:%22CgiGetVkey%22,%22param%22:%7B%22guid%22:%22358840384%22,%22songmid%22:%5B%22001X0PDf0W4lBq%22%5D,%22songtype%22:%5B0%5D,%22uin%22:%221443481947%22,%22loginflag%22:1,%22platform%22:%2220%22%7D%7D,%22comm%22:%7B%22uin%22:%2218585073516%22,%22format%22:%22json%22,%22ct%22:24,%22cv%22:0%7D%7D

我们需要注意的就是API中加粗的部分,每首歌曲都有不同的songmid,输入哪首歌曲的songmid就可以得到哪首歌曲的下载地址,所以我们要做的就是找到歌曲的songmid。

1.搜索歌曲,获取歌曲的songmid

我们先打开控制台,在搜索框中输入我们想听的歌曲,按回车后会看到一个文件

在这里插入图片描述

打开后,有会看到有一个列表

在这里插入图片描述

我们可以发现,列表中的项数和我们搜索到的歌曲的数量一致,其实列表中的每一项对应的就是我们搜索到的每首歌曲,然后我们打开一项,查看详细信息

在这里插入图片描述

可以发现有一个mid的属性,试了一下,将他带入到API中,确实得到了这首歌曲的下载地址,所以这就是我们想要的。

2.获取下载地址

找到歌曲的songmid后,将songmid带入到API中,在网页中打开后看到以下信息
在这里插入图片描述

这里我们发现不了什么,所以在控制台中查看

在这里插入图片描述

在这里插入图片描述
我们可以查看每首歌曲的下载地址
在这里插入图片描述

我们可以发现,歌曲的播放地址为:

https://isure.stream.qqmusic.qq.com/ + purl

所以我们可以遍历我们搜索的歌曲列表中的每首歌曲,以此找到每首歌曲的下载地址,就可以实现将我们搜索的歌曲批量下载,代码部分:

    music_list = dict_html["data"]["song"]["list"]  #获取搜索到的所有歌曲的列表
for music in music_list:
    per_songmid = music["mid"]  #歌曲的songmid
    per_songname = music["name"]  #歌曲名称
    singer = music["singer"][0]["name"]  #歌手名称
    music_document_url_part = "https://u.y.qq.com/cgi-bin/musicu.fcg?format=json&data=%7B%22req_0%22%3A%7B%22module%22%3A%22vkey.GetVkeyServer%22%2C%22method%22%3A%22CgiGetVkey%22%2C%22param%22%3A%7B%22guid%22%3A%22358840384%22%2C%22songmid%22%3A%5B%22{}%22%5D%2C%22songtype%22%3A%5B0%5D%2C%22uin%22%3A%221443481947%22%2C%22loginflag%22%3A1%2C%22platform%22%3A%2220%22%7D%7D%2C%22comm%22%3A%7B%22uin%22%3A%2218585073516%22%2C%22format%22%3A%22json%22%2C%22ct%22%3A24%2C%22cv%22%3A0%7D%7D".format(per_songmid)
    music_document_html_json = requests.get(music_document_url_part).text
    music_document_html_dict = json.loads(music_document_html_json)  #将文件从json格式转化为字典格式
    music_url_part = music_document_html_dict["req_0"]["data"]["midurlinfo"][0]["purl"]  #歌曲下载地址的后部分
    if music_url_part == "":  #有的没有版权的歌曲应该是没有地址的 所以为空的话 就直接重新循环下一首就行了
        continue
    music_url = "https://isure.stream.qqmusic.qq.com/" + music_url_part  #歌曲完整下载地址

3.自动搜索歌曲

我们查看第一张图片中文件的地址
在这里插入图片描述
地址:https://c.y.qq.com/soso/fcgi-bin/client_search_cp?ct=24&qqmusic_ver=1298&new_json=1&remoteplace=txt.yqq.song&searchid=55043609442258168&t=0&aggr=1&cr=1&catZhida=1&lossless=0&flag_qc=0&p=1&n=10&w=%E5%A4%8F%E5%A4%A9%E7%9A%84%E9%A3%8E&g_tk_new_20200303=5381&g_tk=5381&loginUin=229554158&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq.json&needNewCode=0

只是地址进行编码了,我们解码看看

解码后:

https://c.y.qq.com/soso/fcgi-bin/client_search_cp?ct=24&qqmusic_ver=1298&new_json=1&remoteplace=txt.yqq.song&searchid=55043609442258168&t=0&aggr=1&cr=1&catZhida=1&lossless=0&flag_qc=0&p=1&n=10&w=夏天的风&g_tk_new_20200303=5381&g_tk=5381&loginUin=229554158&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq.json&needNewCode=0

我们可以发现里面的关键字,就是加粗的部分,我们可以改为我们想要搜索的歌曲名,访问后就可以得到所有歌曲。
在这里插入图片描述
所以我们可以通过改变关键字的方式来实现自动搜索我们想要找的歌曲,代码部分:

search_url = "https://c.y.qq.com/soso/fcgi-bin/client_search_cp?ct=24&qqmusic_ver=1298&new_json=1&remoteplace=txt.yqq.center&searchid=52176551238941872&t=0&aggr=1&cr=1&catZhida=1&lossless=0&flag_qc=0&p=1&n=10&w={}&g_tk_new_20200303=5381&g_tk=5381&loginUin=229554158&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq.json&needNewCode=0".format(name)
json_html = requests.get(search_url).text  #json格式转化为字典格式 转化的类型必须是字符串型
dict_html = json.loads(json_html)

4.完整代码

前面主要是分析一下,现在将完整代码奉上,基本都有注释,结合上面每段代码所实现的功能就很容易看懂了。

实现的功能主要是:将我们搜索的歌曲的所有版本都下载下来(其中有的歌曲没有版权应该是没办法下载,见谅)

def allmusic(name):
    headers = {"Cookie": "eas_sid=h1u5a6D7N0u089l4P8g4S1o9N5; pgv_pvi=1399294976; pgv_pvid=3522250322; RK=RaCE+Qw87C; ptcz=645f699f302818f2aa4fb78a5bf7f75fd0d90f1c95db5a5d0acb695283fdc8b7; tvfe_boss_uuid=306cd91a9af78b43; LW_uid=11f5K7a8I9K0K2E7d6G2B4q7P1; LW_sid=K1o518v1X5U1h0o5J6i4x9g0n9; o_cookie=229554158; pac_uid=1_229554158; uin_cookie=o0229554158; ied_qq=o0229554158; ptui_loginuin=2863778213; luin=o0229554158; lskey=00010000161fb2266a89bdbedc615f48e2b88d65a280a1be4207429dee5c429e0fcb7e0da641297e832f0555",
               "User-Agent": "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.108 Safari/537.36"}
    search_url = "https://c.y.qq.com/soso/fcgi-bin/client_search_cp?ct=24&qqmusic_ver=1298&new_json=1&remoteplace=txt.yqq.center&searchid=52176551238941872&t=0&aggr=1&cr=1&catZhida=1&lossless=0&flag_qc=0&p=1&n=10&w={}&g_tk_new_20200303=5381&g_tk=5381&loginUin=229554158&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq.json&needNewCode=0".format(name)
    json_html = requests.get(search_url).text  #json格式转化为字典格式 转化的类型必须是字符串型
    dict_html = json.loads(json_html)
    file_address = "D:/Pycharm文件/网络爬虫/QQ音乐"
    music_list = dict_html["data"]["song"]["list"]  #获取搜索到的所有歌曲的列表
    for music in music_list:
        per_songmid = music["mid"]  #歌曲的songmid
        per_songname = music["name"]  #歌曲名称
        singer = music["singer"][0]["name"]  #歌手名称
        music_document_url_part = "https://u.y.qq.com/cgi-bin/musicu.fcg?format=json&data=%7B%22req_0%22%3A%7B%22module%22%3A%22vkey.GetVkeyServer%22%2C%22method%22%3A%22CgiGetVkey%22%2C%22param%22%3A%7B%22guid%22%3A%22358840384%22%2C%22songmid%22%3A%5B%22{}%22%5D%2C%22songtype%22%3A%5B0%5D%2C%22uin%22%3A%221443481947%22%2C%22loginflag%22%3A1%2C%22platform%22%3A%2220%22%7D%7D%2C%22comm%22%3A%7B%22uin%22%3A%2218585073516%22%2C%22format%22%3A%22json%22%2C%22ct%22%3A24%2C%22cv%22%3A0%7D%7D".format(per_songmid)
        music_document_html_json = requests.get(music_document_url_part).text
        music_document_html_dict = json.loads(music_document_html_json)  #将文件从json格式转化为字典格式
        music_url_part = music_document_html_dict["req_0"]["data"]["midurlinfo"][0]["purl"]  #歌曲下载地址的后部分
        if music_url_part == "":  #有的没有版权的歌曲应该是没有地址的 所以为空的话 就直接重新循环下一首就行了
            continue
        music_url = "https://isure.stream.qqmusic.qq.com/" + music_url_part  #歌曲完整下载地址
        music_url_content = requests.get(music_url,headers=headers).content  #将歌曲下载地址转化为二进制格式
        music_file_name = per_songname + "---" + singer + ".mp3"  #歌曲保存的名称  歌曲名+歌手名
        try:
            if not os.path.exists(file_address):
                os.mkdir(file_address)
            if not os.path.exists(file_address + music_file_name):
                print("正在下载     %s" % music_file_name)
                with open(file_address + music_file_name,"wb") as f:  #保存语句
                    f.write(music_url_content)
                print(music_file_name,"   下载成功")
            else:
                print(music_file_name,"   文件已存在")
        except:
            print("下载失败")

allmusic("汪苏泷")

5.结语

我也是新手,有很多问题,代码肯定也有很多不足之处,也请大神们多多指点,我们一起进步呀。

喜欢或者对你有所帮助的话,也请多多支持呀。

  • 13
    点赞
  • 63
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值