改善C#程式的方法-3 比較器和LINQ排序

来源:https://www.cnblogs.com/wwwen/archive/2022/07/27/16524999.html
-Advertisement-
Play Games

一 創建對象時考慮實現比較器 假設有這樣的場景,有一個40個人的學生列表,業務中需針對學生的成績來進行排序。 可以考慮用IComparable介面和ICompare介面實現: class Program { static void Main(string[] args) { var stus = n ...


一 創建對象時考慮實現比較器

假設有這樣的場景,有一個40個人的學生列表,業務中需針對學生的成績來進行排序。

可以考慮用IComparable介面和ICompare介面實現:

class Program
{
    static void Main(string[] args)
    {
        var stus = new List<Student>();
        stus.Add(new Student() { Name = "zhangsan", EnglishGrades = 80.5, MathGrades = 90 });
        stus.Add(new Student() { Name = "lisi", EnglishGrades = 74, MathGrades = 91 });
        stus.Add(new Student() { Name = "wangwu", EnglishGrades = 94, MathGrades = 85.5 });
        stus.Add(new Student() { Name = "zhaoliu", EnglishGrades = 88.5, MathGrades = 86 });
      
        stus.Sort();
        Console.WriteLine("使用預設比較器排序:");
        foreach (var stu in stus)
        {
            Console.WriteLine($"Name:{stu.Name},\tEnglish:{stu.EnglishGrades},\tMath:{stu.MathGrades}");
        }
        
        stus.Sort(new MathComparer());
        Console.WriteLine("使用自定義比較器排序:");
        foreach (var stu in stus)
        {
            Console.WriteLine($"Name:{stu.Name},\tEnglish:{stu.EnglishGrades},\tMath:{stu.MathGrades}");
        }
        Console.ReadLine();
    }
}

//Student通過IComparable介面,實現預設比較器
class Student : IComparable<Student>
{
    public string Name { get; set; }

    public double EnglishGrades { get; set; }

    public double MathGrades { get; set; }

    public int CompareTo(Student stu)
    {
        if (EnglishGrades > stu.EnglishGrades)
        {
            return 1;
        }
        else if (EnglishGrades == stu.EnglishGrades)
        {
            return 0;
        }
        else
        {
            return -1;
        }
        //return EnglishGrades.CompareTo(stu.EnglishGrades); double類型的預設比較方法
    }
}

//通過IComparer介面實現自定義的比較器
class MathComparer : IComparer<Student>
{
    public int Compare(Student x, Student y)
    {
        return x.MathGrades.CompareTo(y.MathGrades);
    }
}

輸出:

使用預設比較器排序:
Name:lisi,      English:74,     Math:91
Name:zhangsan,  English:80.5,   Math:90
Name:zhaoliu,   English:88.5,   Math:86
Name:wangwu,    English:94,     Math:85.5
使用自定義比較器排序:
Name:wangwu,    English:94,     Math:85.5
Name:zhaoliu,   English:88.5,   Math:86
Name:zhangsan,  English:80.5,   Math:90
Name:lisi,      English:74,     Math:91

 二 使用LINQ取代集合中的比較器

上述的方法實現的排序存在2個問題:

  • 可擴展性太低,如果存在新的排序要求,就必須實現新的比較器;
  • 對代碼的侵入性太高,為類型繼承了介面,新增了方法。

LINQ提供了類似於SQL的語法來實現遍歷、篩選和投影集合的強大功能,可以實現上述的排序要求。

static void Main(string[] args)
{
    var stus = new List<Student>();
    stus.Add(new Student() { Name = "zhangsan", EnglishGrades = 80.5, MathGrades = 90 });
    stus.Add(new Student() { Name = "lisi", EnglishGrades = 74, MathGrades = 91 });
    stus.Add(new Student() { Name = "wangwu", EnglishGrades = 94, MathGrades = 85.5 });
    stus.Add(new Student() { Name = "zhaoliu", EnglishGrades = 88.5, MathGrades = 86 });

    var orderByStus = from s in stus orderby s.EnglishGrades select s;
    //orderByStus = stus.OrderBy(s => s.EnglishGrades);
    foreach (var stu in orderByStus)
    {
        Console.WriteLine($"Name:{stu.Name},\tEnglish:{stu.EnglishGrades},\tMath:{stu.MathGrades}");
    }
    Console.WriteLine();
    
    orderByStus = from s in stus orderby s.MathGrades select s;
    //orderByStus = stus.OrderBy(s => s.MathGrades);
    foreach (var stu in orderByStus)
    {
        Console.WriteLine($"Name:{stu.Name},\tEnglish:{stu.EnglishGrades},\tMath:{stu.MathGrades}");
    }

    Console.ReadLine();
}

LINQ此功能的實現本身是藉助於FCL泛型集合的比較器、迭代器和索引器。LINQ封裝了這些功能,讓我們使用更加方便。

在命名空間System.Linq下的Enumerable方法中為泛型集合提供了很多擴展方法。

如排序中使用到的OrderBy方法:

   public static IOrderedEnumerable<TSource> OrderBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector);

它為繼承了IEnumerable<T>介面的集合提供排序的功能。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • [數據結構-線性表1.1] 數組(.NET源碼學習) 數組,一種數據類型(在絕大數語言中不是基本數據類型)且為引用類型,在記憶體中以連續的記憶體單元進行分配,所以其大小在創建對象後為定值,不可更改。 一.記憶體分配 對於兩種不同數據類型而言,其記憶體分配方式是不同的。值類型直接在棧(C#中稱為堆棧Stack ...
  • 一 併發編程簡介 1.1 關於併發和並行 併發和並行的概念: 併發:(Concurrent),在某個時間段內,如果有多個任務執行,即有多個線程在操作時,如果系統只有一個CPU,則不能真正同時進行一個以上的線程, 它只能把CPU運行時間劃分成若幹個時間段,再將時間段分配給各個線程執行,在一個時間段的線 ...
  • 前言 接著上周寫的截圖控制項繼續更新 縮放操作。 1.WPF實現截屏「仿微信」 2.WPF 實現截屏控制項之移動(二)「仿微信」 正文 實現拉伸放大或縮小縮放操作需在矩形四個方向繪製8個Thumb,這裡有兩種方式 1)可以自行在XAML中硬編寫8個Thumb 2)使用裝飾器Adorner 本章使用了第二 ...
  • ASP.NET Web 應用 Docker踩坑歷程發表後,也開始使用Docker了,然而發佈的過程比較痛苦,經常發生下圖的事情: 據說是nuget包還原時發生錯誤 百度了半天也找不到解決的方法,而發生的概率有相當高,很是無語。 仔細看了自動生成的Dockerfile FROM mcr.microso ...
  • 今天在處理一個接收API通過Post方式傳送Json數據的方法時,碰到接收的Json數據一直是空的問題。最好找了好久才解決,現在把需要的問題列出來。 1. 在一般處理程式中,需要設置 context.Request.InputStream.Position = 0; 剛開始設置了這個,但後面還是為空 ...
  • 前言 如標題所述,在ASP.NET應用程式開發中,兩個集合做比較時 我們使用微軟IEnumerable封裝的 Except/Intersect/Union 取 差集/交集/並集 方法是非常的方便的; 但以上對於不太熟悉的小伙伴來講,在遇到求包含引用類型(不包含string)集合時就非常的苦惱; 下麵 ...
  • WPF的拖曳效果,基本配置一下,就可以了,但是自繪的話,就得自己控制,按鍵點擊,按鍵移動和按鍵鬆開的事件,與其配合達到目的。 ...
  • 1.前言 hi,大家好,我是三合。我是怎麼想起寫一篇關於資料庫快速批量插入的博客的呢?事情起源於我們工作中的一個需求,簡單來說,就是有一個定時任務,從資料庫里獲取大量數據,在應用層面經過處理後再把結果批量插入回到資料庫里。這個任務每十分鐘執行一次,但是有的時候數據量太大,迴圈插入資料庫的時候會超時, ...
一周排行
    -Advertisement-
    Play Games
  • 1.部署歷史 猿友們好,作為初來實習的我,已經遭受社會的“毒打”,所以請容許我在下麵環節適當吐槽,3Q! 傳統部署 ​ 回顧以往在伺服器部署webapi項目(非獨立發佈),dotnet環境、守護進程兩個逃都逃不掉,正常情況下還得來個nginx代理。不僅僅這仨,可能牽扯到yum或npm。node等都要 ...
  • 隨著技術的進步,跨平臺開發已經成為了標配,在此大背景下,ASP.NET Core也應運而生。本文主要基於ASP.NET Core+Element+Sql Server開發一個校園圖書管理系統為例,簡述基於MVC三層架構開發的常見知識點,前一篇文章,已經簡單介紹瞭如何搭建開發框架,和登錄功能實現,本篇... ...
  • 這道題只要會自定義cmp恰當地進行排序,其他部分沒有什麼大問題。 上代碼: 1 #include<bits/stdc++.h> 2 using namespace std; 3 int n,s,h1,h2,cnt; 4 struct apple{ 5 int height,ns;//height為蘋 ...
  • 這篇文章主要描述RPC的路由策略,包括為什麼需要請求隔離,為什麼不在註冊中心中實現請求隔離以及不同粒度的路由策略。 ...
  • 簡介: 中介者模式,屬於行為型的設計模式。用一個中介對象來封裝一系列的對象交互。中介者是各對象不需要顯式地相互引用,從而使其耦合鬆散,而且可以獨立地改變他們之間的交互。 適用場景: 如果平行對象間的依賴複雜,可以使用中介者解耦。 優點: 符合迪米特法則,減少成員間的依賴。 缺點: 不適用於系統出現對 ...
  • 【前置內容】Spring 學習筆記全系列傳送門: Spring學習筆記 - 第一章 - IoC(控制反轉)、IoC容器、Bean的實例化與生命周期、DI(依賴註入) Spring學習筆記 - 第二章 - 註解開發、配置管理第三方Bean、註解管理第三方Bean、Spring 整合 MyBatis 和 ...
  • 簡介: 享元模式,屬於結構型的設計模式。運用共用技術有效地支持大量細粒度的對象。 適用場景: 具有相同抽象但是細節不同的場景中。 優點: 把公共的部分分離為抽象,細節依賴於抽象,符合依賴倒轉原則。 缺點: 增加複雜性。 代碼: //用戶類 class User { private $name; fu ...
  • 這次設計一個通用的多位元組SPI介面模塊,特點如下: 可以設置為1-128位元組的SPI通信模塊 可以修改CPOL、CPHA來進行不同的通信模式 可以設置輸出的時鐘 狀態轉移圖和思路與多位元組串口發送模塊一樣,這裡就不給出了,具體可看該隨筆。 一、模塊代碼 1、需要的模塊 通用8位SPI介面模塊 `tim ...
  • AOP-03 7.AOP-切入表達式 7.1切入表達式的具體使用 1.切入表達式的作用: 通過表達式的方式定義一個或多個具體的連接點。 2.語法細節: (1)切入表達式的語法格式: execution([許可權修飾符] [返回值類型] [簡單類名/全類名] [方法名]([參數列表]) 若目標類、介面與 ...
  • 測試一、虛繼承與繼承的區別 1.1 單個繼承,不帶虛函數 1>class B size(8): 1> + 1> 0 | + (base class A) 1> 0 | | _ia //4B 1> | + 1> 4 | _ib //4B 有兩個int類型數據成員,占8B,基類邏輯存在前面 1.2、單個 ...