为什么不推荐使用Linq?

linq · 浏览次数 : 9

小编点评

## 标题:关于Linq在性能敏感型和零内存分配场景中的使用建议 **引言** 在.NET生态中,Linq(Language Integrated Query)一直以其优雅的语法和强大的查询能力受到许多开发者的喜爱。然而,在某些特定的场景下,如性能敏感型应用或追求零内存分配的应用,我们是否应该慎用Linq呢?本文将通过BenchmarkDotNet进行性能测试,来探讨这个问题。 ## 文章正文 首先,我们来看一个简单的场景:将一个字符串拆分成整数数组,并对数组中的每个整数求和。为了验证这个场景,我们编写了以下两个基准测试: 1. 使用Linq的SumUsingLinq方法 2. 使用For循环的SumUsingForLoop方法 基准测试代码如下: ```csharp internal class Program { static void Main(string[] args) { BenchmarkRunner.Run(); Console.ReadKey(); } } [Benchmark] public void SumUsingLinq() { int sum = _row.Split(',').Select(int.Parse).Sum(); } [Benchmark(Baseline = true)] public void SumUsingForLoop() { var rowSpan = _row.AsSpan(); int sum = 0; for (int i = 0; i < rowSpan.Length; i++) { if (rowSpan[i] == ',') { sum += int.Parse(rowSpan.Slice(0, i)); rowSpan = rowSpan.Slice(i + 1); i = 0; } } } ``` 经过BenchmarkDotNet的运行,我们得到了以下测试结果: - 不使用Linq的SumUsingForLoop方法:Mean平均值 75.09 ns,Allocated 无 - 使用Linq的SumUsingLinq方法:Mean平均值 270.18 ns,Allocated 400 B 从结果中,我们可以看到,不使用Linq的SumUsingForLoop方法在性能上更优,且内存分配方面也是不使用Linq的占优。两者的性能差距约为3.5倍。 ## 结论 综上所述,对于追求零内存分配(zero-copy)目标或者速度敏感型应用,我们应该慎用Linq。因为在这些场景下,Linq扩展方法里的大量的判断和内部对象的创建及内存开销可能会成为性能瓶颈。 ## 后记 想坚持写技术文章、写博客、写公众号,真的很难。。希望这篇文章能给你带来一些帮助。如果有任何问题或建议,请随时留言交流。

正文

相信很多.NETer看了标题,都会忍不住好奇,点进来看看,并且顺便准备要喷作者!

这里,首先要申明一下,作者本人也非常喜欢Linq,也在各个项目中常用Linq。

我爱Linq,Linq优雅万岁!!!(PS:顺便吐槽一下,隔壁Java从8.0版本推出的Streams API,抄了个四不像,一点都不优雅,而且很难用。)​

正文

不罗嗦,就一句话:“在性能敏感型应用和追求零内存分配场景不推荐使用Linq!

让我们用Benchmark结果来说话!!!

这里用一个简单的场景来验证:

  1. 拆分一个String字符串为String[]数组。
  2. 转换数组中的每个String字符串为​Int32数值。
  3. 对这些数值求和​。

让我们通过BenchmarkDotNet运行性能测试,看看用Linq和不用Linq,这两者之间的性能差异​。

Benchmark代码:

internal class Program
{
    static void Main(string[] args)
    {
        BenchmarkRunner.Run<LinqTest>();
        Console.ReadKey();
    }
}
​
[MemoryDiagnoser, MemoryRandomization]
public class LinqTest
{
    private static readonly string _row = "1,2,3,4,5,6,7,8,9,10";
​
    [Benchmark]
    public void SumUsingLinq()
    {
        int sum = _row.Split(',').Select(int.Parse).Sum();
    }
​
    [Benchmark(Baseline = true)]
    public void SumUsingForLoop()
    {
        var rowSpan = _row.AsSpan();
        int sum = 0;
        for (int i = 0; i < rowSpan.Length; i++)
        {
            if (rowSpan[i] == ',')
            {
                sum += int.Parse(rowSpan.Slice(0, i));
                rowSpan = rowSpan.Slice(i + 1);
                i = 0;
            }
        }
    }
}

Benchmark结果:

从结果中,我们可以看到,不使用Linq的SumUsingForLoop方法,Mean平均值 75.09 ns,Allocated 无;
反观使用了Linq的SumUsingLinq方法,Mean平均值 270.18 ns,Allocated 400 B;
两者之间有约3.5倍的性能差距,而内存分配表现方面也是不使用Linq的占优。

总结:
对于追求 零内存分配(zero-copy)目标 或者 速度敏感型应用,这两种场景,都应该​慎用Linq。因为Linq扩展方法里有大量的判断,甚至还有很多内部对象的创建和内存开销。

PS: 想要坚持写技术文章、写博客、写公众号,真的很难。

与为什么不推荐使用Linq?相似的内容:

为什么不推荐使用Linq?

相信很多.NETer看了标题,都会忍不住好奇,点进来看看,并且顺便准备要喷作者! 这里,首先要申明一下,作者本人也非常喜欢Linq,也在各个项目中常用Linq。 我爱Linq,Linq优雅万岁!!!(PS:顺便吐槽一下,隔壁Java从8.0版本推出的Streams API,抄了个四不像,一点都不优雅

[转帖]为什么不推荐使用/etc/fstab

https://www.jianshu.com/p/af49a5d0553f 对于工作中使用服务器的公司来讲,每到节假日来临时,总免不了对服务器进行下电。而收假回来的早上,则会有一个早上的时间会花费在服务区的开机和修复工作上。是的,修复无法开启或开机失败的服务器。 几乎每次节假日售价回来,恢复服务器

如何创建一个线程池,为什么不推荐使用Executors去创建呢?

我们在学线程的时候了解了几种创建线程的方式,比如继承Thread类,实现Runnable接口、Callable接口等,那对于线程池的使用,也需要去创建它,在这里我们提供2种构造线程池的方法: 方法一: 通过ThreadPoolExecutor构造函数来创建(首选) 这是JDK中最核心的线程池工具类,

为什么不推荐在Spring Boot中使用@Value加载配置

@Value注解相信很多Spring Boot的开发者都已经有接触了,通过使用该注解,我们可以快速的把配置信息加载到Spring的Bean中。 比如下面这样,就可以轻松的把配置文件中key为com.didispace.title配置信息加载到TestService中来使用 @Service publ

android中Room数据库的基本使用

简介: 还在使用原生的sqllite?有这么清爽且稳如狗的room为啥不用呢? Room是Google官方推荐使用的数据库,相比较某些优秀数据库框架来说,不用过于担心某天库会停止维护,且访问数据库非常流畅,并且提供了与常规的ORM框架一样,通过添加编译期注解来进行表和字段的配置,譬如@Databas

《最新出炉》系列初窥篇-Python+Playwright自动化测试-6-元素定位大法-下篇

1.简介 上一篇主要是讲解我们日常工作中在使用Playwright进行元素定位的一些比较常用的定位方法的理论基础知识以及在什么情况下推荐使用。今天这一篇讲解和分享一下,在日常中很少用到或者很少见的定位,但是遇到了我们也要会,俗话说:手里有粮心里不慌。 2.阴影定位-Shadow DOM 在做web自

关于.Net 6.0 在Linux ,Docker容器中,不安装任何依赖就生成图形验证码!!!!!!!!!!!

在.Net Framework时代,我们生成验证码大多都是用System.Drawing。 在.Net 6中使用也是没有问题的。 但是,System.Drawing却依赖于Windows GDI+。 为了实现跨平台,我陷入了沉思!! 微软推荐使用SkiaSharp 进行替代,所以就开始了,踩坑之旅

4款.NET开源的Redis客户端驱动库

前言 今天给大家推荐4款.NET开源免费的Redis客户端驱动库(以下排名不分先后)。 Redis是什么? Redis全称是REmote DIctionary Service,即远程字典服务。Redis 是一个使用C语言编写的、开源的(遵守 BSD 协议)、支持网络、可基于内存亦可持久化的日志型、K

[转帖]Linux性能测试之unixbench

https://www.modb.pro/db/487945 大家好,昨天为大家带来了一篇关于在Linux下性能测试的文章《性能测试之LTP》,今天继续为大家推荐系列工具之unixbench,本工具用于Linux中cpu系统的测试,详情请查看百度,这里不多赘述,本文主要用于演示如何使用此工具。本文使

[转帖]Linux性能测试之unixbench

https://www.modb.pro/db/487945 大家好,昨天为大家带来了一篇关于在Linux下性能测试的文章《性能测试之LTP》,今天继续为大家推荐系列工具之unixbench,本工具用于Linux中cpu系统的测试,详情请查看百度,这里不多赘述,本文主要用于演示如何使用此工具。本文使