• C#下载歌词文件


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

    异步下载实现

    创建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));
            }
    View Code

    最后演示

    点击下载时会有线程等待感觉像程序”卡死”,而异步下载则非常流畅。后面将继续介绍使用async和await来实现异步下载及播放器桌面歌词

    附件下载:http://download.csdn.net/detail/oyipiantian/9531861

  • 相关阅读:
    MFC菜单、工具栏和状态栏
    MFC控件
    MFC对话框
    MFC消息映射机制
    性能优化可使用的工具
    常见的内存问题分析思路
    k8s之Probe
    docker swarm
    keepalived配置虚拟ip(vip)实现双机热备以及自动切换主备
    linux安装openjdk1.8
  • 原文地址:https://www.cnblogs.com/liuxiaobo93/p/5530804.html
Copyright © 2020-2023  润新知