1、简介

再说Windows的异步I/O操作前,先聊聊一些题外话,能帮助我们更好的理解异步I/O操作,常规的Web程序,当用户发起一次请求,当请求通过管道到达客户端的这个过程,会唤起一个线程池线程(后台线程),处理我们的业务代码,即所有的用户请求是通过异步的方式发起的,这个过程,.Net Framework会自动进行,即使我们没有显示的通过代码来实现这个过程.所以这个过程明显是存在性能瓶颈的,假设现在有一个4核服务器,意味这该服务器同时只能处理4个用户请求(超理想情况下,一般不可能),但是这个时候来了10000个用户请求(并发执行)的情况下,那么意味者大量线程会堆积起来,等待着前面的线程执行完毕,同时进行频繁的上下文切换,这个时候你会发现CPU会爆表.

上面只是一个例子,再说一个数据库的例子,现在需要向数据库插入20000条记录,分为三个版本去实现,第一个版本是单个线程同步插入,第二个版本多线程同步插入(Parallel),第三个版本多线程异步插入,来比较下性能和CPU利用零及使用情况.

(1)、单线程同步版本

这个场景是只有一个用户请求进来,进行20000次的数据库插入操作,这个版本不会产生线程堆积,因为所有的插入操作都只由主线程完成.

        private static readonly string ConnectionStrings;
static Program()
{
//配置数据库连接
ConnectionStrings = ConfigurationManager.ConnectionStrings["connStr"].ConnectionString;
} static void Main(string[] args)
{ var stop = Stopwatch.StartNew();
InstertSync();
stop.Stop();
Console.WriteLine($"同步执行20000次插入操作,耗时:{stop.ElapsedMilliseconds/1000}秒");
Console.ReadKey();
} private static void InstertSync()
{
var totalCount = ;
var failCount = ; //这里以同步方式执行数据库操作,注这里只有一个线程执行所有的数据库插入操作
for (int i = ; i <= ; i++)
{
var conn = new SqlConnection(ConnectionStrings);
conn.Open();
try
{ //模拟数据库耗时操作
var sql = "insert into [dbo].[User]([Amount]) values (@Amount)";
var command = new SqlCommand(sql, conn);
command.Parameters.Add(
new SqlParameter("@Amount", i)
);
//这里线程会等待这一段时间,等待数据库返回结果,并继续执行下面的代码
var result = command.ExecuteNonQuery();
if (result == )
{
totalCount+=1;
}
else
{
failCount+=1;
} Console.WriteLine($"成功插入{totalCount}条记录,插入失败{failCount}条记录");
}
catch (Exception ex)
{ throw ex;
}
finally
{
conn.Close();
conn.Dispose();
}
}
}

再看看数据库的批请求数数据

大概稳定在300次左右每秒

(2)、多线程同步

这个场景是大多数没有使用Async Await模型的Web应用程序(Parallel代表同时有多个用户请求进来),同时数据库也使用的是同步Api,这个时候以同步的方式发起数据库请求,每个线程会等待不确定的时间,等待数据库返回结果,同时另一个线程开启,也会等待数据库返回结果,这样用户请求一多,就会产生大量的线程堆积,造成大量的内存浪费,而且当数据库开始响应线程时,线程会被唤醒,全部开始执行,这时候CPU又会开始繁忙的执行.

        private static readonly string ConnectionStrings;
static Program()
{
//配置数据库连接
ConnectionStrings = ConfigurationManager.ConnectionStrings["connStr"].ConnectionString;
} static void Main(string[] args)
{
InsertAsync();
Console.ReadKey();
} private static void InsertAsync()
{
var stop = Stopwatch.StartNew();
var totalCount = ;
var failCount = ;
var res = Parallel.For(, , i =>
{
var conn = new SqlConnection(ConnectionStrings);
conn.Open();
try
{
//模拟数据库耗时操作
var sql = "insert into [dbo].[User]([Amount]) values (@Amount)";
var command = new SqlCommand(sql, conn);
command.Parameters.Add(
new SqlParameter("@Amount", i)
);
var result = command.ExecuteNonQuery();
if (result == )
{
Interlocked.Add(ref totalCount, );
}
else
{
Interlocked.Add(ref failCount, );
} Console.WriteLine($"成功插入{totalCount}条记录,插入失败{failCount}条记录");
}
catch (Exception ex)
{ throw ex;
}
finally
{
conn.Close();
conn.Dispose();
}
});
if (res.IsCompleted)
{
stop.Stop();
Console.WriteLine($"同步执行20000次插入操作,耗时:{stop.ElapsedMilliseconds / 1000}秒");
}
}

去除Interlocked稍稍快一些.明显可以发现在多线程环境下,使用同步的数据库操作api,效率显著下降.CPU的利用率也很低,同时跑了很多操作线程,但数据库使用同步Api,只能响应一个线程,其余的都需要排队.

再看看数据库批请求数

只能稳定在130次左右,说明多线程环境下,使用同步数据库操作,阻碍了请求的提交速度.个人理解.

(3)、多线程异步

这个场景用户使用基于Async Await模型的Web程序,且使用数据库的异步Api

        private static readonly string ConnectionStrings;
static Program()
{
//配置数据库连接
ConnectionStrings = ConfigurationManager.ConnectionStrings["connStr"].ConnectionString;
} static void Main(string[] args)
{
InsertAsync();
Console.ReadKey();
} private static void InsertAsync()
{
var stop = Stopwatch.StartNew();
var totalCount = ;
var failCount = ;
var res = Parallel.For(, ,async i =>
{
var conn = new SqlConnection(ConnectionStrings);
conn.Open();
try
{ //模拟数据库耗时操作
var sql = "insert into [dbo].[User]([Amount]) values (@Amount)";
var command = new SqlCommand(sql, conn);
command.Parameters.Add(
new SqlParameter("@Amount", i)
); var result =await command.ExecuteNonQueryAsync();
if (result == )
{
Interlocked.Add(ref totalCount, );
}
else
{
Interlocked.Add(ref failCount, );
} Console.WriteLine($"成功插入");
}
catch (Exception ex)
{ throw ex;
}
finally
{
conn.Close();
conn.Dispose();
}
});
if (res.IsCompleted)
{
stop.Stop();
Console.WriteLine($"同步执行20000次插入操作,耗时:{stop.ElapsedMilliseconds / 1000}秒");
}
}

可以发现这个模式插入效率非常之高.但是它的插入是无序的,因为Parallel执行线程的顺序是无序的.CPU的利用率也是极高的.

再看看数据库批请求数

直线飙升>1000次的请求提交,说明使用异步Api数据库每秒接收的请求数,远大于同步方式,也是使用异步Api如此之快的原因.

.Net 环境下比较各种数据库插入操作的性能的更多相关文章

  1. WIN7 环境下搭建 PHP7(64 位)操作步骤

    WIN7 环境下搭建 PHP7(64 位)操作步骤 一.安装与配置 Apache 1.下载 Apache下载地址:https://www.apachelounge.com/download/ 2.安装 ...

  2. Java CAS同步机制 原理详解(为什么并发环境下的COUNT自增操作不安全): Atomic原子类底层用的不是传统意义的锁机制,而是无锁化的CAS机制,通过CAS机制保证多线程修改一个数值的安全性。

    精彩理解:  https://www.jianshu.com/p/21be831e851e ;  https://blog.csdn.net/heyutao007/article/details/19 ...

  3. 【转】编写微信聊天机器人4《聊天精灵WeChatGenius》:实时获取到微信聊天消息,hook数据库插入操作。

    接上篇,使用Xposed来hook微信,找到微信进程:https://blog.csdn.net/weixin_42127613/article/details/81839537 既然已经找到了微信进 ...

  4. python 数据库插入操作

    数据库插入操作 以下实例使用执行 SQL INSERT 语句向表 EMPLOYEE 插入记录: #!/usr/bin/python # -*- coding: UTF-8 -*- import MyS ...

  5. 生产环境下,MySQL大事务操作导致的回滚解决方案

    如果mysql中有正在执行的大事务DML语句,此时不能直接将该进程kill,否则会引发回滚,非常消耗数据库资源和性能,生产环境下会导致重大生产事故. 如果事务操作的语句非常之多,并且没有办法等待那么久 ...

  6. linux下的mongodb数据库原生操作

    mongodb,是一种结构最像mysql的nosql mysql中的数据库,mongodb中也有,区别在于, myql中数据库下的是表,字段和数据的形式存在 mongodb数据库下的是叫集合(和pyt ...

  7. Linux环境下安装MySQL数据库

    Linux安装mysql服务分两种安装方法: (1).源码安装,优点是安装包比较小,只有十多M,缺点是安装依赖的库多,安装编译时间长,安装步骤复杂容易出错: (2).使用官方编译好的二进制文件安装,优 ...

  8. 监控Mysql主从环境下Slave延迟状态的操作记录

    在MySQL主从环境下,通常会根据Seconds_Behind_Master的值来判断slave的延迟状态,这么做在大部分情况下尚可接受,但其实是并不够准确的.对于Slave延迟状态的监控,应该考虑多 ...

  9. Linux环境下实现对文件读写操作

    ---- 今天分享一下在linux系统在实现对文件读写一些基本的操作,在这之前我们要掌握一些基本的技能在Linux环境下.比如查看命令和一个函数的具体用法,就是相当于查手册,在Linux下有一个man ...

随机推荐

  1. String StringBuilder 包装类

    1. String 概述 程序中直接写上双引号的字符串就在字符串常量池中,new的不在池当中 java6之前常量池在方法区,java7以后将字符串常量池放在堆中 因为字符串是对象,应该在堆中 相同的字 ...

  2. 创建Gitblit本地服务器(For windows )01

    1.先下载gitblit  貌似需要FQ,百度云链接https://pan.baidu.com/s/1WUtBswj5TkFFcd_hiFFCcw,提取码: xr9n .因为gitblit是基于jav ...

  3. http post 请求详解

    一步一步了解http post 请求 (大白话版). 1.创建一 个 CloseableHttpClient  对象 CloseableHttpClient client = HttpClients. ...

  4. [ERROR] Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.2:compile (default-compile) on project taotao-manager-pojo: Compilation failure

    运行maven项目时报错 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.2:compi ...

  5. SEO常用命令大全

    SEO常用单个命令如下: link: 指某个特定网站的外部链接,这是一个主要获取排名和权重的主要因素. site: 呵呵,这个是经常用到的了,它是指某个特定网站收录情况. 用法:site: www.  ...

  6. HDU 4609 3-idiots (组合数学 + FFT)

    题意:给定 n 条边,问随机选出 3 条边,能组成三角形的概率是多少. 析:答案很明显就是  能组成三角形的种数 / (C(n, 3)).现在的问题是怎么求能组成三角形的种数. 这个博客说的非常清楚了 ...

  7. bootstrap-datepicker简单使用

    粗略整理,可能存在其他的方式请大家多多指教 选择年份 html <div class="dropdown"> <label class="search- ...

  8. tab选项卡--jq

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  9. 负载均衡器之 Haproxy

    1. 编译安装haproxy 官网: http://www.haproxy.org 1.1 下载haproxy # wget http://www.haproxy.org/download/1.6/s ...

  10. index.php入口文件至根目录

      登录|注册       咖啡如同生活的专栏 从不羡慕别人的优秀,因为相信自己也可以优秀. 闲下来时看看书,书本里的故事,总有你学到的人生.       目录视图 摘要视图 订阅 异步赠书:9月重磅 ...