C#当动态数据过大画图有困难时怎么办?

这篇具有很好参考价值的文章主要介绍了C#当动态数据过大画图有困难时怎么办?。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

笔者最近写了一个监控软件, 数据量较大, 而所用插件刷新有压力. 所以就写了一个支持数据抽取的大数据维护类.

        // 设计需求:
        // 会发生数据的新增, 并且在数据达到上限时自动去掉前半截数据,
        // 高频的等间隔读取抽有数据,
        // 数据结构大小是可以在生成对象时提供, 并且对不提供大小参数的情况下使用默认大小
        // 数据量较大(大约有40万个元素),
        // 支持等距提取若干个数据,
        //      比如从最后20000个树种提取2000个数据(可通过参数设置这个数量),
        //      提取的规则是:返回提取的数据量, 将提取的数写入到调用者提供的list中

        // 设计思路:
        // 使用List<T> 或LinkedList<T>都有一些弊端:
        // List<T> 在去除数据时,需要将后半部分的数据移动到前面,对于大数据量来说,性能开销较大。
        // LinkedList<T> 在随机访问数据时,效率较低,因为需要从头开始遍历到目标位置。
        // 在C#中,没有现成的数据结构可以同时满足这些需求。因此,自己设计一个数据结构来实现这些功能。
        // 将两个List<T>结合使用,每个List<T>存储一半的数据。当数据达到上限时,丢弃第一个List<T>,并创建一个新的List<T>来存储新的数据。
        public interface IDataExtractor<T>
        {
            (int startIndex, double interval) ExtractDataFromEnd(int lastCount, int extractCount, List<T> targetList);
        }

        public class LargeDataStructure<T> : IDataExtractor<T>
        {
            private const int DefaultHalfMaxRecordsNum = 60 * 60 * 24 * 7 * 2; // Default half of max records
            private int halfMaxRecordsNum;
            private List<T> firstHalf;
            private List<T> secondHalf;
            public int AllDataCount => firstHalf.Count + secondHalf.Count;

            public LargeDataStructure(int? maxRecordsNum = null)
            {
                this.halfMaxRecordsNum = (maxRecordsNum ?? DefaultHalfMaxRecordsNum) / 2;
                this.firstHalf = new List<T>(halfMaxRecordsNum);
                this.secondHalf = new List<T>(halfMaxRecordsNum);
            }

            public void Append(T data)
            {
                if (secondHalf.Count >= halfMaxRecordsNum)
                {
                    // Discard the first half and move the second half to the first
                    firstHalf = secondHalf;
                    secondHalf = new List<T>(halfMaxRecordsNum);
                }

                if (firstHalf.Count < halfMaxRecordsNum)
                {
                    firstHalf.Add(data);
                }
                else
                {
                    secondHalf.Add(data);
                }
            }

            public T this[int index] // indexer
            {
                get
                {
                    if (index < halfMaxRecordsNum)
                    {
                        return firstHalf[index];
                    }
                    else
                    {
                        return secondHalf[index - halfMaxRecordsNum];
                    }
                }
            }

            public (int, double) ExtractDataFromEnd(int lastCount, int extractCount, List<T> targetList)
            {
                targetList.Clear();
                int startIndex = Math.Max(0, AllDataCount - lastCount);
                int validCount = AllDataCount - startIndex;
                double interval = 1;

                if (extractCount >= validCount)
                {
                    // 提取的数量大于等于最后的数据量
                    for (int i = startIndex; i < AllDataCount; i++)
                    {
                        targetList.Add(this[i]);
                    }
                }
                else
                {
                    interval = (double)(validCount-1) / (extractCount - 1); // 计算浮点数等距间隔

                    for (int i = 0; i < extractCount; i++)
                    {
                        int index = (int)Math.Round(startIndex + interval * i);
                        index = Math.Min(index, AllDataCount - 1); // 确保不超过数据范围
                        targetList.Add(this[index]);
                    }
                }
                return (startIndex, interval);
            }
        }
        private void TestExtractDataFromEnd()
        {
            LargeDataStructure<int> dataStructure = new LargeDataStructure<int>();

            // 添加一些数据用于测试
            for (int i = 0; i < 20; i++)
            {
                dataStructure.Append(i);
            }

            List<int> result = new List<int>();

            // 用例1.测试数据量不足,提取 30 个数据
            dataStructure.ExtractDataFromEnd(50, 30, result);
            Console.WriteLine("数据量不足,提取 30 个数据:");
            foreach (int data in result)
            {
                Console.Write(data + " ");
            }
            Console.WriteLine();

            // 再添加一些数据用于测试
            for (int i = 20; i < 100; i++)
            {
                dataStructure.Append(i);
            }


            // 用例2.测试整数等距提取,提取 5 个数据
            dataStructure.ExtractDataFromEnd(10, 5, result);
            Console.WriteLine("整数等距提取(10个数据中提取5个):");
            foreach (int data in result)
            {
                Console.Write(data + " ");
            }
            Console.WriteLine();

            // 用例3.测试浮点数等距提取,提取 20 个数据
            dataStructure.ExtractDataFromEnd(30, 20, result);
            Console.WriteLine("浮点数等距提取(30个数据中提取20个):");
            foreach (int data in result)
            {
                Console.Write(data + " ");
            }
            Console.WriteLine();

            // 用例4.测试尾部数超过需求数, 想从5个数中提取100个(不合法参数
            dataStructure.ExtractDataFromEnd(5, 100, result);
            Console.WriteLine("尾部数超过需求数(5个数据中提取100个):");
            foreach (int data in result)
            {
                Console.Write(data + " ");
            }
            Console.WriteLine();

            Console.ReadLine();
        }

当涉及处理大量数据时,LargeDataStructure类可以作为一个有用的工具。

该类可以存储大量数据,并提供了一些功能来操作和提取这些数据。以下是LargeDataStructure类的总结:

  1. LargeDataStructure类的目的是处理大量数据,它被设计为实现IDataExtractor接口。

  2. LargeDataStructure类内部使用两个List<int>对象(firstHalf和secondHalf)来存储数据。这两个列表分别表示数据的前半部分和后半部分。

  3. 构造函数LargeDataStructure(int? maxRecordsNum = null)允许你创建一个LargeDataStructure对象,并可选地指定最大记录数(maxRecordsNum)。如果未提供最大记录数,将使用默认值。

  4. 方法Append(int data)用于将数据添加到LargeDataStructure对象中。如果secondHalf列表的大小达到最大记录数的一半(halfMaxRecordsNum),则会丢弃firstHalf列表中的数据,并将secondHalf作为新的firstHalf。然后,创建一个新的空的secondHalf列表来接收后续的数据。

  5. LargeDataStructure类还提供了索引器(this[int index]),允许你通过索引访问LargeDataStructure对象中的数据。该索引器考虑了数据的分布,以正确地获取指定索引位置的数据。

  6. 方法ExtractDataFromEnd(int lastCount, int extractCount, List<int> targetList)允许你从LargeDataStructure对象的末尾提取指定数量的数据。你可以指定最后的数据数量(lastCount),要提取的数据数量(extractCount)以及目标列表(targetList)来接收提取的数据。

  7. 如果提取的数量大于等于最后的数据数量,ExtractDataFromEnd方法将提取所有最后的数据。否则,它将根据浮点数等距间隔从数据中提取相应数量的数据。在计算浮点数等距间隔时,会考虑有效数据的数量,以确保提取不超过数据的范围。

LargeDataStructure类提供了一种高效且灵活的方式来处理大量数据,并具有数据添加、数据访问以及从数据末尾提取数据的功能。这个类可以根据你的需求进行进一步的扩展和定制。文章来源地址https://www.toymoban.com/news/detail-490979.html

到了这里,关于C#当动态数据过大画图有困难时怎么办?的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • MySQL主从数据不一致,怎么办?

    先给大家说个身边的故事。 小伙伴二狗最近面宇宙厂,前面被问MySQL索引、锁、主从复制原理时答的都很开心。 当 面试官 问到 :“你们遇到主从不一致的问题怎么解决呢?你有什么更好的方案吗?” 二狗 懵了。不就是读写时候走主,纯读走从吗。。难道还有什么别的办法

    2023年04月20日
    浏览(51)
  • MySQL数据被误删怎么办?

    很多年前,被公司外派到一家单位驻场开发一个OA项目,两个开发对接各部门的需求,需求还要及时生效(一边开发一边使用)。有一次生产环境的一个bug本地没办法复现,由于没有测试人员,也就不存在测试环境,所以本地连了生产库去调试。不出意外的话要出意外了:在

    2024年02月01日
    浏览(42)
  • 电脑提示数据错误循环冗余检查怎么办?

    有些时候,我们尝试在磁盘上创建分区或清理硬盘时,还可能会遇到这个问题:数据错误循环冗余检查。这是如何导致的呢?我们又该如何解决这个问题呢?下面我们就来了解一下。 数据错误循环冗余检查怎么解决呢?在回答这个问题之前我们需要先了解一下导致问题的原因

    2024年02月12日
    浏览(57)
  • 领导临时要数据库文档怎么办?

    很多时候,我们为了着急忙慌赶项目进度,很容易忽略整理文档这件事 某一天,领导心血来潮,要搞一次突击检查, 想看看我们的数据库设计的是否规范, 但他又不想亲自去数据库查验(毕竟这么大领导) 那么,我们该怎么办? 第一种方法:离职,世界那么大,我想去看

    2024年02月08日
    浏览(40)
  • 数据库同步 Elasticsearch 后数据不一致,怎么办?

    Q1:Logstash 同步 postgreSQL 到 Elasticsearch 数据不一致。 在使用 Logstash 从 pg 库中将一张表导入到 ES 中时,发现 ES 中的数据量和 PG 库中的这张表的数据量存在较大差距。如何快速比对哪些数据没有插入?导入过程中,Logstash 日志没有异常。PG 中这张表有 7600W。 Q2:mq 异步双写数

    2024年02月15日
    浏览(41)
  • 服务器数据被盗了该怎么办

    随着互联网的发展,如今人们习惯了在互联网上分享生活、购物等等。便捷了人们的生活,也让互联网企业蓬勃生机,但同时也暗藏着危机。其中服务器被入侵是常见的黑客攻击方式,不仅会给企业带来经济损失,同时也让企业在公众面前失去了信誉。下面我们来看看一些案

    2024年02月04日
    浏览(60)
  • mysql数据库忘记密码了怎么办

    本人用的mysql8版本 看到网上很多教程,什么修改配置文件my.ini。在8版本根本没用。以下是8版本解决办法。亲测可用。 1、用管理员身份打开命令行工具。(强调:管理员身份) 2、停止mysql服务: 3、输入以下命令无密码启动mysql 4、 另开一个命令行窗口,输入mysql -u root无密

    2024年02月11日
    浏览(46)
  • app渗透测试抓不到数据包怎么办?

    1、app渗透测试常见的模拟器有夜神,闪电模拟器等。 2、遇到问题,数据包抓不到? (1)最常见的解决方法调低模拟器安卓版本,使用MuMu模拟器,大部分app都适用,但是也有些app无法抓取。 (2)安装证书到系统根证书,可以解决上面的问题(一般情况下手机在安装了BurpSuite的伪证

    2024年02月13日
    浏览(37)
  • 数据分析过程中,发现数值缺失,怎么办?

    按照数据缺失机制,数据分析过程中,我们可以将其分为以下几类: (1)完全随机缺失(MCAR):所缺失的数据发生的概率既与已观察到的数据无关,也与未观察到的数据无关。 (2)随机缺失(MAR):假设缺失数据发生的概率与所观察到的变量是有关的,而与未观察到的数

    2024年02月05日
    浏览(52)
  • 使用JOIN查询数据重复,怎么办?使用count统计怎么写SQL?

    比如现在有两个表tbl_a和tbl_b,如下: tbl_a id name 1 Bruce 2 Mike 3 Angela tbl_b id a_id :-: :-: 1 1 2 1 3 2 4 3 5 3 6 3 这时候我们如果联查的话,就会出现重复数据: 这样查出来的数据就会像下面这样: id name 1 Bruce 1 Bruce 2 Mike 3 Angela 3 Angela 3 Angela 去重的方法有三种: 一、distinct 二、group

    2024年02月16日
    浏览(52)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包