C#下载歌词文件的同步和异步方法

所属分类: 软件编程 / C#教程 阅读数: 93
收藏 0 赞 0 分享

前段时间写了一篇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));
  }

最后演示:

点击下载时会有线程等待感觉像程序”卡死”,而异步下载则非常流畅。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

c#开发word批量转pdf源码分享

已经安装有Office环境,借助一些简单的代码即可实现批量Word转PDF,看下面的实例源码吧
收藏 0 赞 0 分享

c# xml API操作的小例子

这篇文章主要介绍了c# xml API操作的小例子,有需要的朋友可以参考一下
收藏 0 赞 0 分享

c#唯一值渲染实例代码

这篇文章主要介绍了c#唯一值渲染实例代码,有需要的朋友可以参考一下
收藏 0 赞 0 分享

淘宝IP地址库采集器c#代码

这篇文章主要介绍了淘宝IP地址库采集器c#代码,有需要的朋友可以参考一下
收藏 0 赞 0 分享

C#在后台运行操作(BackgroundWorker用法)示例分享

BackgroundWorker类允许在单独的专用线程上运行操作。如果需要能进行响应的用户界面,而且面临与这类操作相关的长时间延迟,则可以使用BackgroundWorker类方便地解决问题,下面看示例
收藏 0 赞 0 分享

c#文本加密程序代码示例

这是一个加密软件,但只限于文本加密,加了窗口控件的滑动效果,详细看下面的代码
收藏 0 赞 0 分享

c#生成站点地图(SiteMapPath)文件示例程序

这篇文章主要介绍了c#生成站点地图(SiteMapPath)文件的示例,大家参考使用
收藏 0 赞 0 分享

C# 键盘Enter键取代Tab键实现代码

这篇文章主要介绍了C# 键盘Enter键取代Tab键实现代码,有需要的朋友可以参考一下
收藏 0 赞 0 分享

C# WinForm导出Excel方法介绍

在.NET应用中,导出Excel是很常见的需求,导出Excel报表大致有以下三种方式:Office PIA,文件流和NPOI开源库,本文只介绍前两种方式
收藏 0 赞 0 分享

C#串口通信程序实例详解

在.NET平台下创建C#串口通信程序,.NET 2.0提供了串口通信的功能,其命名空间是System.IO.Ports,创建C#串口通信程序的具体实现是如何的呢?让我们开始吧
收藏 0 赞 0 分享
查看更多