最近,在优化一个自己写的音乐播放器。主要目的是回顾、归纳,并希望能够写出一个属于自己的common lib。今天,主要是关于在线音乐API的一些分析结果。此次,主要分析的是歌词、专辑部分。在线搜索音乐、热门音乐及mp3的下载等,会在PART 2.2进行补充。

原始API来源于网络资料,部分是后面使用个人补充的。主要包括百度API、腾讯API及歌词迷API,其中只有歌词迷的API是官方正式发布的。三个API都有着各自的优点、缺点,如下:

(1) 百度API,请求方式稳定,速度快,资源最多,获取歌词比较准确;但是数据结构相对繁杂些,每行的歌词长度差异比较大。

(2) 腾讯API,请求方式相对稳定,速度快,资源较多,准确度高,每行的歌词长度相当;但JSON(Xml相对正常)数据结构并不完全标准,解析麻烦, 专辑图片封面(约50KB|500 x 500 像素)较大。

(3)歌词迷API,有官方正式API,使用简单,专辑封面相对小些(约10KB|185 x 160 像素);遗憾的是资源相对少,尤其在最新的资源方面,有点慢。

提醒:以上全是个人开发的总结,并没有完整体系性的验证。

如专辑封面大小问题,视乎个人开发需要而定,如果需要大图片,腾讯的保真度高,如果需要小图片,无疑歌词迷更好些。

本人在歌词方面使用的腾讯API,专辑封面使用的是歌词迷API。


整个实现思路比较明确,大体上的类图设计如下:

直接使用LyricLoader的loadLyric()方法进行歌词下载,loadLyric()方法封装了具体的处理逻辑,具体实现下载,由子类实现IDownload<Lyric>接口。摘取部分代码:

/**
* 歌词助手
*
* @author Osmondy
*
*/
public abstract class LyricLoader implements IDownload<Lyric>
{ public LyricLoader(String name)
{ } /**
* 获取网络请求歌词地址
*
* @param music
* @return
*/
public abstract String getServerLyricUrl(Music music); /**
* 返回本地存储歌词的路径
*
* @param music
* @return
*/
protected String getLocalLyricPath(String songname, String singername)
{ } /**
* 返回歌词, Step1: 本地歌词目录加载; Step2: 网络下载.
*
* @param music
* @return
*/
public Lyric loadLyric(Music music)
{
if (TextUtils.isEmpty(music.getArtist()) || TextUtils.isEmpty(music.getTitle()))
{
Log.W(TAG, "Empty aritst or title, can't find lyric."); return null;
} Lyric lyric = null;
String localPath = getLocalLyricPath(music.getTitle(), music.getArtist()); File file = new File(localPath);
if (file.exists())
{
// 本地存在歌词文件, 直接加载此歌词.
Log.D(TAG, "Loading lyric from local path.");
try
{
lyric = loadLocalLyric(localPath);
if (lyric != null)
{
lyric.setSongname(music.getTitle());
lyric.setSingername(music.getArtist()); Log.I(TAG, "Load local lyric finished. Lyric: " + lyric);
}
}
catch (IOException e)
{
if (e instanceof FileNotFoundException)
{
Log.W(TAG, "Lyric not found.");
}
else
{
e.printStackTrace();
}
} return lyric;
} String requestUrl = getServerLyricUrl(music); if (!TextUtils.isEmpty(requestUrl))
{
Log.D(TAG, "---------- Download lyric start ----------");
try
{
lyric = download(requestUrl, localPath);
}
catch (HttpRequestException e)
{
e.printStackTrace();
} Log.D(TAG, "---------- Download lyric end ----------"); return lyric;
} Log.W(TAG, "Not found a correct server lyric path."); return null; } /**
* 保存歌曲文件, 默认保存至{@link AppConfig#DIRECTORY_LYRIC}, 子类可自行重写保存至其它路径. 保存时,
* 先保存成*.lrc.tmp, 下载及保存成功后, 再重命名为*.lrc. 防止异常或停止下载歌词, 下次无法再次下载.
*
* @param is
* @param music
* @return
*/
protected boolean saveLyric(InputStream is, String savePath)
{ } /**
* 返回指定地址的歌词文件
*
* @param path
* @return
* @throws IOException
*/
public Lyric loadLocalLyric(String path) throws IOException
{ } }

抽象类LyricLoader提供了对歌词保存、加载的默认处理方式,子类可以自行重写saveLyric()、loadLocalLyric()定义自己的处理方式。子类的实现以百度API为例,它使用的是父类LyricLoader提供的默认实现。

/**
* 歌词来源于Baidu
*
* @author Osmondy
*
*/
public class BaiduLyricHelper extends LyricLoader
{ private static final String TAG = "BaiduLyricHelper"; /**
* 歌曲信息请求地址
*/
protected static final String SONGINFO_BASE_URL = "http://box.zhangmen.baidu.com/x"; /**
* 歌词文件请求地址
*/
protected static final String LYRIC_BASE_URL = "http://box.zhangmen.baidu.com/bdlrc"; public BaiduLyricHelper()
{
super("BaiDu");
} @Override
public Lyric download(String requestUrl, String savePath) throws HttpRequestException
{ } @Override
public String getServerLyricUrl(Music music)
{ } }

比较完整的代码已经上传至github:https://github.com/osmondy/LyricApi


原始API如下:

(1) 百度API

歌曲信息请求地址:http://box.zhangmen.baidu.com/x?op=12&count=1&title=歌词名称$$歌手名称$$$$

歌词信息请求地址:http://box.zhangmen.baidu.com/bdlrc/歌词ID除以100/歌词ID.lrc

/**
* 返回请求歌词的地址, 通过 SongInfo生成最终可请求到歌词文件的地址. </br>
*
* @param songInfo
* @return
*/
protected String getServerLyricUrlBySongInfo(SongInfo songInfo)
{
int lrcid = songInfo.getLrcid();
int postfix = lrcid / ; StringBuffer sb = new StringBuffer();
sb.append(LYRIC_BASE_URL);
sb.append("/");
sb.append(postfix);
sb.append("/");
sb.append(lrcid);
sb.append(".lrc"); return sb.toString();
} @Override
public String getServerLyricUrl(Music music)
{
if (TextUtils.isEmpty(music.getTitle()) || TextUtils.isEmpty(music.getArtist()))
{
return null;
}
//protected static final String SONGINFO_BASE_URL = "http://box.zhangmen.baidu.com/x?op=12&count=1&title=";
Log.D(TAG, "Songname: " + music.getTitle() + ", Singername: " + music.getArtist());
StringBuffer sb = new StringBuffer();
try
{
sb.append(SONGINFO_BASE_URL);
sb.append("?");
sb.append("op=12");
sb.append("&");
sb.append("count=1");
sb.append("&");
sb.append("title=");
sb.append(URLEncoder.encode(music.getTitle(), "utf-8"));
sb.append("$$");
sb.append(URLEncoder.encode(music.getArtist(), "utf-8"));
sb.append("$$$$");
}
catch (UnsupportedEncodingException e)
{
e.printStackTrace();
} return sb.toString();
}

构建请求的URL

(2) 腾讯API

编码并非是UTF-8,而是GBK(gb2312)。

歌曲信息请求地址:http://qqmusic.qq.com/fcgi-bin/qm_getLyricId.fcg?name=连哭都是我的错&singer=东来东往&from=qqplayer

歌词请求地址:http://music.qq.com/miniportal/static/lyric/歌曲ID求余100/歌曲ID.xml

专辑封面请求地址:http://imgcache.qq.com/music/photo/album/专辑ID求余100/albumpic_专辑ID_0.jpg

/**
* 返回请求歌词的地址, 通过 SongInfo生成最终可请求到歌词文件的地址. </br>
* 请求地址格式: http://music.qq.com/miniportal/static/lyric/67/183767.xml
*
* @param songInfo
* @return
*/
protected String getServerLyricUrlBySongInfo(SongInfo songInfo)
{
String id = songInfo.getId(); if (!StringUtils.isNumeric(id))
{
return null;
}
int postfix = Integer.parseInt(id) % ; StringBuffer sb = new StringBuffer();
sb.append(LYRIC_BASE_URL);
sb.append("/");
sb.append(postfix);
sb.append("/");
sb.append(id);
sb.append(".xml"); return sb.toString();
} /**
* 返回请求歌曲信息的地址.
* 请求地址格式: http://qqmusic.qq.com/fcgi-bin/qm_getLyricId.fcg?name=连哭都是我的错&singer=东来东往&from=qqplayer
*/
@Override
public String getServerLyricUrl(Music music)
{
if (TextUtils.isEmpty(music.getTitle()) || TextUtils.isEmpty(music.getArtist()))
{
return null;
} Log.D(TAG, "Songname: " + music.getTitle() + ", Singername: " + music.getArtist());
StringBuffer sb = new StringBuffer();
try
{
sb.append(SONGINFO_BASE_URL);
sb.append("?");
sb.append("name=" + URLEncoder.encode(music.getTitle(), "gbk"));
sb.append("&");
sb.append("singer=" + URLEncoder.encode(music.getArtist(), "gbk"));
sb.append("&");
sb.append("from=qqplayer");
}
catch (UnsupportedEncodingException e)
{
e.printStackTrace();
} return sb.toString();
}

构建请求的URL

(3)歌词迷API

直接提供官方地址:http://api.geci.me/en/latest/

歌词请求地址:http://geci.me/api/lyric/:歌曲名/:歌手名

专辑封面请求地址:http://geci.me/api/cover/:专辑ID

@Override
public String getServerLyricUrl(Music music)
{
if (TextUtils.isEmpty(music.getTitle()) || TextUtils.isEmpty(music.getArtist()))
{
return null;
} Log.D(TAG, "Songname: " + music.getTitle() + ", Singername: " + music.getArtist());
StringBuffer sb = new StringBuffer();
try
{
sb.append(LYRIC_BASE_PATH);
sb.append("/");
sb.append(URLEncoder.encode(music.getTitle(), "utf-8"));
sb.append("/");
sb.append(URLEncoder.encode(music.getArtist(), "utf-8"));
}
catch (UnsupportedEncodingException e)
{
e.printStackTrace();
} return sb.toString();
} /**
* 返回歌曲专辑信息请求地址
*
* @param albumId
* @return
*/
public String getServerAlbumUrl(String albumId)
{
return ALBUM_BASE_PATH + "/" + albumId;
}

构建请求的URL

最后,附上腾讯如何获取新歌榜及总榜的请求。

新歌榜:http://music.qq.com/musicbox/shop/v3/data/hit/hit_newsong.js
      总榜:http://music.qq.com/musicbox/shop/v3/data/hit/hit_all.js

[原创] 在线音乐API的研究 (Part 2.1)的更多相关文章

  1. 在线音乐API的研究 (Part 2.1)

    本文转载于:http://www.cnblogs.com/osmondy/p/LyricApi.html 最近,在优化一个自己写的音乐播放器.主要目的是回顾.归纳,并希望能够写出一个属于自己的comm ...

  2. 在线音乐播放器-----酷狗音乐api接口抓取

    首先身为一个在线音乐播放器,需要前端和数据库的搭配使用. 在数据库方面,我们没有办法制作,首先是版权问题,再加上数据量.所以我们需要借用其他网络播放器的数据库. 但是这些在线播放器,如百度,酷狗,酷我 ...

  3. 百度音乐API抓取

    百度音乐API抓取 前段时间做了一个本地音乐的播放器 github地址,想实现在线播放的功能,于是到处寻找API,很遗憾,不是歌曲不全就是质量不高.在网上发现这么一个APIMRASONG博客,有“获取 ...

  4. 【QQ音乐Api】移花接木 打造自己的音乐电台

    最近突发奇想想做个在线音乐小网页.需求很简单,如下 搜索歌曲 或 歌手 在线播放音乐 借用qq 或者 百度的 音乐接口 需求明确那就直接动手了 我首先尝试的百度音乐,但是不能在线播放(提示forbid ...

  5. iOS在线音乐播放SZKAVPlayer(基于AVPlayer的封装)

    由于最近闲着没事,想找有关在线音乐播放的demo学习一下,在gitHub跟code4APP上面查找了很多帖子,结果很多在线音乐都是基于AudioStream实现的,我感觉用起来不太方便.后来突然发现, ...

  6. 如何使用 python 爬取酷我在线音乐

    前言 写这篇博客的初衷是加深自己对网络请求发送和响应的理解,仅供学习使用,请勿用于非法用途!文明爬虫,从我做起.下面进入正题. 获取歌曲信息列表 在酷我的搜索框中输入关键词 aiko,回车之后可以看到 ...

  7. Andriod小项目——在线音乐播放器

    转载自: http://blog.csdn.net/sunkes/article/details/51189189 Andriod小项目——在线音乐播放器 Android在线音乐播放器 从大一开始就已 ...

  8. 在线音乐网站【04】Part two 功能实现

       上一篇博客里面已近总结了三个功能的具体实现,今天把剩余功能的具体实现补充总结,如果你想对整个小项目有清楚的了解,建议去看下前几篇博客. 1.在线音乐网站(1)需求和功能结构 2.在线音乐网站(2 ...

  9. 在线音乐网站【03】Part one 功能实现

    今天打算把网站功能的具体实现给总结一下,如果你想了解整个小项目,建议你先看看前面2篇博客. 1.在线音乐网站(1)需求和功能结构 2.在线音乐网站(2)数据库和开发环境 7.网站主要模块实现 a.在线 ...

随机推荐

  1. HDU 2829 斜率优化DP Lawrence

    题意:n个数之间放m个障碍,分隔成m+1段.对于每段两两数相乘再求和,然后把这m+1个值加起来,让这个值最小. 设: d(i, j)表示前i个数之间放j个炸弹能得到的最小值 sum(i)为前缀和,co ...

  2. Cocoa-Cocoa框架

    1.Cocoa是什么? Cocoa是OS X和 iOS操作系统的程序的运行环境. 是什么因素使一个程序成为Cocoa程序呢?不是编程语言,因为在Cocoa开发中你可以使用各种语言:也不是开发工具,你可 ...

  3. windows phone 网络开发三部曲(一)各种包的各种抓法

    首先感谢大家对我上一篇博客的支持,让我也体验了一把上榜的感觉. 这无疑是对我这个刚刚打算,认真写写博客的人的莫大的鼓励,再次感谢(鞠躬)!! 接下来想和大家分享一些关于windows phone网络开 ...

  4. js中取绝对值的2种方法!

    1.abs() var aaa=-20; var bbb=Math.abs(aaa); 2.加减法 var aaa=-20; var bbb=-aaa

  5. Lenovo笔记本电脑进入BIOS的方法

    使用NOVO键开机进入BIOS的操作方法 适用范围:2012年后发布的部分笔记本产品,含:IdeaPad全系列.Lenovo G系列部分IdeaPad U或S系列,YOGA/FLEX全系列产品Leno ...

  6. POJ-2187 Beauty Contest,旋转卡壳求解平面最远点对!

     凸包(旋转卡壳) 大概理解了凸包A了两道模板题之后在去吃饭的路上想了想什么叫旋转卡壳呢?回来无聊就搜了一下,结果发现其范围真广. 凸包: 凸包就是给定平面图上的一些点集(二维图包),然后求点集组成的 ...

  7. BZOJ 3926 [Zjoi2015]诸神眷顾的幻想乡 ——广义后缀自动机

    神奇的性质,叶子节点不超过20个. 然后把这些节点提出来构成一颗新树,那么这些树恰好包含了所有的情况. 所以直接广义后缀自动机. 然后统计本质不同的字符串就很简单显然了. #include <c ...

  8. 刷题总结——art2(ssoj)

    题目: 题解: o(n)复杂度扫一遍再用一个stack维护就可以了·····mdzz这道题都不会做·· 代码: #include<iostream> #include<cstdio& ...

  9. Contest Hunter #46 T1 磁力块 [分块]

    描述 在一片广袤无垠的原野上,散落着N块磁石.每个磁石的性质可以用一个五元组(x,y,m,p,r)描述,其中x,y表示其坐标,m是磁石的质量,p是磁力,r是吸引半径.若磁石A与磁石B的距离不大于磁石A ...

  10. 【POJ3415】Common Substrings(后缀数组,单调栈)

    题意: n<=1e5 思路: 我的做法和题解有些不同 题解是维护A的单调栈算B的贡献,反过来再做一次 我是去掉起始位置不同这个限制条件先算总方案数,再把两个串内部不合法的方案数减去 式子展开之后 ...