数据库教程:针对Sqlserver大数据量插入速度慢或丢失数据的解决方法

我的设备上每秒将2000条数据插入数据库,2个设备总共4000条,当在程序里面直接用insert语句插入时,两个设备同时插入大概总共能插入约2800条左右,数据丢失约120

我的设备上每秒将2000条数据插入数据库,2个设备总共4000条,当在程序里面直接用insert语句插入时,两个设备同时插入大概总共能插入约2800条左右,数据丢失约1200条左右,测试了很多方法,整理出了两种效果比较明显的解决办法:

方法一:使用sql server函数:

1.将数据组合成字串,使用函数将数据插入内存表,后将内存表数据复制到要插入的表。

2.组合成的字符换格式:’111|222|333|456,7894,7458|0|1|2014-01-01 12:15:16;1111|2222|3333|456,7894,7458|0|1|2014-01-01 12:15:16′,每行数据中间用“;”隔开,每个字段之间用“|”隔开。

3.编写函数:

  create function [dbo].[fun_funcname](@str varchar(max),@splitchar char(1),@splitchar2 char(1))   --定义返回表    returns @t table(maxvalue float,phase int,slopevalue float,data varchar(600),alarm int,almlev int,gpstime datetime,updatetime datetime) as     /*       author:hejun li     create date:2014-06-09     */     begin     declare @substr varchar(max),@substr2 varchar(max)  --申明单个接收值   declare @maxvalue float,@phase int,@slopevalue float,@data varchar(8000),@alarm int,@almlev int,@gpstime datetime   set @substr=@str     declare @i int,@j int,@ii int,@jj int,@ijj1 int,@ijj2 int,@m int,@mm int   set @j=len(replace(@str,@splitchar,replicate(@splitchar,2)))-len(@str)--获取分割符个数     if @j=0       begin        --insert into @t values (@substr,1) --没有分割符则插入整个字串       set @substr2=@substr;     set @ii=0     set @jj=len(replace(@substr2,@splitchar2,replicate(@splitchar2,2)))-len(@substr2)--获取分割符个数       while @ii<=@jj          begin            if(@ii<@jj)              begin                set @mm=charindex(@splitchar2,@substr2)-1 --获取分割符的前一位置                if(@ii=0)                  set @maxvalue=cast(left(@substr2,@mm) as float)                else if(@ii=1)                  set @phase=cast(left(@substr2,@mm) as int)                else if(@ii=2)                  set @slopevalue=cast(left(@substr2,@mm) as float)                else if(@ii=3)                  set @data=cast(left(@substr2,@mm) as varchar)                else if(@ii=4)                  set @alarm=cast(left(@substr2,@mm) as int)                else if(@ii=5)                  set @almlev=cast(left(@substr2,@mm) as int)                else if(@ii=6)                  insert into @t values(@maxvalue,@phase,@slopevalue,''+@data+'',@alarm,@almlev,cast(@substr2 as datetime),getdate())                set @substr2=right(@substr2,len(@substr2)-(@mm+1)) --去除已获取的分割串,得到还需要继续分割的字符串              end            else              begin                --当循环到最后一个值时将数据插入表                insert into @t values(@maxvalue,@phase,@slopevalue,''+@data+'',@alarm,@almlev,cast(@substr2 as datetime),getdate())              end          --end          set @ii=@ii+1        end    end     else     begin      set @i=0      while @i<=@j      begin       if(@i<@j)       begin       set @m=charindex(@splitchar,@substr)-1 --获取分割符的前一位置    --insert into @t values(left(@substr,@m),@i+1)     -----二次循环开始    --1.线获取要二次截取的字串    set @substr2=(left(@substr,@m));    --2.初始化二次截取的起始位置    set @ii=0    --3.获取分隔符个数    set @jj=len(replace(@substr2,@splitchar2,replicate(@splitchar2,2)))-len(@substr2)--获取分割符个数    while @ii<=@jj      begin        if(@ii<@jj)          begin            set @mm=charindex(@splitchar2,@substr2)-1 --获取分割符的前一位置            if(@ii=0)              set @maxvalue=cast(left(@substr2,@mm) as float)            else if(@ii=1)              set @phase=cast(left(@substr2,@mm) as int)            else if(@ii=2)              set @slopevalue=cast(left(@substr2,@mm) as float)            else if(@ii=3)              set @data=cast(left(@substr2,@mm) as varchar)            else if(@ii=4)              set @alarm=cast(left(@substr2,@mm) as int)            else if(@ii=5)              set @almlev=cast(left(@substr2,@mm) as int)            else if(@ii=6)              insert into @t values(@maxvalue,@phase,@slopevalue,''+@data+'',@alarm,@almlev,cast(@substr2 as datetime),getdate())            set @substr2=right(@substr2,len(@substr2)-(@mm+1)) --去除已获取的分割串,得到还需要继续分割的字符串          end        else          begin            --当循环到最后一个值时将数据插入表            insert into @t values(@maxvalue,@phase,@slopevalue,''+@data+'',@alarm,@almlev,cast(@substr2 as datetime),getdate())          end      --end      set @ii=@ii+1    end    -----二次循环结束    set @substr=right(@substr,len(@substr)-(@m+1)) --去除已获取的分割串,得到还需要继续分割的字符串       end      else       begin    --insert into @t values(@substr,@i+1)--对最后一个被分割的串进行单独处理     -----二次循环开始    --1.线获取要二次截取的字串    set @substr2=@substr;    --2.初始化二次截取的起始位置    set @ii=0    --3.获取分隔符个数    set @jj=len(replace(@substr2,@splitchar2,replicate(@splitchar2,2)))-len(@substr2)--获取分割符个数    while @ii<=@jj      begin        if(@ii<@jj)          begin            set @mm=charindex(@splitchar2,@substr2)-1 --获取分割符的前一位置            if(@ii=0)              set @maxvalue=cast(left(@substr2,@mm) as float)            else if(@ii=1)              set @phase=cast(left(@substr2,@mm) as int)            else if(@ii=2)              set @slopevalue=cast(left(@substr2,@mm) as float)            else if(@ii=3)              set @data=cast(left(@substr2,@mm) as varchar)            else if(@ii=4)              set @alarm=cast(left(@substr2,@mm) as int)            else if(@ii=5)              set @almlev=cast(left(@substr2,@mm) as int)            else if(@ii=6)              insert into @t values(@maxvalue,@phase,@slopevalue,''+@data+'',@alarm,@almlev,cast(@substr2 as datetime),getdate())            set @substr2=right(@substr2,len(@substr2)-(@mm+1)) --去除已获取的分割串,得到还需要继续分割的字符串          end        else          begin            --当循环到最后一个值时将数据插入表            insert into @t values(@maxvalue,@phase,@slopevalue,''+@data+'',@alarm,@almlev,cast(@substr2 as datetime),getdate())          end      set @ii=@ii+1    end    -----二次循环结束    end      set @i=@i+1       end     end     return     end     

4.调用函数语句:

  insert into [mytable] select * from [dbo].[fun_funcname]('111|222|333|456,7894,7458|0|1|2014-01-01 12:15:16;1111|2222|3333|456,7894,7458|0|1|2014-01-01 12:15:16',';','|');  

5.结果展示:

  select * from [mytable] ;  

方法二:使用bulk insert

大数据量插入第一种操作,使用bulk将文件数据插入数据库

sql代码

创建数据库

  create database [db_mgr]   go     

创建测试表

  use db_mgr   create table dbo.t_student(     f_id [int] identity(1,1) not null,     f_code varchar(10) ,     f_name varchar(100) ,     f_memo nvarchar(500) ,     f_memo2 ntext ,     primary key (f_id)   )   go    

填充测试数据

  insert into t_student(f_code, f_name, f_memo, f_memo2) select  'code001', 'name001', 'memo001', '备注' union all select  'code002', 'name002', 'memo002', '备注' union all select  'code003', 'name003', 'memo003', '备注' union all select  'code004', 'name004', 'memo004', '备注' union all select  'code005', 'name005', 'memo005', '备注' union all select  'code006', 'name006', 'memo006', '备注'  

开启xp_cmdshell存储过程(开启后有安全隐患)

  exec sp_configure 'show advanced options', 1;   reconfigure;exec sp_configure 'xp_cmdshell', 1;   exec sp_configure 'show advanced options', 0;   reconfigure;    

使用bcp导出格式文件:

  exec master..xp_cmdshell 'bcp db_mgr.dbo.t_student format nul -f c:/student_fmt.xml -x -c -t'    

使用bcp导出数据文件:

  exec master..xp_cmdshell 'bcp db_mgr.dbo.t_student out c:/student.data -f c:/student_fmt.xml -t'    

将表中数据清空

  truncate table db_mgr.dbo.t_student    

使用bulk insert语句批量导入数据文件:

  bulk insert db_mgr.dbo.t_student   from 'c:/student.data'  with  (     formatfile = 'c:/student_fmt.xml'  )  

使用openrowset(bulk)的例子:

t_student表必须已存在

  insert into db_mgr.dbo.t_student(f_code, f_name) select f_code, f_name   from openrowset(bulk n'c:/student.data', formatfile=n'c:/student_fmt.xml') as new_table_name  

使用openrowset(bulk)的例子:

tt表可以不存在

需要了解更多数据库技术:针对Sqlserver大数据量插入速度慢或丢失数据的解决方法,都可以关注数据库技术分享栏目—计算机技术网(www.ctvol.com)!

  select f_code, f_name into db_mgr.dbo.tt   from openrowset(bulk n'c:/student.data', formatfile=n'c:/student_fmt.xml') as new_table_name    

本文来自网络收集,不代表计算机技术网立场,如涉及侵权请联系管理员删除。

ctvol管理联系方式QQ:251552304

本文章地址:https://www.ctvol.com/dtteaching/616102.html

(0)
上一篇 2021年5月20日
下一篇 2021年5月20日

精彩推荐