C#学习笔记--实现一个可以重复权重并且能够自动排序的容器--MultiplySortedSet

news/2024/4/28 14:35:15/文章来源:https://blog.csdn.net/qq_52855744/article/details/130582226

目录

    • 前言
    • SortedSet
      • C#自带类型
      • 自定义类
      • SortedSet权值重复
    • 需求
    • 自定义容器 -- MultiplySortedSet
      • 核心实现思路
    • MultiplySortedSet 使用
      • C#自带类型
      • 自定义类

前言

最近需要在C#中实现一个功能
有一个容器,该容器能自动对里面的元素进行排序,类似C++的优先队列,也就是堆。
但是查询了C#提供的一些容器,只有 SortedSet 和我们的需求很符合。

SortedSet

其实就是一个可以自动排序的容器

我们在Unity下使用看看

C#自带类型

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class SortedSetTest : MonoBehaviour
{private void Awake(){SortedSet<int> ssi = new SortedSet<int>();ssi.Add(10);ssi.Add(1);ssi.Add(5);ssi.Add(-1);foreach (var item in ssi){print(item);}}
}

在这里插入图片描述

自定义类

public class Student : IComparable<Student>
{private int id;public Student(int id_in){id = id_in;}public int CompareTo(Student other){return id.CompareTo(other.id);}public override string ToString(){return $"Student ID {id}";}
}
using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class SortedSetTest : MonoBehaviour
{private void Awake(){SortedSet<Student> sss = new SortedSet<Student>();sss.Add(new Student(10));sss.Add(new Student(1));sss.Add(new Student(5));sss.Add(new Student(-1));foreach (var item in sss){print(item);}}
}

在这里插入图片描述

SortedSet权值重复

我们能看出来它可以自动进行排序

但是如果权值重复了该怎么办?我们来试试看

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class SortedSetTest : MonoBehaviour
{private void Awake(){SortedSet<int> ssi = new SortedSet<int>();ssi.Add(10);ssi.Add(1);ssi.Add(1);ssi.Add(-1);foreach (var item in ssi){print(item);}}
}

在这里插入图片描述
我们发现相同的 1 并没有被加入到集合中,这也算是 SortedSet 的特性,毕竟是 Set 嘛。

需求

但是我们现在需要的是重复的权值也可以保存下来,并且进行排序。那该怎么办呢?
这里我并没有找到C#自带的符合需求的容器(C++有优先队列),所以我们这里自定义一个容器来实现。

自定义容器 – MultiplySortedSet

多说无益,直接上代码。

using System;
using System.Collections;
using System.Collections.Generic;namespace Util
{//排序方法public enum MultiplySortedType{Ascending,      //升序Descending,     //降序}public class MultiplySortedSet<T> : IEnumerable<T> where T : IComparable<T>{private SortedSet<KeyValuePair<T, Guid>> _elements;private Dictionary<T, Guid> _elementIds;public MultiplySortedSet(IComparer<KeyValuePair<T, Guid>> comparer = null){//默认为升序if(comparer == null){comparer = new ComparerAscending();}_elements = new SortedSet<KeyValuePair<T, Guid>>(comparer);_elementIds = new Dictionary<T, Guid>();}public MultiplySortedSet(MultiplySortedType sortedType){IComparer<KeyValuePair<T, Guid>> comparer = null;switch (sortedType){case MultiplySortedType.Ascending:comparer = new ComparerAscending();break;case MultiplySortedType.Descending:comparer = new ComparerDescending();break;}_elements = new SortedSet<KeyValuePair<T, Guid>>(comparer);_elementIds = new Dictionary<T, Guid>();}public void Add(T element){Guid guid = Guid.NewGuid();_elements.Add(new KeyValuePair<T, Guid>(element, guid));_elementIds[element] = guid;}public void Remove(T element){//如果Ids存在,则说明_elements里面肯定有该元素//RemoveWhere是为了删除单一元素//因为不同T对象的Compare可能相同,但是Guid肯定不同if (_elementIds.ContainsKey(element)){_elements.RemoveWhere(x => x.Key.CompareTo(element) == 0 && x.Value.CompareTo(_elementIds[element]) == 0);_elementIds.Remove(element);}}public bool Contains(T element){return _elementIds.ContainsKey(element);}public int Count{get { return _elements.Count; }}public IEnumerator<T> GetEnumerator(){foreach (var element in _elements){yield return element.Key;}}IEnumerator IEnumerable.GetEnumerator(){return GetEnumerator();}private class ComparerAscending : IComparer<KeyValuePair<T, Guid>>{public int Compare(KeyValuePair<T, Guid> x, KeyValuePair<T, Guid> y){//先调用T的Compare,如果相等就比较Guidint result = x.Key.CompareTo(y.Key);if (result == 0){result = x.Value.CompareTo(y.Value);}return result;}}private class ComparerDescending : IComparer<KeyValuePair<T, Guid>>{public int Compare(KeyValuePair<T, Guid> x, KeyValuePair<T, Guid> y){//先调用T的Compare,如果相等就比较Guidint result = y.Key.CompareTo(x.Key);if (result == 0){result = x.Value.CompareTo(y.Value);}return result;}}}
}

核心实现思路

其实核心思路就一点,多做一层封装,当对象权重相等时,比较另外一个值。

  1. 首先 SortedSet 不可以存储相同权重,所以我们这里使用 KeyValuePair 来封装一层。

Key为排序的对象,这里就拿泛型T替代。
Value存在的意义就是,当某些T对象权重相等时,拿Value的值来作比较。也就是说每一个相等权重的对象T的Value一定要不一样。所以这里Value可以拿System.Guid来做。

  1. 自定义一个 Comparer ,如果Key值相等,那么就比较Value值。

MultiplySortedSet 使用

C#自带类型

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Victor_Util;public class Test : MonoBehaviour
{private void Awake(){MultiplySortedSet<int> vs = new MultiplySortedSet<int>();vs.Add(9);vs.Add(1);vs.Add(1);vs.Add(1);vs.Add(-5);vs.Add(100);foreach (var item in vs){print(item);}}
}

在这里插入图片描述

自定义类

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Victor_Util;public class Test : MonoBehaviour
{private void Awake(){MultiplySortedSet<Student> vs2 = new MultiplySortedSet<Student>();vs2.Add(new Student(9));vs2.Add(new Student(1));vs2.Add(new Student(1));vs2.Add(new Student(1));vs2.Add(new Student(-5));vs2.Add(new Student(100));foreach (var item in vs2){print(item);}}
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_298111.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

FS5175AE降压型1-4节锂电池充电芯片

FS5175AE是一款工作于5V到24V的多串锂电池同步开关降压充电管理芯片。内置MOS管集成了低导通阻抗的NMOS&#xff0c;FS5175AE采用1MHz同步开关架构&#xff0c;实现高 效率充电并简化外围器件&#xff0c;降低BOM成本。通过调节检测电阻&#xff0c;可实现**2A充电电流&#xf…

SpringCloud(22):Sentinel对Feign的支持

Sentinel 适配了 Feign组件。如果想使用&#xff0c;除了引入 spring-cloud-starter-alibaba-sentinel 的依赖外还需要 2个步骤&#xff1a; 配置文件打开 Sentinel 对 Feign 的支持&#xff1a;feign.sentinel.enabledtrue加入 spring-cloud-starter-openfeign 依赖使 Sentin…

基于Linux系统在线安装RabbitMQ

一、前言 二、Erlang下载安装 三、RabbitMQ下载安装 三、RabbitMQ Web界面管理 一、前言 本次安装使用的操作系统是Linux centOS7。 二、Erlang下载安装 在确定了RabbitMQ版本号后&#xff0c;先下载安装Erlang环境。下面演示操作过程&#xff1a; Erlang下载链接&#…

[工具]Pytorch-lightning的使用

Pytorch-lightning的使用 Pytorch-lightning介绍Pytorch-lightning与Pytorch的区别Pytorch-lightning框架的优势Pytorch-lightning框架 重要资源 Pytorch-lightning介绍 这里介绍Pytorch_lighting框架. Pytorch-lightning与Pytorch的区别 Pytorch-lightning可以简单的看作是…

强化学习p3-策略学习

Policy Network (策略网络) 我们无法知道策略函数 π \pi π所以要做函数近似&#xff0c;求一个近似的策略函数 使用策略网络 π ( a ∣ s ; θ ) \pi(a|s;\theta) π(a∣s;θ) 去近似策略函数 π ( a ∣ s ) \pi(a|s) π(a∣s) ∑ a ∈ A π ( a ∣ s ; θ ) 1 \sum_{a\in …

《狂飙》原著来了,邀你重新见证

2023年的开篇十分精彩且丰富&#xff0c;经历过生活的不幸&#xff0c;新的一年万物复兴&#xff0c;每个人心底那颗躁动的心又重新释放&#xff0c;希望新的开始不负所望&#xff0c;年末复盘时所得皆所愿&#xff01; 开篇 开年影视第一炮&#xff0c;炸出了所有人被压抑的内…

告别PPT手残党!这6款AI神器,让你秒变PPT王者!

如果你是一个PPT手残党&#xff0c;每每制作PPT总是让你焦头烂额&#xff0c;那么你一定需要这篇幽默拉风的推广文案&#xff01; 我向你保证&#xff0c;这篇文案将帮助你发现6款AI自动生成PPT的神器&#xff0c;让你告别PPT手残党的身份&#xff0c;成为一名PPT王者。 无论…

计算机图形学 | 实验六:旋转立方体

计算机图形学 | 实验六&#xff1a;旋转立方体 计算机图形学 | 实验六&#xff1a;旋转立方体Z-缓冲GLM函数库PVM矩阵PVM矩阵的使用 华中科技大学《计算机图形学》课程 MOOC地址&#xff1a;计算机图形学&#xff08;HUST&#xff09; 计算机图形学 | 实验六&#xff1a;旋转…

单词词义、词性、例句查询python代码

单词发音、词义、词性、例句查询、输出结果更简洁&#xff0c;一次可查多个单词 运行该代码&#xff0c;命令窗口输入单词&#xff0c;单词用“/”分开&#xff0c;例如&#xff1a;noisy/problem/community/neighbor 可以更多。先安装两个python包requests、 beautifulsoup4&…

卖一辆亏5.8万美元!福特的困扰

随着电动化进入关键的「抢量」周期&#xff0c;加上年初掀起的降价潮&#xff0c;对于还无法适应转型节奏的传统汽车制造商来说&#xff0c;现在是一个艰难的时刻。 本月初&#xff0c;福特首席执行官Jim Farley表示&#xff0c;电动汽车市场的降价是"令人担忧的趋势"…

2023/5/8总结

JAVA基础知识&#xff08;2&#xff09; 1.方法 1、方法定义 格式&#xff1a;public static void 方法名&#xff08;&#xff09;{ //方法体 } 2、方法调用 格式&#xff1a;方法名&#xff08;&#xff09;&#xff1b; 3、方法的通用格式 public static 返回值类型方法名&…

车载测试-can报文解析规则实例

报文解析 报文组成 一般报文主要有以下几个参数&#xff08;比较全的情况下&#xff09; 例 解析报文时主要用到的是帧ID和帧数据 帧ID 接收到的帧ID是十六进制的形式&#xff0c;由29位标识符转换的&#xff0c;目前大多数的通信协议中都直接给出了相应的帧ID&#xff0c…

mathtype不激活能用吗 mathtype产品密钥如何取得

在文档中输入数学式子时一般会用到mathtype&#xff0c;虽然mathtype为广大用户提供了一定期限的试用期&#xff0c;但试用期后如果没有成为正式用户&#xff0c;那么部分功能可能就用不了了。有些小伙伴可能会对mathtype不激活能用吗&#xff0c;mathtype产品密钥如何取得这两…

kt:reified和sam转换(Single Abstract Method Conversions)

什么是refied关键字 ​由于我们都知道Kotlin和Java一样都存在着泛型擦除问题&#xff0c;而Kotlin它知道Java所带来的这个问题&#xff0c;所以对此Kotlin留了一个后门&#xff0c;就是通过inline函数保证使得泛型类的类型实参在运行时能够保留&#xff0c;这样的操作 Kotlin 中…

git简介和使用、基础命令

文章目录 一、git的安装与配置二、Git工作区原理三、Git的使用和仓库的创建四、Git的常用操作五、配置公钥六、IDEA中配置Git 一、git的安装与配置 https://tortoisegit.org/ 下载对应版本安装即可 注意&#xff1a;配置中输入邮箱和密码一定要和自己的git账户一致 git的配置…

copilot 逆向

原文&#xff1a; copilot-explorer | Hacky repo to see what the Copilot extension sends to the server 对我来说&#xff0c;Github Copilot 极其有用。它经常能神奇地读懂我的想法并给出有用的建议。最让我惊讶的是&#xff0c;它能够从周围的代码中正确地“猜测”出函数…

设计模式 -- 备忘录模式

前言 月是一轮明镜,晶莹剔透,代表着一张白纸(啥也不懂) 央是一片海洋,海乃百川,代表着一块海绵(吸纳万物) 泽是一柄利剑,千锤百炼,代表着千百锤炼(输入输出) 月央泽,学习的一种过程,从白纸->吸收各种知识->不断输入输出变成自己的内容 希望大家一起坚持这个过程,也同…

电脑cpu占用率高?怎么办?1分钟快速解决!

案例&#xff1a;电脑cup过高怎么办&#xff1f; 【我的电脑运行缓慢&#xff0c;导致我学习和工作的效率很低。刚刚查看了一下电脑&#xff0c;发现它的cpu占用率很高。有没有小伙伴知道如何解决此电脑cpu过高的问题&#xff1f;】 电脑是我们生活中不可缺少的工具&#xff…

1— .Net MVC之控制器

在上下文中使用的控制器 问题 答案 什么是控制器&#xff1f; 控制器包含用于接收请求、更新应用程序状态或模型以及选择将发送给客户端的响应的逻辑 控制器有什么用&#xff1f; 控制器是MVC项目的核心&#xff0c;并包含Web应用程序的逻辑 如何使用控制器&#xff1f; …

接口自动化测试神器:Python+Requests+Unittest让你的测试用例飞起来

B站首推&#xff01;2023最详细自动化测试合集&#xff0c;小白皆可掌握&#xff0c;让测试变得简单、快捷、可靠 随着互联网的发展&#xff0c;越来越多的应用程序采用了分布式架构&#xff0c;并通过API接口进行数据交换。因此&#xff0c;接口自动化测试已经成为了保证软件质…