SourceGenerator 生成db to class代码优化结果记录
优化
上一次实验 代码写的较为随意,本次穷尽所学,优化了一把,
不过果然还是没 比过 Dapper aot, 虽然没使用 Interceptor, 但理论上其优化不该有这么大差距
知识差距不少呀,都看不懂 Dapper aot 利用了什么姿势领先, 有大神们能教教吗?
优化点
减少类型判断
提前 做类型判断,并在生成时利用 switch case 减少判断
之前
var needConvert = typeof(string) != reader.GetFieldType(i);
s.Add((d,r) => d.Name = DBExtensions.ReadToString(r,j,needConvert));
之后
switch (name)
{
case "age":
s.Add(type == typeof(int) ? 1 : 2);
break;
switch (ss[j])
{
case 1:
d.Age = EntitiesGenerator.ReadToInt32Nullable(reader,j);
break;
case 2:
d.Age = EntitiesGenerator.ReadToInt32NullableConvert(reader,j);
break;
避免生成委托
去除委托生成使用
之前
var s = new List<Action<BenchmarkTest.Dog, IDataReader>>(reader.FieldCount);
for (int i = 0; i < reader.FieldCount; i++)
{
var j = i;
switch (reader.GetName(j).ToLower())
{
case "age":
{
// int?
var needConvert = typeof(int) != reader.GetFieldType(i);
s.Add((d,r) => d.Age = DBExtensions.ReadToInt32Nullable(r,j,needConvert));
}
break;
case "name":
{
// string
var needConvert = typeof(string) != reader.GetFieldType(i);
s.Add((d,r) => d.Name = DBExtensions.ReadToString(r,j,needConvert));
}
break;
case "weight":
{
// float?
var needConvert = typeof(float) != reader.GetFieldType(i);
s.Add((d,r) => d.Weight = DBExtensions.ReadToFloatNullable(r,j,needConvert));
}
break;
default:
break;
}
}
while (reader.Read())
{
var d = new BenchmarkTest.Dog();
foreach (var item in s)
{
item?.Invoke(d,reader);
}
yield return d;
}
之后
var s = new List<int>(reader.FieldCount);
for (int i = 0; i < reader.FieldCount; i++)
{
var name = reader.GetName(i).ToLower();
var type = reader.GetFieldType(i);
switch (name)
{
case "age":
s.Add(type == typeof(int) ? 1 : 2);
break;
case "name":
s.Add(type == typeof(string) ? 3 : 4);
break;
case "weight":
s.Add(type == typeof(float) ? 5 : 6);
break;
default:
break;
}
}
ss = s.ToArray();
var d = new BenchmarkTest.Dog();
for (int j = 0; j < ss.Length; j++)
{
switch (ss[j])
{
case 1:
d.Age = EntitiesGenerator.ReadToInt32Nullable(reader,j);
break;
case 2:
d.Age = EntitiesGenerator.ReadToInt32NullableConvert(reader,j);
break;
case 3:
d.Name = EntitiesGenerator.ReadToString(reader,j);
break;
case 4:
d.Name = EntitiesGenerator.ReadToStringConvert(reader,j);
break;
case 5:
d.Weight = EntitiesGenerator.ReadToFloatNullable(reader,j);
break;
case 6:
d.Weight = EntitiesGenerator.ReadToFloatNullableConvert(reader,j);
break;
default:
break;
}
}
添加 reader 字段判断缓存
添加缓存,减少重复生成
var h = reader.GetColumnHash();
if (!tokenCache.TryGetValue(h, out var ss))
{
var s = new List<int>(reader.FieldCount);
for (int i = 0; i < reader.FieldCount; i++)
结果
BenchmarkDotNet v0.13.12, Windows 10 (10.0.19045.4651/22H2/2022Update)
Intel Core i7-10700 CPU 2.90GHz, 1 CPU, 16 logical and 8 physical cores
.NET SDK 9.0.100-preview.5.24307.3
[Host] : .NET 8.0.6 (8.0.624.26715), X64 RyuJIT AVX2
DefaultJob : .NET 8.0.6 (8.0.624.26715), X64 RyuJIT AVX2
Method | Categories | Mean | Error | StdDev | Ratio | RatioSD | Gen0 | Gen1 | Gen2 | Allocated | Alloc Ratio |
---|---|---|---|---|---|---|---|---|---|---|---|
SourceGeneratorMappingFirst | 1 | 434.7 ns | 8.67 ns | 7.69 ns | 0.84 | 0.02 | 0.0401 | 0.0396 | - | 336 B | 1.20 |
SetClassFirst | 1 | 516.8 ns | 9.86 ns | 10.55 ns | 1.00 | 0.00 | 0.0334 | 0.0324 | 0.0019 | 280 B | 1.00 |
DapperMappingFirst AOT | 1 | 1,333.4 ns | 2.49 ns | 2.33 ns | 2.58 | 0.06 | 0.0324 | - | - | 280 B | 1.00 |
DapperMappingFirst | 1 | 1,421.4 ns | 3.08 ns | 2.88 ns | 2.84 | 0.12 | 0.0496 | - | - | 416 B | 1.49 |
SetClass | 1000 | 8,139.8 ns | 130.22 ns | 115.43 ns | 1.00 | 0.00 | 6.7902 | 1.6937 | - | 56840 B | 1.00 |
DapperMapping AOT | 1000 | 16,373.8 ns | 275.34 ns | 244.08 ns | 2.01 | 0.05 | 6.7749 | 0.9460 | - | 56840 B | 1.00 |
SourceGeneratorMapping | 1000 | 20,911.5 ns | 77.69 ns | 60.65 ns | 2.57 | 0.04 | 6.7749 | 1.6785 | - | 56896 B | 1.00 |
DapperMapping | 1000 | 48,707.3 ns | 430.05 ns | 381.23 ns | 5.67 | 0.29 | 12.5122 | 2.0752 | - | 105120 B | 1.85 |
SourceGenerator 生成db to class代码优化结果记录的更多相关文章
- java生成db文件
工作中遇到一个需求,就是需要把mysql里的一些表数据生成db文件,给客户端使用,客户端使用sqlite数据库: 首先我们需要在项目中添加Sqlite JDBC 依赖 <dependency&g ...
- Linux中生成Core Dump系统异常信息记录文件的教程
Linux中生成Core Dump系统异常信息记录文件的教程 http://www.jb51.net/LINUXjishu/473351.html
- Visual Studio 2017 - Windows应用程序打包成exe文件(2)- Advanced Installer 关于Newtonsoft.Json,LINQ to JSON的一个小demo mysql循环插入数据、生成随机数及CONCAT函数 .NET记录-获取外网IP以及判断该IP是属于网通还是电信 Guid的生成和数据修整(去除空格和小写字符)
Visual Studio 2017 - Windows应用程序打包成exe文件(2)- Advanced Installer Advanced Installer :Free for 30 da ...
- IntelliJ IDEA 类和方法注释的生成以及Javadoc的简单使用记录
idea,设置类注释和,方法注释的常见的设置方法(不同的版本设置方法有所偏差,简单记录一些目前自己在使用的方法,) 方法注释:在keyMap中搜索Fix doc comment ,后点击右键设置一个快 ...
- Django创建表时报错django.db.utils.InternalError: (1366问题解决记录
问题出现 执行Python manage.py makemigrations生成创建表的py文件 执行python manage.py migrate创建数据表 界面出现报错 问题原因 网上搜索原因, ...
- CORE EF生成ORACLE数据库模型报错问题记录
需求:最近在新开发一套在LINUX运行的API接口,需要用到net core api框架以及oracle数据库,首先需要解决的就是连接数据库问题,由于是DBFirst 加上之前很多老表不规范,导致了c ...
- python DB.fetchall()--获取数据库所有记录列表
查询到的数据格式为列表: 多个元素的列表:
- 8、android代码优化技术记录
1.length.length().size的优化 举例: int array_one[] = {1,2,3,4,5,6,7,8,9,10}; int array_two[] = {1,2,3,4,5 ...
- Inverse是hibernate双向关系中的基本概念。inverse的真正作用就是指定由哪一方来维护之间的关联关系。当一方中指定了“inverse=false”(默认),那么那一方就有责任负责之间的关联关系,说白了就是hibernate如何生成Sql来维护关联的记录
<set name ='students' table="students_table" inverse='false'(默认不用写) > <key column ...
- mysql快速生成truncate脚本清空数据库表记录
语句格式: select CONCAT('truncate TABLE ',table_schema,'.',TABLE_NAME, ';') from INFORMATION_SCHEMA.TABL ...
随机推荐
- Maven工程报错org.codehaus.plexus.component.repository.exception.ComponentLookupException
本人电脑环境:jdk1.8,idea2021,maven 3.6.3 问题:今天调整maven项目的三个配置参数后,idea报了下面的错误 org.codehaus.plexus.component. ...
- Excel相关技巧
(1)如何实现EXCEL某个单元格满足条件整行变色? 链接:https://jingyan.baidu.com/article/75ab0bcba47c19d6864db2cf.html (2)获取表 ...
- webpack配置图片处理
# 安装 npm i -D url-loader html-loader file-loader # loader配置 module: { rules: [ // 图片处理 { test: /\.(p ...
- INFINI Labs 产品更新 | Easysearch 新增快照搜索功能,Console 支持 OpenSearch 存储
INFINI Labs 产品又更新啦~,包括 Easysearch v1.7.0.Console v1.13.0.本次各产品更新了 Easysearch 快照搜索功能:Console 支持 OpenS ...
- C# WINFORM 获取上级目录
MessageBox.Show(Application.StartupPath); DirectoryInfo di = new DirectoryInfo(string.Format(@" ...
- Vue学习:2.V标签综合2
接上一篇... V标签综合使用:书架案例 功能: 实现列表的渲染和删除 思路: 使用 v-for 渲染数据列表,并在每个列表项内放置一个绑定了 del方法的"删除"按钮,点击按钮时 ...
- idea设置jdk和设置文件编码格式utf-8
1.idea设置jdk 2.idea设置文件编码格式utf-8 create utf-8 files with NO BOM 不要更改,否则编译会出错误.
- Timing!!!
End or Beginning "毕业",一个令人无限憧憬的具象化名词.适逢高考结束,又有一批人将奔赴更远的地方,离开他们生活了十八年的城市,在这之中亦然有着曾经的我们.但大家把 ...
- 一次phoniex表查询报出 org.apache.hadoop.hbase.NotServingRegionException
org.apache.hadoop.hbase.NotServingRegionException: SYSTEM.STATS,,1607503004410.334266e1a9b7d9859dbfb ...
- pytest-allure 命令生成的报告,test body 没有具体的参数和日志
run.py: pytest.main([命令参数执行]),pytest命令执行完毕后,使用os.system()执行allure的命令 原因: 使用了命令:os.system('allure gen ...