C#+EF+SqlServer性能优化笔记

文章目录


前言

数据库和服务器降配之后,系统开始了各种超时,查询超时,保存超时,本身系统逻辑也很复杂,开发了五六年了,各种shi代码,慢sql,以下是一些经验的记录,便于以后忘了可以看看,也希望能帮到你!


一、C#EF 代码优化

1.接口代码改异步

代码(示例):

csharp 复制代码
[ResponseType(typeof(object)), Route("api/OpenApi/AddDataApi"), HttpPost]
public async Task<IHttpActionResult> AddDataApi(RuqestModel model)
{
	if(model == null)
	{
		return BadRequest();
	}
	var res = await AddDataAsync(model);//增加await,调用异步方法
	
	return OK(res);
}

private async Task<int> AddDataAsync(DbContext db, Tables model, long currentId)
{
    db.Tables.Add(model);
	await db.SaveChangesAsync();//增加await,调用异步保存方法
	return 1;
}

2.查询异步,只查询需要的数据

代码(示例):

csharp 复制代码
//1、异步方法可以提示效率
//2、通过select方法只查询Value字段,在大数据量的情况下,减少了传输的数据,可以提示效率
var Config = await db.MdmConfigs.Where(x => x.IsDeleted == false && x.Code == "Code1").Select(x => new { x.Value }).FirstOrDefaultAsync();

3.查询数据判断时

代码(示例):

csharp 复制代码
//1、判断数据是否存在时
var Config = await db.MdmConfigs.Where(x => x.IsDeleted == false && x.Code == "Code1").FirstOrDefaultAsync();

if(Config == null)
{
	return BadRequest();
}
//可以优化为:
var IsExist= await db.MdmConfigs.AnyAsync(x => x.IsDeleted == false && x.Code == "Code1");

if(!IsExist)
{
	return BadRequest();
}

4.直接使用sql查询

代码(示例):

csharp 复制代码
//复杂查询可以直接使用sql查询,下面的sql是简写
var config = db.Database.SqlQuery<MdmConfig>("SELECT * FROM MdmConfig WHERE IsDeleted = 0 and Code = 'Code1'").FirstOrDefault();

二、数据库优化

1.减少关联表,一些基础数据,字典表可以考虑放到redis中,在代码中映射

2.增加索引,删除无效索引,查看索引碎片率,重建索引

代码(示例):

sql 复制代码
--查看表索引详情
EXEC sp_helpindex 'YourSchema.YourTableName';

--查看数据库缺失索引
USE DbName
GO
SELECT 
    mid.statement AS TableName,
    mid.equality_columns AS EqualityColumns,
    mid.inequality_columns AS InequalityColumns,
    mid.included_columns AS IncludedColumns,
    migs.user_seeks AS UserSeeks,
    migs.user_scans AS UserScans,
    migs.avg_user_impact AS AvgUserImpact,
    migs.avg_total_user_cost AS AvgTotalUserCost,
    'CREATE INDEX IX_' + OBJECT_NAME(mid.OBJECT_ID) + '_' + CAST(mid.index_handle AS VARCHAR(10)) + 
    ' ON ' + mid.statement + 
    ' (' + ISNULL(mid.equality_columns, '') + 
    CASE WHEN mid.equality_columns IS NOT NULL AND mid.inequality_columns IS NOT NULL THEN ',' ELSE '' END + 
    ISNULL(mid.inequality_columns, '') + ')' + 
    ISNULL(' INCLUDE (' + mid.included_columns + ')', '') AS CreateIndexStatement
FROM 
    sys.dm_db_missing_index_details AS mid
JOIN 
    sys.dm_db_missing_index_groups AS mig ON mid.index_handle = mig.index_handle
JOIN 
    sys.dm_db_missing_index_group_stats AS migs ON mig.index_group_handle = migs.group_handle
WHERE 
  mid. statement  LIKE '%DbName%' -- 确保只针对当前数据库运行
ORDER BY 
    migs.avg_user_impact DESC;
    

--查看数据库无效索引
SELECT
	ind.index_id,
	obj.name AS TableName,
	ind.name AS IndexName,
	ind.type_desc,
	indUsage.user_seeks,
	indUsage.user_scans,
	indUsage.user_lookups,
	indUsage.user_updates,
	indUsage.last_system_seek,
	indUsage.last_user_scan,
	'drop index [' + ind.name + '] ON [' + obj.name + ']' AS DropIndexCommand
FROM
	sys.indexes AS ind
INNER JOIN sys.objects AS obj ON ind.object_id = obj.object_id
LEFT JOIN sys.dm_db_index_usage_stats indUsage ON ind.object_id = indUsage.object_id
AND ind.index_id = indUsage.index_id
WHERE
	ind.type_desc <> 'HEAP'
AND obj.type <> 'S'
AND OBJECTPROPERTY(
	obj.object_id,
	'isusertable'
) = 1
AND (
	ISNULL(indUsage.user_seeks, 0) = 0
	AND ISNULL(indUsage.user_scans, 0) = 0
	AND ISNULL(indUsage.user_lookups, 0) = 0
)
ORDER BY
	obj.name,
	ind.name


------------查看索引碎片率并大于30%
SELECT OBJECT_NAME(ind.OBJECT_ID) AS TableName,
    ind.name AS IndexName,
    indexstats.index_type_desc AS IndexType,
    indexstats.avg_fragmentation_in_percent 
FROM sys.dm_db_index_physical_stats(DB_ID(), NULL, NULL, NULL, NULL) indexstats 
    INNER JOIN sys.indexes ind ON ind.object_id = indexstats.object_id AND ind.index_id = indexstats.index_id 
WHERE indexstats.avg_fragmentation_in_percent >30 ------------碎片率
ORDER BY indexstats.avg_fragmentation_in_percent DESC


--重新组织索引(把第1步里的表名TableName复制进来,并调整阈值)
ALTER INDEX ALL ON [TableName] 
REBUILD WITH (FILLFACTOR = 90, SORT_IN_TEMPDB = ON,STATISTICS_NORECOMPUTE = ON); 

--查看索引碎片率并大于30%并进行整理
DECLARE @SQL NVARCHAR(MAX) = N'';

SELECT 
    @SQL += 'ALTER INDEX [' + ind.name + '] ON [' + OBJECT_NAME(ind.OBJECT_ID) + '] REBUILD WITH (ONLINE=ON, FILLFACTOR=90);' + CHAR(13)
FROM 
    sys.dm_db_index_physical_stats(DB_ID(), NULL, NULL, NULL, NULL) AS indexstats
INNER JOIN 
    sys.indexes AS ind ON ind.object_id = indexstats.object_id AND ind.index_id = indexstats.index_id
WHERE 
    indexstats.avg_fragmentation_in_percent > 30
    AND ind.name IS NOT NULL;

PRINT @SQL;
-- 如果需要执行,取消以下注释
-- EXEC sp_executesql @SQL;

3.更深层可以参考这篇文章:

数据库优化 ------关于tempdb系统库影响IO和数据库性能的一些问题

相关推荐
体育分享_大眼2 小时前
从零搭建高并发体育直播网站:架构设计、核心技术与性能优化实战
java·性能优化·系统架构
电商api接口开发7 小时前
如何在C#中使用LINQ对数据库进行查询操作?
数据库·c#·linq
搬砖工程师Cola7 小时前
<C#>在 .NET 开发中,依赖注入, 注册一个接口的多个实现
开发语言·c#·.net
经年小栈8 小时前
性能优化-Spring参数配置、数据库连接参数配置、JVM调优
数据库·spring·性能优化
vil du11 小时前
C# string和其他引用类型的区别
c#
向宇it12 小时前
【blender小技巧】Blender导出带贴图的FBX模型,并在unity中提取材质模型使用
开发语言·unity·c#·游戏引擎·blender·材质·贴图
看海的四叔13 小时前
【SQL】MySql常见的性能优化方式
hive·sql·mysql·性能优化·数据分析·索引优化·sql语法
大模型铲屎官15 小时前
# Unity动画控制核心:Animator状态机与C#脚本实战指南 (Day 29)
c语言·unity·c#·游戏引擎·游戏开发·动画控制·animator状态机
安全菜鸟15 小时前
DeepSeek 接入 Word 完整教程
开发语言·c#·word
vil du15 小时前
c# AI编程助手 — Fitten Code
开发语言·c#·ai编程