c# Linq常用的小技巧
前言
在C#語言發(fā)展的歷史長河中,Linq是一個極其重要的里程碑!
Linq的語法吸取了SQL語法的特性,同時配合Lambda表達式又可以使代碼更加優(yōu)雅!
可以這么說,用好了Linq可以大大提高程序猿的工作效率,畢竟我們的日常工作本質就是對數(shù)據(jù)的處理。經歷了十多年的發(fā)展,現(xiàn)在微軟自帶的內庫包含的Linq函數(shù)已經非常多了,幾乎滿足我們日常工作。
下面根據(jù)一個對科室數(shù)據(jù)操作的例子,就個人覺得日常高頻使用的Linq小技巧貼出來,權當是做個筆記了。
初始化數(shù)據(jù)
定義模型
這里定義一個科室對象,模擬我們日常工作的科室信息??剖掖嬖趯蛹夑P系,還有一個員工數(shù)量的屬性。模型如下:
public class DepartmentDto
{
public int Id { get; set; }
public int? ParentId { get; set; }
public string Name { get; set; }
public string TelPhone { get; set; }
public string Address { get; set; }
public string Remark { get; set; }
public int EmployeeNumber { get; set; }
}
初始化數(shù)據(jù)
public List<DepartmentDto> InitDepartmentData()
{
List<DepartmentDto> lst = new List<DepartmentDto>();
lst.AddRange(new DepartmentDto[] {
new DepartmentDto() {
Address ="一馬路XX號",
Id=1,
Name="一級一號科室",
Remark="",
TelPhone="0731-6111111",
EmployeeNumber=3,
},
new DepartmentDto() {
Address ="二馬路XX號",
Id=2,
Name="一級二號科室",
Remark="",
TelPhone="0731-6111111",
EmployeeNumber=4,
},
new DepartmentDto() {
Address ="三馬路XX號",
Id=3,
Name="一級三號科室",
Remark="",
TelPhone="0731-6222222",
EmployeeNumber=6,
},
new DepartmentDto() {
Address ="一馬路XX號",
ParentId=1,
Id=4,
Name="二級一號科室",
Remark="",
TelPhone="0731-6222222",
EmployeeNumber=7,
},
new DepartmentDto() {
Address ="二馬路XX號",
ParentId=2,
Id=5,
Name="二級二號科室",
Remark="",
TelPhone="0731-6222222",
EmployeeNumber=5,
},
});
return lst;
}
獲取未存在父級科室的科室集合
List<DepartmentDto> lstDepartItems = InitDepartmentData();
//1、獲取未存在父級科室的科室集合 1、2、3
List<DepartmentDto> notExistsParentDepartmentIdLst = lstDepartItems
.Where(p => !p.ParentId.HasValue)
.ToList();
這里比較簡單,Where內校驗下ParentId的值為空即可,不使用Linq則需要自己手寫一個循環(huán)搞定,如下:
List<DepartmentDto> notExistsParentDepartmentIdLst_1 = new List<DepartmentDto>();
foreach (DepartmentDto department in lstDepartItems)
{
if (!department.ParentId.HasValue)
notExistsParentDepartmentIdLst_1.Add(department);
}
這么看感覺便捷性不太明顯是吧~~ 沒事,萬丈高樓平地起,咋們循行漸進~
獲取存在子科室的科室集合
//2、獲取存在子科室的科室集合 1、2
List<DepartmentDto> existsParentDepartmentIdLst1 = lstDepartItems
.Where(p => lstDepartItems.Select(k => k.ParentId).Contains(p.Id))
.ToList();
這里通過引用了外部的集合對象進行關聯(lián),在Where內對子科室的ParentId字段與當前集合的科室Id校驗,從而得到已存在子科室的科室集合。如果不使用Linq則自己需要寫兩個循環(huán)才能搞定。
如下:
List<DepartmentDto> existsParentDepartmentIdLst1_1 = new List<DepartmentDto>();
foreach (DepartmentDto parentDepart in lstDepartItems)
{
foreach (DepartmentDto childDepart in lstDepartItems)
{
if (parentDepart.Id == childDepart.ParentId)
{
existsParentDepartmentIdLst1_1.Add(parentDepart);
continue;
}
}
}
這么看,Linq帶來的便捷性是否足夠明顯了,代碼優(yōu)雅了太多了~
科室根據(jù)地址分組,獲取科室總數(shù)、科室平均數(shù)
var groupDto = lstDepartItems
.GroupBy(p => p.Address)
.Select(p => new
{
Address = p.Key,
SumEmployeeCount = p.Sum(p => p.EmployeeNumber),
AvgEmployeeCount = p.Average(p => p.EmployeeNumber),
}).ToList();
獲取兩個集合不相等的元素
引用類型的比較處理
這里需要留意我們的科室對象是class,class在C#里屬于引用類型。引用類型的比較和值類型的比較大不同相同!引用類型的比較是比較對象在內存堆里指向的地址,并非對象包含的屬性值 但是我們預期的比較就是單純的對值進行比較,所以這里需要通過實現(xiàn)IEqualityComparer<T>接口來對兩個引用類型集合對象進行比較。
創(chuàng)建IEqualityComparer<DepartmentDto>對象
public class DepartmentEqualityComparer : IEqualityComparer<DepartmentDto>
{
public bool Equals([AllowNull] DepartmentDto x, [AllowNull] DepartmentDto y)
{
// 這里可以寫比較的關鍵代碼~
return x.Id == y.Id;
}
public int GetHashCode([DisallowNull] DepartmentDto obj)
{
return obj.ToString().GetHashCode();
}
}
接下來,定義一個新的集合,再手動向這個集合追加元素。
List<DepartmentDto> lstDepartItemsCPs = InitDepartmentData();
lstDepartItemsCPs.Add(new DepartmentDto()
{
Address = "三馬路XX號",
Id = 6,
Name = "二級三號科室",
Remark = "",
TelPhone = "0731-6222222",
EmployeeNumber = 7
});
集合比較代碼:
// 這里如果DepartmentDto為引用類型(class)則需要使用比較器DepartmentEqualityComparer才能返回我們的預期值(根據(jù)ID值判斷是否相等) List<DepartmentDto> diffList = lstDepartItemsCPs.Except(lstDepartItems, new DepartmentEqualityComparer()).ToList(); // 獲取相等元素 List<DepartmentDto> diffList1 = lstDepartItemsCPs.Intersect(lstDepartItems.Select(p => p), new DepartmentEqualityComparer()).ToList(); // 需要添加IEqualityComparer,因為集合內的內容為引用類型!所以兩個集合的“值”是不同的,引用類型的值在這里還包含了指向的內存堆的引用地址 bool isEqual = lstDepartItems.SequenceEqual(InitDepartmentData(), new DepartmentEqualityComparer());
值類型的比較處理
可能你覺得需要去創(chuàng)建IEqualityComparer<DepartmentDto>對象過于麻煩,那么想下是否一定需要將科室對象的類型設定為class,是否有更合適的類型可以替換? 答案是有的,微軟推薦如果對具體模型不需要多次執(zhí)行裝箱、拆箱操作最好將模型設置為結構struct而非class。 現(xiàn)在我們回過頭將科室對象的類型更新為struct。
然后發(fā)現(xiàn)上面集合比較的代碼可以簡化成這樣:
// 這里如果DepartmentDto為值類型(struct)則不需要使用比較器DepartmentEqualityComparer即可返回我們的預期值(根據(jù)ID值判斷是否相等)
List<DepartmentDto> diffList3 = lstDepartItemsCPs.Except(lstDepartItems).ToList();
// 獲取相等元素
List<DepartmentDto> diffList4 = lstDepartItemsCPs.Intersect(lstDepartItems.Select(p => p)).ToList();
// 如果把DepartmentDto的類型改為值類型則可以不需要IEqualityComparer進行判斷的結果也會為true
isEqual = lstDepartItems.SequenceEqual(InitDepartmentData());
OfType和Cast的不同之處
OfType允許對集合的項進行隱性轉換(非強轉Convert)且在轉換前會進行判斷,當類型不允許轉換則continue到下一個集合項。而Cast則是子項不進行判斷,直接隱性轉換,所以理論上效率更高,當然相對的出錯率也更高~
public void ConvertListTest()
{
try
{
object[] ss = { 1, "2", 3, "四", "五", "7" };
// 1、3 OfType的本質是循環(huán)集合,對每個集合項進行類型驗證(不是強轉,所以此處的結果是1、3 而不是1、2、3、7)
var lst = ss.ToList().OfType<int>().ToList();
// 3
int max = ss.OfType<int>().Max();
// 這句代碼會提示“System.InvalidCastException:“Unable to cast object of type 'System.String' to type 'System.Int32'.”異常,原因:Cast的執(zhí)行效率會略高與OfType,因為在對集合項進行類型轉換前不會對其進行類型校驗,當你確保集合的類型是安全的則可以用Cast,但是能用到Cast和OfType的時候基本上都是用OfType了..
int maxCast = ss.Cast<int>().Max();
}
catch (Exception ex)
{
Console.WriteLine(ex);
}
}
上述代碼已上傳到github,地址: https://github.com/QQ897878763/LinqStudySample.git
以上就是c# Linq常用的小技巧的詳細內容,更多關于c# Linq小技巧的資料請關注腳本之家其它相關文章!
相關文章
C#實現(xiàn)循環(huán)發(fā)送電腦屏幕截圖
這篇文章主要為大家詳細介紹了C#實現(xiàn)循環(huán)發(fā)送電腦屏幕截圖,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-07-07

