前段时间写了一篇c#解析lrc歌词文件,对lrc文件进行解析,支持多个时间段合并。本文借下载歌词文件来探讨一下同步和异步方法。
lrc文件在网络上随处可见,我们可以通过一些方法获取,最简单的就是别人的接口,如: http://geci.me/api/lyric/不得不爱 返回下面的json,这样我们就很容易得到歌词文件了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
{ "count" : 2, "code" : 0, "result" : [ { "aid" : 2727794, "lrc" : "http://s.geci.me/lrc/327/32793/3279317.lrc" , "song" : "不得不爱" , "artist_id" : 2, "sid" : 3279317 }, { "aid" : 3048347, "lrc" : "http://s.geci.me/lrc/371/37129/3712941.lrc" , "song" : "不得不爱" , "artist_id" : 2, "sid" : 3712941 } ] } |
在c#解析lrc歌词文件中我们创建了lrc类,我们继续在该类中添加方法。
同步下载实现
创建searchlrc静态方法,该方法实现对歌词的搜索:首先查看本地文件夹(我的文件夹是d:\lrc\)是否存在lrc文件,如果不存在就下载lrc文件,返回lrc对象。
1
2
3
4
5
6
7
8
9
10
11
12
|
public static lrc searchlrc( string musicname) { string path = @"d:\lrc\" + musicname + ".lrc"; if (system.io.file.exists(path)) { return initlrc(path); } else { return downloadlrc(musicname, path); } } |
下载歌词利用webclient,首先用downloadstring方法将获取json,再利用javascriptserializer反序列化为自定义对象,这样就得到了lrc文件的url,最后通过url将lrc文件下载到本地,再调用initlrc方法返回lrc对象。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
public class tempjosnmain { public int count { get ; set ; } public int code { get ; set ; } public list<tempjsonchild> result { get ; set ; } } public class tempjsonchild { public int aid { get ; set ; } public string lrc { get ; set ; } public string song { get ; set ; } public int artist_id { get ; set ; } public int sid { get ; set ; } } static lrc downloadlrc( string musicname, string path) { if (musicname.contains( "-" )) musicname = musicname.split( '-' )[1].trim(); string url = "http://geci.me/api/lyric/" + musicname; webclient wc = new webclient(); string json = wc.downloadstring(url); javascriptserializer js = new javascriptserializer(); tempjosnmain res = js.deserialize<tempjosnmain>(json); if (res.count > 0) { wc.downloadfile( new uri(res.result[0].lrc), path); wc.dispose(); return initlrc(path); } return new lrc(); } |
异步下载实现
创建searchlrcasyc静态方法,该方法没有返回值,所以我们用回调方法作为参数(该回调方法用lrc作为参数并且没有返回值),异步下载主要体现在json数据和文件的下载
1
2
3
4
5
6
7
8
9
10
11
12
|
public static void searchlrcasyc( string musicname, action<lrc> action) { string path = @"d:\lrc\" + musicname + ".lrc"; if (system.io.file.exists(path)) { action(initlrc(path)); } else { downloadlrcasyc(musicname, path, action); } } |
webclient的downloadstringasync实现异步下载字符串,不会阻止调用线程。
downloadstringcompleted事件在下载字符串完成后触发。我们可以使用
downloadstringasync方法的构造来传递参数,从而达到在downloadstringcompleted内部调用我们的action<lrc>函数。而我们的参数有两个,所以需要封装成一个对象。
1
2
3
4
|
public void downloadstringasync( uri address, object usertoken ) |
address
包含要下载的 uri 的 uri。
usertoken
一个用户定义对象,此对象将被传递给完成异步操作时所调用的方法。在downloadstringcompleted方法中通过e.userstate来获取
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
public class callbackobject { public string path { get ; set ; } public action<lrc> action { get ; set ; } } static void downloadlrcasyc( string musicname, string path, action<lrc> action) { if (musicname.contains( "-" )) musicname = musicname.split( '-' )[1].trim(); string url = "http://geci.me/api/lyric/" + musicname; webclient wc = new webclient(); callbackobject co = new callbackobject() { action = action, path = path }; wc.downloadstringcompleted += new downloadstringcompletedeventhandler(wc_downloadstringcompleted); wc.downloadstringasync( new uri(url), co); } static void wc_downloadstringcompleted( object sender, downloadstringcompletedeventargs e) { javascriptserializer js = new javascriptserializer(); tempjosnmain res = js.deserialize<tempjosnmain>(e.result); if (res.count > 0) { webclient wc = sender as webclient; if (wc == null ) wc = new webclient(); callbackobject co = e.userstate as callbackobject; wc.downloadfilecompleted += new system.componentmodel.asynccompletedeventhandler(wc_downloadfilecompleted); wc.downloadfileasync( new uri(res.result[0].lrc), co.path, co); } } static void wc_downloadfilecompleted( object sender, system.componentmodel.asynccompletedeventargs e) { callbackobject co = e.userstate as callbackobject; co.action(initlrc(co.path)); } |
最后演示:
点击下载时会有线程等待感觉像程序”卡死”,而异步下载则非常流畅。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。