Fork me on GitHub

WinForm(九)UI加载“大”数据

  由于WinForm的UI是绘制的,所以在加载大量数据数据时会有一定的延时,本篇就讨论几个减少延时的方法。

  在加载有规律数据时,可以考虑用递归,简单方便快捷来加载数据,如下,把一个文件夹下的所有文件或文件夹加载到树形菜单上,以树形展示,代码实现如下。

var rootPath = "d://abc/bcd";
var rootNode = treeView1.Nodes.Add(rootPath, Path.GetFileName(rootPath));
LoadFile(rootNode);


void LoadFile(TreeNode node)
{
    foreach (var file in Directory.GetFiles(node.Name))
    {
        node.Nodes.Add(file, Path.GetFileName(file));
    }
    foreach (var dir in Directory.GetDirectories(node.Name))
    {
        var childNode = node.Nodes.Add(dir, Path.GetFileName(dir));
        LoadFile(childNode);
    }
}

  上面的是练了个手,接下来我们加载一个大点的数据,一个全国的行政区划表,有省,市 ,县,乡镇四级,一共49000多条数据,数据字段有sid,pid,name。本篇我们主要是看从内存list到UI上,所以加载数据不是重点,可以是数据库,也可以是文件中,最终数据会在内存的list中。

class Province
{
    public string sid { get; set; }
    public string pid { get; set; }
    public string name { get; set; }
}
var rootNode = treeView1.Nodes.Add("0", "中国");
LoadProvince(rootNode);

void LoadProvince(TreeNode node)
{
    foreach (var item in list.Where(s => s.pid == node.Name))
    {
        var childNode = node.Nodes.Add(item.sid, item.name);
        LoadProvince(childNode);
    }
}

  如果直接用递归加载,速度太慢,为了加快速度,就得并行加载了,于时就增加Task.Run,因为是大多线程中异步操作UI,所以还得用this.Invoke,代码如下。运行,会看到,速度显然快了不了,但还不是理想结果,理想是无感。

var rootNode = treeView1.Nodes.Add("0", "中国");
LoadProvince(rootNode);
void LoadProvince(TreeNode node)
{
    Task.Run(() =>
    {
        foreach (var item in list.Where(s => s.pid == node.Name).OrderBy(s => s.sid))
        {
            this.Invoke(() =>
            {
                node.Nodes.Add(item.sid, item.name);
                if (node.Level == 0)
                {
                    node.Expand();
                }
            });
        }
        foreach (TreeNode childNode in node.Nodes)
        {
            LoadProvince(childNode);
        }
    });
}

  后来又想到,可不可以把树形菜单给序列化,窗体启动时,返序列化回来,用BinaryFormatter来实现(现在官方不鼓励用),首先TreeView不支持序列化,只能换成TreeView的Nodes属笥来序列化。窗体启动时,它的加载速度与上面的异步递归差不多,没有明显改善。

  即然一次加载大量数据不行,就再换一下思路,一次加载少一些,因为是UI,用户肯定有交互,利用用户的交互来触发加载他想要看的数据,这个少量多少为好,对于树形控件来说,如果想看不出来,那就是两级,用户点开第二级的时候再加载两级,这样用户始终感觉用数据。

private void Form1_Load(object sender, EventArgs e)
{
    var rootNode = treeView2.Nodes.Add("0", "中国");
    LoadProvince(rootNode, 1);
    this.treeView2.BeforeExpand += new System.Windows.Forms.TreeViewCancelEventHandler(this.treeView2_BeforeExpand);
}
private void treeView2_BeforeExpand(object sender, TreeViewCancelEventArgs e)
{
    if (e.Node != null & e.Node.Nodes != null &&e.Node.Nodes.Count > 0)
    {
        e.Node.Nodes.Clear();
        LoadProvince(e.Node, 1);
    }
}

  性能的问题永远没有最好,也没有一种方式能就通吃各种场景,得一个个换思路来解决,不过上面的思路肯定不是最好的,如果你有一次性加载全部更快的解决方案,请告我,我实现,然后再分享给更多的人。

  想要更快更方便的了解相关知识,可以关注微信公众号

 

 

posted @ 2022-12-15 21:43  桂素伟  阅读(103)  评论(0编辑  收藏  举报