C#下載歌詞文件的同步和異步方法
前段時間寫了一篇C#解析Lrc歌詞文件,對lrc文件進行解析,支持多個時間段合并。本文借下載歌詞文件來探討一下同步和異步方法。
Lrc文件在網(wǎng)絡上隨處可見,我們可以通過一些方法獲取,最簡單的就是別人的接口,如: http://geci.me/api/lyric/不得不愛 返回下面的json,這樣我們就很容易得到歌詞文件了。
{ "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歌詞文件中我們創(chuàng)建了Lrc類,我們繼續(xù)在該類中添加方法。
同步下載實現(xiàn)
創(chuàng)建SearchLrc靜態(tài)方法,該方法實現(xiàn)對歌詞的搜索:首先查看本地文件夾(我的文件夾是D:\lrc\)是否存在lrc文件,如果不存在就下載lrc文件,返回Lrc對象。
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文件下載到本地,再調(diào)用InitLrc方法返回Lrc對象。
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(); }
異步下載實現(xiàn)
創(chuàng)建SearchLrcAsyc靜態(tài)方法,該方法沒有返回值,所以我們用回調(diào)方法作為參數(shù)(該回調(diào)方法用Lrc作為參數(shù)并且沒有返回值),異步下載主要體現(xiàn)在json數(shù)據(jù)和文件的下載
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實現(xiàn)異步下載字符串,不會阻止調(diào)用線程。
DownloadStringCompleted事件在下載字符串完成后觸發(fā)。我們可以使用
DownloadStringAsync方法的構造來傳遞參數(shù),從而達到在DownloadStringCompleted內(nèi)部調(diào)用我們的Action<Lrc>函數(shù)。而我們的參數(shù)有兩個,所以需要封裝成一個對象。
public void DownloadStringAsync( Uri address, object userToken )
address
包含要下載的 URI 的 Uri。
userToken
一個用戶定義對象,此對象將被傳遞給完成異步操作時所調(diào)用的方法。在DownloadStringCompleted方法中通過e.UserState來獲取
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)); }
最后演示:
點擊下載時會有線程等待感覺像程序”卡死”,而異步下載則非常流暢。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
C#實現(xiàn)DataTable轉換成IList的方法
這篇文章主要介紹了C#實現(xiàn)DataTable轉換成IList的方法,涉及C#針對DataTable操作的相關技巧,需要的朋友可以參考下2016-03-03