前段時間寫了一篇C#解析Lrc歌詞文件,對lrc文件進行解析,支持多個時間段合并。本文借下載歌詞文件來探討一下同步和異步方法。
Lrc文件在網絡上隨處可見,我們可以通過一些方法獲取,最簡單的就是別人的接口,如: 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歌詞文件中我們創建了Lrc類,我們繼續在該類中添加方法。
同步下載實現
創建SearchLrc靜態方法,該方法實現對歌詞的搜索:首先查看本地文件夾(我的文件夾是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文件下載到本地,再調用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(); }
異步下載實現
創建SearchLrcAsyc靜態方法,該方法沒有返回值,所以我們用回調方法作為參數(該回調方法用Lrc作為參數并且沒有返回值),異步下載主要體現在json數據和文件的下載
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>函數。而我們的參數有兩個,所以需要封裝成一個對象。
public void DownloadStringAsync( Uri address, object userToken)
address
包含要下載的 URI 的 Uri。
userToken
一個用戶定義對象,此對象將被傳遞給完成異步操作時所調用的方法。在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)); }
最后演示:
點擊下載時會有線程等待感覺像程序”卡死”,而異步下載則非常流暢。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。
新聞熱點
疑難解答