发布网友 发布时间:2022-04-26 22:49
共6个回答
懂视网 时间:2022-04-08 06:00
但是不仅如此,如果客户提供给你的excel本身存在着重复数据,或是excel中的某些数据已经在数据库存在,那这时,在向数据库插入数据前你还得判重,如果不存在才进行导入
通常,我们第一步就会通过上传的方式把excel中的数据读到内存,然后通过循环的方式得出一条一条数据,接着对于每条数据用关键字段去往数据库中进行一次查重,若存在则不做事情,若
不存在则向数据库中插入一条数据。这样一来,我们每一条数据都会与数据库打两次交道,众所周知,每连接一次数据库那是需要时间的,次数一多相当影响性能。若是成千上万条数据的话,可
想而知,这个导入过程会有多慢,尤其是Web应用程序,很有可能在我的请求还没执行完,突然程序就被终止了。
当然,还有个办法,就是拼SQL,每循环一条数据,首先判断,若不重复,我写一条SQL语句保存在某个变量中,直到循环到最后一条,可能会拼出多条Insert语句,最后送到数据库一次执
行,但是大家有没有想过,一旦数据量过大,几万,或几十万条数据拼成的字符串可想而知会有多长,送到数据库就会有被截断的可能。更何况还是逃离不了每次都要查询重复的惨况。
一般来说,SQL语句离数据库端越近,执行效率越高,有没有可能,我把所要插入的数据集合一次性送给数据库,让判断重复,插入,返回重复数据的工作统统由数据库来执行呢?这样我只
要与数据库打一次交道,等待数据库给我结果就行。那么接下来就是我要讲到的利用表值参数来完成这一功能。
由于为了讲述,例子中的表都比较简单
1.首先我们来创建一张表(建库的过程在这里我就不说明了,为了大家看得清晰,尽量去除了无关的脚本行)
1 CREATE TABLE [dbo].[BulkTestTable]( 2 [Id] [int] NOT NULL, 3 [UserName] [nvarchar](32) NULL, 4 [Pwd] [varchar](16) NULL, 5 PRIMARY KEY CLUSTERED 6 ( 7 [Id] ASC 8 )WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY] 9 ) ON [PRIMARY] 10 11 GO
2.创建表值类型
1 USE [BulkTestDB] 2 GO 3 4 /****** Object: UserDefinedTableType [dbo].[BulkTestTableType] Script Date: 07/08/2015 16:04:38 ******/ 5 CREATE TYPE [dbo].[BulkTestTableType] AS TABLE( 6 [Id] [int] NULL, 7 [UserName] [nvarchar](32) NULL, 8 [Pwd] [varchar](16) NULL 9 ) 10 GO
3.写批量插入的存储过程
1 CREATE procedure [dbo].[usp_BulkTestTable_Import]
2 (@paratable as BulkTestTableType readonly) --此处的BulkTestTableType就是上面所定义的表类型,实际不用对它进行操作,只需要在程序中传入一个表给它,然后从它里面进行读取
3 AS
4 Insert INTO dbo.BulkTestTable(Id,UserName,Pwd)
5 select * from @paratable A
6 WHERE Not EXISTS(select B.Id from BulkTestTable B WHERE B.Id=A.Id)
7 SELECT * from @paratable A WHERE EXISTS (select B.Id from BulkTestTable B WHERE B.Id=A.Id) --查询出重复的记录
8 GO
在这里,我的Id不是自动增长的,仅仅是主键而已,所以这里通过Id来判断记录是否唯一或重复
接下来在C#代码里面写一个导入方法,调用这个存储过程。
这里的参数dt就是我们导入excel时生成的DataTable,这个DataTable的表结构也就是列要与我们定义的表值“BulkTestTableType”结构相同,返回值就是我们所要的重复记录
1 public static DataSet BatchInsert(DataTable dt) 2 { 3 SqlParameter parameter = new SqlParameter("@paratable",dt); 4 parameter.SqlDbType = SqlDbType.Structured; 5 parameter.TypeName = "BulkTestTableType"; //这里的类型名称应与我们定义的表值名称相同
6 SqlParameter[] sqlParameters = new SqlParameter[] { parameter };
7return SqlHelper.RunProcedure(CommandType.StoredProcedure, "usp_BulkTestTable_Import", sqlParameters);
8 }
热心网友 时间:2022-04-08 03:08
在C语言中,++的含义是:单目运算符,’++‘是原始值累加1。
C语言是一门通用计算机编程语言,应用广泛。C语言的设计目标是提供一种能以简易的方式编译、处理低级存储器、产生少量的机器码以及不需要任何运行环境支持便能运行的编程语言。
尽管C语言提供了许多低级处理的功能,但仍然保持着良好跨平台的特性,以一个标准规格写出的C语言程序可在许多电脑平台上进行编译,甚至包含一些嵌入式处理器(单片机或称MCU)以及超级电脑等作业平台。
二十世纪八十年代,为了避免各开发厂商用的C语言语法产生差异,由美国国家标准局为C语言制定了一套完整的国际标准语法,称为ANSI C,作为C语言最初的标准。
热心网友 时间:2022-04-08 04:43
单独的++n和n++跟n=n+1没有区别,就是对整型变量n加1
放在具体表达式中会有区别
n=1;
a=n++*2 //先计算出表达式的值,再对n自增。结果a=2;n=2
a=++n*2 //先对n自增,再计算出表达式的值。结果a=4;n=2
不知道你看懂了没。
热心网友 时间:2022-04-08 06:34
++n、--n是n最后的值为加1后的值 。 后面的相反
热心网友 时间:2022-04-08 08:42
++n即n=n+1,调用前++
--n即n=n-1,调用前--
n--即n=n-1,调用后--
n++即n=n+1,调用后--
可以用
main()
{
int n=2
printf("%d",n++);
printf("%d",n);
n=2
printf("%d",++n);
printf("%d",n);
)
看出n++和++n的区别
热心网友 时间:2022-04-08 11:07
付费内容限时免费查看回答您好,您的问题我已经看到了,正在整理答案,请稍等一会儿哦~
例如:int *a;这个语句声明了一个变量a,a的数据类型是int *,也就是整型变量的指针类型(如果不懂什么是指针,那这个问题就没有意义了)。也就是说 a的值是一个内存地址,在这个地址所在的内存空间中存放的是一个整型变量。再看:int **b;这个语句也声明了一个变量b,b的数据类型是int **,也就是整型变量的指针的指针类型(二级指针)。也就是说 b的值是一个内存地址,该地址所在的内存空间中存放的是一个整型变量的指针(一级指针,或许就是上面那个a的值)。
请问你还有其他问题咨询吗 我可以继续为您解答