目标网站:酷狗TOP500_排行榜_乐库频道_酷狗网 (kugou.com)

具体页面如下:

1.网页结构分析

1)获取具体音乐的url

在目标页面中【右击】,点击页面检查,在菜单栏点击‘Network’,刷新页面,点击media(媒体),会出现相关数据流'mp3'文件,双击该文件,通过“headers”来看他的request url,在该网址中点击{}.mp3中的音乐文件署名,通过查找(ctrl+F)来查找相关文件信息,查找到‘php’文件,双击随后在‘Preview’下查看信息,会看见hash、album_id、play_url等信息,其中play_url就是我们要找的具体音乐地址,通过它可以去具体下载音乐文件。

首先,这是我们所获取的基础数据流信息。

其次,通过它去寻找需要的php文件

再次,我们会得到以下data信息,以两首歌对比为例,会看见其中hash和album_id不同,同时在里面也会看到play_url等。

最后,获取到url为play_url,通过它会得到音乐文件。

2、代码分步实现

第一、导入相关的库和设定相关的参数

import requests
import re
import time

其中我们获取到的url也设置出来

url = 'https://www.kugou.com/yy/rank/home/1-8888.html?from=rank'

第二、伪装设定并获取页面回应

def get_response(url):
    headers = {
        'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:43.0) Gecko/20100101 Firefox/43.0'
    }
    response = requests.get(url=url,headers=headers)
    return response

第三、函数封装

首先、我们通过对比两首歌曲的php文件的data信息可以看出它们的hash和album_id不同,同时它们也是我们获取play_url这个网站的网址所要的信息。下面的网址就是具体的网址,其中将hash和album_id虚处理。其中的其他信息dfid、mid、以及最后的数字串(时间戳)等都是无关信息。

https://wwwapi.kugou.com/yy/index.php?r=play/getdata&hash={Hash}&dfid=3ex60E2pQb582fRAwB1wYhA1&appid=1014&mid=94c23e6bf948c957d06d24c4dec18b1e&platid=4&album_id={music_id}&_=1695860308340

因此我们就需要来获取到我们所爬取榜单的歌曲的所有对应的hash和album_id

以hash查找为例在榜单页面的具体页面源代码中搜索hash信息,会查找到相关也数据。我们通过下面的封装函数来获取到所有的hash和id。

def get_music_id(url):
    response = get_response(url)
    Hash_list = re.findall('"Hash":"(.*?)"',response.text)
    album_id_list = re.findall('"album_id":(\d+)', response.text)
    music_id_list = zip(Hash_list,album_id_list)
    return music_id_list

 并存储为list

第三、获取网站信息及play_url

通过我们第二步得到的信息,带到网址里面,同时获取到我们的需要的基础信息(标题)等。

def get_music_info(Hash,music_id):
    """list"""
    link_list = f'https://wwwapi.kugou.com/yy/index.php?r=play/getdata&hash={Hash}&dfid=3ex60E2pQb582fRAwB1wYhA1&appid=1014&mid=94c23e6bf948c957d06d24c4dec18b1e&platid=4&album_id={music_id}&_=1695860308340'
    response = get_response(url=link_list)
    title = response.json()['data']['album_name']
    play_url = response.json()['data']['play_url']
    music_info = [title,play_url]
    # print(music_info)
    return music_info

第四、再获取到页面信息后,来进行音乐文件的保存。

def save(title,play_url):
    music_content = get_response(url=play_url).content
    with open("D:/musics/{}.mp3".format(title),mode='wb') as f:
        f.write(music_content)
        print(title,'保存成功')
        print('*'*20)

以保存到D盘中的music文件为例。以“歌曲的名称”为文件命名,保存为mp3文件。

第五、根据上述函数、步骤综合写出主函数。

def main(url):
    music_id_list = get_music_id(url=url)
    for Hash, music_id in music_id_list:
        try:
            music_info = get_music_info(Hash, music_id)
            save(music_info[0], music_info[1])
        except Exception as e:
            pass
        continue

我们在里面加入了try、catch来抛出异常(因为在榜单中有的音乐是),以防止我们对后续歌曲的爬取出现报错。

第六、运行函数

if __name__ == '__main__':
    url = 'https://www.kugou.com/yy/rank/home/1-8888.html?from=rank'
    main(url)
    time.sleep(2)

通过带入“url”信息,以TOP500榜单为例,我们进行试爬取。

会看到已经爬取下来,在这我们加上sleep来减缓速度。

同时再看存储文件,已经存储成功。

3、全部代码

import requests
import re
import time
def get_response(url):
    headers = {
        'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:43.0) Gecko/20100101 Firefox/43.0'
    }
    response = requests.get(url=url,headers=headers)
    return response

def get_list_url(url):
    response = get_response(url)
    list_url = re.findall('<a title="(.*?)".*?href="(.*?)"',response.text)
    return list_url

def get_music_id(url):
    response = get_response(url)
    Hash_list = re.findall('"Hash":"(.*?)"',response.text)
    album_id_list = re.findall('"album_id":(\d+)', response.text)
    music_id_list = zip(Hash_list,album_id_list)
    return music_id_list

def get_music_info(Hash,music_id):
    """list"""
    link_list = f'https://wwwapi.kugou.com/yy/index.php?r=play/getdata&hash={Hash}&dfid=3ex60E2pQb582fRAwB1wYhA1&appid=1014&mid=94c23e6bf948c957d06d24c4dec18b1e&platid=4&album_id={music_id}&_=1695860308340'
    response = get_response(url=link_list)
    title = response.json()['data']['album_name']
    play_url = response.json()['data']['play_url']
    music_info = [title,play_url]
    # print(music_info)
    return music_info

def save(title,play_url):
    music_content = get_response(url=play_url).content
    with open("D:/musics/{}.mp3".format(title),mode='wb') as f:
        f.write(music_content)
        print(title,'保存成功')
        print('*'*20)

def main(url):
    music_id_list = get_music_id(url=url)
    for Hash, music_id in music_id_list:
        try:
            music_info = get_music_info(Hash, music_id)
            save(music_info[0], music_info[1])
        except Exception as e:
            pass
        continue

if __name__ == '__main__':
    url = 'https://www.kugou.com/yy/rank/home/1-8888.html?from=rank'
    url1 = 'https://www.kugou.com/yy/html/rank.html'
    main(url)
    time.sleep(2)

4、改进

如果想将各个榜单的音乐都爬取下来,除了可以将url信息更换外(因为各个榜单的格式是大同小异的),还可以通过总的榜单网址,将各个榜单的url信息全部拿到。如下:

总榜为:https://www.kugou.com/yy/html/rank.html

根据页面源代码可以看到总榜下的每个页面榜单的“标题”和“网址链接href”都存在这个标签当中。

可以通过findall获取到所有的所要信息。

def get_list_url(html_url):
    response = get_response(html_url)
    list_url = re.findall('<a title="(.*?)".*?href="(.*?)"',response.text)
    # print(list_url)
    return list_url

 将获取到的信息存储到list_url。

接下来只需将main()函数更改以下信息即可。

def main(html_url):
    list_url = get_list_url(html_url=html_url)
    for list_name,link in list_url:
        print(f'-----------正在爬取{list_name}-------------------')
        music_id_list = get_music_id(html_url=link)
        for Hash,music_id in music_id_list:
            try:
                music_info = get_music_info(Hash, music_id)
                save(music_info[0],music_info[1])
            except Exception as e:
                pass
            continue

可以将每个榜单的歌曲都爬取下来。

全部代码(改进后):

 

import pprint
import re
import parsel
import requests
def get_response(html_url):
    headers = {
        'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:43.0) Gecko/20100101 Firefox/43.0'
    }
    response = requests.get(url=html_url,headers=headers)
    return response
def get_list_url(html_url):
    response = get_response(html_url)
    list_url = re.findall('<a title="(.*?)".*?href="(.*?)"',response.text)
    # print(list_url)
    return list_url
def get_music_id(html_url):
    """hash id"""
    response = get_response(html_url)
    Hash_list = re.findall('"Hash":"(.*?)"',response.text)
    album_id_list = re.findall('"album_id":(\d+)', response.text)
    music_id_list = zip(Hash_list,album_id_list)
    return music_id_list
def get_music_info(Hash,music_id):
    """list"""
    link_list = f'https://wwwapi.kugou.com//yy/index.php?r=play/getdata&hash={Hash}&dfid=08yHnP1NIxEY4F1hXr3wleBb&appid=1014&mid=bebaa22b82662cab54aa33fb486be51d&platid=4&album_id={music_id}&_=1639379015124'
    response = get_response(html_url=link_list)
    title = response.json()['data']['album_name']
    play_url = response.json()['data']['play_url']
    music_info = [title,play_url]
    # print(music_info)
    return music_info
def save(title,play_url):
    music_content = get_response(html_url=play_url).content
    with open("D:/musics/{}.mp3".format(title),mode='wb') as f:
        f.write(music_content)
        print(title,'保存成功')
def main(html_url):
    list_url = get_list_url(html_url=html_url)
    for list_name,link in list_url:
        print(f'-----------正在爬取{list_name}-------------------')
        music_id_list = get_music_id(html_url=link)
        for Hash,music_id in music_id_list:
            try:
                music_info = get_music_info(Hash, music_id)
                save(music_info[0],music_info[1])
            except Exception as e:
                pass
            continue


if __name__ == '__main__':
    url = 'https://www.kugou.com/yy/html/rank.html'
    main(url)
   

仅供参考,支持原版歌曲。

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐