前段时间写了一篇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));
}
|
最后演示:
点击下载时会有线程等待感觉像程序”卡死”,而异步下载则非常流畅。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。