C# LINQ语法学习

LINQ(Language Integrated Query)是一种内置于 C# 和 .NET 中的数据查询工具。它允许开发者以类似于 SQL 的语法来查询内存对象、数据库、XML 文件等数据源。以下是一个详细的 LINQ 教程,涵盖了从基础到高级的内容。

一、LINQ 基础

LINQ 的基本操作可以用两种语法来表达:

  • 查询表达式语法(Query Syntax):类似于 SQL 查询语法。
  • 方法语法 (Method Syntax):使用扩展方法(如 WhereSelect 等)进行查询。

这两种语法功能相同,可以互换使用。

1. 示例:查询表达式语法
csharp 复制代码
int[] numbers = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };

// 查询所有偶数
var evenNumbers = from num in numbers
                  where num % 2 == 0
                  select num;

foreach (var number in evenNumbers)
{
    Console.WriteLine(number);
}
2. 示例:方法语法
csharp 复制代码
int[] numbers = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };

// 使用方法语法查询所有偶数
var evenNumbers = numbers.Where(num => num % 2 == 0);

foreach (var number in evenNumbers)
{
    Console.WriteLine(number);
}

二、LINQ 的主要操作符

LINQ 包含多种操作符,常见的有过滤、排序、投影、集合操作、分组、连接和聚合。

1. 过滤操作符
  • Where:用于根据条件筛选数据。
csharp 复制代码
int[] numbers = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };

// 筛选大于 5 的数字
var filteredNumbers = numbers.Where(n => n > 5);
2. 排序操作符
  • OrderByOrderByDescending:用于升序和降序排序。
csharp 复制代码
string[] names = { "Alice", "Bob", "Charlie", "David" };

// 按名字升序排序
var sortedNames = names.OrderBy(n => n);

// 按名字降序排序
var descendingNames = names.OrderByDescending(n => n);
  • ThenByThenByDescending:用于多重排序。
csharp 复制代码
var people = new List<Person>
{
    new Person { FirstName = "John", LastName = "Doe" },
    new Person { FirstName = "Jane", LastName = "Smith" },
    new Person { FirstName = "John", LastName = "Smith" }
};

// 先按姓氏排序,再按名字排序
var sortedPeople = people.OrderBy(p => p.LastName).ThenBy(p => p.FirstName);
3. 投影操作符
  • Select:用于选择集合中的元素或其某些部分。
csharp 复制代码
var squares = numbers.Select(n => n * n);
  • SelectMany:将每个元素的结果展开为一个集合。
csharp 复制代码
List<int[]> numberLists = new List<int[]> { new[] { 1, 2 }, new[] { 3, 4 }, new[] { 5, 6 } };

// 使用 SelectMany 展开成单个集合
var allNumbers = numberLists.SelectMany(n => n);
4. 集合操作符
  • Distinct:用于去除重复元素。
  • UnionIntersectExcept:分别用于集合的并集、交集、差集。
csharp 复制代码
int[] numbers1 = { 1, 2, 3 };
int[] numbers2 = { 3, 4, 5 };

// 获取并集
var union = numbers1.Union(numbers2); // 结果为 {1, 2, 3, 4, 5}

// 获取交集
var intersect = numbers1.Intersect(numbers2); // 结果为 {3}

// 获取差集
var except = numbers1.Except(numbers2); // 结果为 {1, 2}
5. 聚合操作符
  • CountSumAverageMinMax:用于对集合进行数学运算。
csharp 复制代码
int[] numbers = { 1, 2, 3, 4, 5 };

// 计算总数
int count = numbers.Count();

// 计算总和
int sum = numbers.Sum();

// 计算平均值
double average = numbers.Average();
6. 分组操作符
  • GroupBy:根据某个键将集合分组。
csharp 复制代码
string[] words = { "apple", "banana", "cherry", "date", "elderberry" };

// 根据单词的长度分组
var groupedWords = words.GroupBy(word => word.Length);

foreach (var group in groupedWords)
{
    Console.WriteLine($"Length: {group.Key}");
    foreach (var word in group)
    {
        Console.WriteLine(word);
    }
}
7. 连接操作符
  • Join:用于连接两个集合。
csharp 复制代码
var categories = new List<Category>
{
    new Category { Id = 1, Name = "Beverages" },
    new Category { Id = 2, Name = "Snacks" }
};

var products = new List<Product>
{
    new Product { Name = "Tea", CategoryId = 1 },
    new Product { Name = "Coffee", CategoryId = 1 },
    new Product { Name = "Chips", CategoryId = 2 }
};

// 根据 CategoryId 连接 categories 和 products
var query = from c in categories
            join p in products on c.Id equals p.CategoryId
            select new { c.Name, p.Name };

foreach (var item in query)
{
    Console.WriteLine($"{item.Name} - {item.Name}");
}

三、LINQ 的延迟执行和立即执行

LINQ 查询默认是延迟执行的,意味着在真正遍历查询结果时才会执行查询操作。

延迟执行
csharp 复制代码
var query = numbers.Where(n => n > 5);

// 此时查询并未执行
立即执行

可以使用 ToList()ToArray()Count() 等方法来立即执行查询。

csharp 复制代码
var resultList = numbers.Where(n => n > 5).ToList(); // 查询在此时执行

四、LINQ 的高级功能

1. 使用 let 关键字

let 可以用于定义临时变量,使查询更加简洁。

csharp 复制代码
var query = from word in words
            let length = word.Length
            where length > 3
            select new { word, length };
2. 使用匿名类型

可以使用匿名类型来创建临时的结果对象。

csharp 复制代码
var results = from word in words
              select new { Word = word, Length = word.Length };
3. 使用 Lambda 表达式和扩展方法

方法语法通常使用 Lambda 表达式,可以实现非常复杂的查询。

csharp 复制代码
var query = words.Where(w => w.Length > 3)
                 .OrderBy(w => w)
                 .Select(w => new { Word = w, Length = w.Length });
4. 混合使用查询表达式和方法语法

在同一个查询中,可以混合使用查询表达式和方法语法。

csharp 复制代码
var query = (from num in numbers
             where num > 5
             select num).Sum();

五、实战:LINQ 查询数据库

LINQ 也可以用于查询数据库,通常结合 Entity Framework 使用。以下是一个查询数据库的例子:

csharp 复制代码
using (var context = new MyDbContext())
{
    var customers = from c in context.Customers
                    where c.City == "New York"
                    select c;

    foreach (var customer in customers)
    {
        Console.WriteLine(customer.Name);
    }
}

总结

  • LINQ 提供了类似 SQL 的查询语法和方法语法,可以方便地对各种数据源进行操作。
  • 它支持过滤、排序、分组、连接、投影等操作,具有强大的数据处理能力。
  • 延迟执行是 LINQ 的一个特点,能提高查询效率,但也需要注意可能的副作用。
  • 在实际开发中,可以结合 Entity Framework 等技术来处理数据库查询。

LINQ 的学习曲线较平缓,但掌握其高级功能和应用场景会显著提升开发效率。

相关推荐
sanguine__2 分钟前
java学习-集合
学习
lxlyhwl2 分钟前
【STK学习】part2-星座-目标可见性与覆盖性分析
学习
nbsaas-boot3 分钟前
如何利用ChatGPT加速开发与学习:以BPMN编辑器为例
学习·chatgpt·编辑器
CV学术叫叫兽42 分钟前
一站式学习:害虫识别与分类图像分割
学习·分类·数据挖掘
神仙别闹43 分钟前
基于C#和Sql Server 2008实现的(WinForm)订单生成系统
开发语言·c#
我们的五年1 小时前
【Linux课程学习】:进程程序替换,execl,execv,execlp,execvp,execve,execle,execvpe函数
linux·c++·学习
一棵开花的树,枝芽无限靠近你1 小时前
【PPTist】添加PPT模版
前端·学习·编辑器·html
VertexGeek2 小时前
Rust学习(八):异常处理和宏编程:
学习·算法·rust
二进制_博客2 小时前
Flink学习连载文章4-flink中的各种转换操作
大数据·学习·flink
codebolt3 小时前
ADS学习记录
学习