使用python下载歌词并嵌入歌曲文件中的实现代码

使用python扫描本地音乐并下载歌词
这次这个真的是干货哦,昨晚弄了半晚上,,,,从8点吃完饭就开始写,一直到了快12点才弄好,,,新手,伤不起呀。。。。
先简单的说下吧,百度提供了一个音乐搜索的api,你想百度请求类似于

http://box.zhangmen.baidu.com/x?op=12&count=1&title=最佳损友$$陈奕迅$$$$

的地址,百度会给你返回一段xml,如下所示

this xml file does not appear to have any style information associated with it. the document tree is shown below.

1

8
2829
1

8
2829
1

f98b6772aa97966550ec80617879becee0233bf4

mp3
3778335
128

简单的说明下,由于我们要做的只是获取到歌曲的lrc歌词地址,所以有用的只有2829这个标签。
而encode和decode里面的拼接起来就是mp3的下载地址,如本例的

http://zhangmenshiting.baidu.com/data2/music/12762845/ymrqamdua21fn6nndk6ap5wxcjlrmg1xljhobwibmgpjk5ztmwizcwrjz5lqbgyelgkwlztubgljz5lka2uanwsxy1qin5t1ywbmzw5ocglhawdnbgtqbze$12762845.mp3?xcode=e6b69cf593ea22ac9d2b9314e565fc0caf85125f065ce3e0&mid=0.31929107437537

就是下载地址,不过音质太差,有时间在研究下这个。
继续说歌词,注意lrcid标签里面的2829
http://box.zhangmen.baidu.com/bdlrc/ 这个是百度lrc歌词存放地址,
然后本例的歌词地址是http://box.zhangmen.baidu.com/bdlrc/28/2829.lrc
看到了吧,歌词地址后面的两个数字的计算方法是在lrcid除以100所获得的整数,就是第一个数字,然后第二个数字就是lrcid,然后后面加上后缀.lrc就搞定了
获得lrc地址之后就简单了,只要请求该地址,然后将获取到的内容写入文件就ok了。
好了,大概就是这样,下面是代码

import os
import os.path
import re
import eyed3
import urllib2
import urllib
from urllib import urlencode
import sys
import os
reload(sys)
sys.setdefaultencoding(‘utf8’)
music_path = r”e:\music”
lrc_path = r”e:\lrc”
os.remove(‘nolrc.txt’)
os.remove(‘lrcxml.txt’)
the_file = open(‘lrcxml.txt’,’a’)
nolrc_file = open(‘nolrc.txt’,’a’)
for root,dirs,files in os.walk(music_path):
for filepath in files:
the_path = os.path.join(root,filepath)
if (the_path.find(“mp3”) != -1):
print the_path
the_music = eyed3.load(the_path)
the_teg = the_music.tag._getalbum()
the_artist = the_music.tag._getartist()
the_title = the_music.tag._gettitle()
# print the_teg
# print the_title
# print the_artist
b = the_title.replace(‘ ‘,’+’)
# print b
a = the_artist.replace(‘ ‘,’+’)
#print urlencode(str(b))
if isinstance(a,unicode):
a = a.encode(‘utf8′)
song_url = “http://box.zhangmen.baidu.com/x?op=12&count=1&title=”+b+”$$”+a+”$$$$ ”
the_file.write(song_url+’\n’)
page = urllib2.urlopen(song_url).read()
print page
theid = 0
lrcid = re.compile(‘(.*?)’,re.s).findall(page)
have_lrc = true
if lrcid != []:
theid = lrcid[0]
else:
nolrc_file.write(the_title+’\n’)
have_lrc = false
print theid
if have_lrc:
firstid = int(theid)/100
lrcurl = “http://box.zhangmen.baidu.com/bdlrc/”+str(firstid)+”/”+theid+”.lrc”
print lrcurl
lrc = urllib2.urlopen(lrcurl).read()
if(lrc.find(‘html’)== -1):
lrcfile = open(lrc_path+”\\”+the_title+”.lrc”,’w’)
lrcfile.writelines(lrc)
lrcfile.close()
else:
nolrc_file.write(the_title+’\n’)
the_file.close()
nolrc_file.close()
print “end!”

有用第一步请求所获取到底是xml格式的,所以本来想着解析xml来获取lrcid,但是在实现过程中遇到了各种问题,别的还容易,就在这一块儿浪费的时间最长,纠结未果之后,只能改用正则表达式来获取了。。。

使用python将歌词嵌入歌曲中
以前一直用的是google play music来作为手机的音乐播放器,可是现在谷歌被墙的这么厉害的,从pc上传到google play的音乐在手机上面同步下来的话特麻烦,索性放弃之买了大名鼎鼎的poweramp播放器,开始使用之后瞬间就被poweramp强大的功能所吸引住了,不愧是安卓端的音乐播放器的王者!唯美的锁屏界面,强大的均衡器功能等等。唯一美中不足的就是歌词.如果要显示歌词的话必须安装第三方软件,或者是把歌词嵌入到音乐中。所以昨天下班之后就开始研究,所幸最后终于搞定了,先上下效果图

20151113154400981.png (175×300)

可以看到,效果还是很不错的呢。
好了,废话不多说,下面上程序
首先,必须安装eyed3模块,还有,我所有的歌词都在e:\lrc这个路径中的

import threading
import time
import datetime
import re
import os
import eyed3
import sys
reload(sys)
sys.setdefaultencoding(‘utf8’)
def getstr(i):
if i 0:
return deallrc(line)
else:
continue
return ”
def getlrcstr(lrc):
mylrcstr=”
#print lrc
for i in range(00,05):
for j in range(00,59):
for k in range(00,99):
timespan=getstr(i)+”:”+getstr(j)+”.”+getstr(k)
mylrcstr+=checklrcfile(lrc, timespan)
#print timespan
return mylrcstr
def getlrc(musicname):
musicname=u”.join(musicname)
musicname=musicname.encode(‘gb2312′)
for root,dirs,files in os.walk(lrcpath):
for filepath in files:
the_path = os.path.join(root,filepath)
if (the_path.find(musicname) != -1):
print the_path
return the_path
def errorlog(path):
file=open(r’e:\nolrc.txt’,’a’)
if path is none:
path=”
path=path+’\n’
file.write(path)
file.close()
def writetag(themusic,lrcstr):
music=eyed3.load(themusic)
lrcstr=lrcstr.decode(‘utf8′)
lrcstr=u”.join(lrcstr)
#lrcstr=unicode(lrcstr)
music.tag.lyrics.set(lrcstr)
music.tag.save()
def dealmusic(path):
print path
the_music = eyed3.load(path)
the_teg = the_music.tag._getalbum()
the_artist = the_music.tag._getartist()
the_title = the_music.tag._gettitle()
#print the_title
try:
lrc=getlrc(the_title)
lrcstr=getlrcstr(lrc)
writetag(path, lrcstr)
except:
errorlog(path)
class writelrc(threading.thread):
def __init__(self,the_path):
threading.thread.__init__(self)
self.thepath=the_path
def run(self):
dealmusic(self.thepath)
if __name__==’__main__’:
count=0
threads=[]
for root,dirs,files in os.walk(musicpath):
for filepath in files:
the_path = os.path.join(root,filepath)
if (the_path.find(“mp3”) != -1):
count+=1
threads.append(writelrc(the_path))
if count%10==0:
for t in threads:
t.start()
for t in threads:
t.join()
threads=[]

好了,大概就是这样,大家有什么问题可以直接提出来,我会尽快回复的。

Posted in 未分类