博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
关于有默认值的字段在用EF做插入操作时的思考
阅读量:6006 次
发布时间:2019-06-20

本文共 2209 字,大约阅读时间需要 7 分钟。

原文:

今天在用EF做插入操作的时候发现数据库中一个datetime类型的字段(CreateDate)的值居然全部为null。于是赶紧看表结构发现CreateDate字段居然是允许为空的。

虽然为空,但是设置了默认值getdate(),按说不应该为null的。于是开始测试。

字段允许Null值的情况

Users表结构如下:

假如一个字段有了默认值,并且又允许为Null,在做插入操作时会发生什么?

如上图中的表结构,CreateDate是允许为null的,而又有默认值getdate()。这样在用传统SQL语句和EF做插入操作时分别会发生什么呢?

先看传统SQL语句:

insert into Users(Username,Password) values('test','123456')

插入结果:

使用SQL完全没有问题。

再看使用EF:

Users user = new Users();user.Username = "test2";user.Password = "123456";TestEntities entity = new TestEntities();entity.Users.Add(user);int result = entity.SaveChanges();

插入结果:

使用EF插入居然为NULL!!!于是立刻打开SQL Server Profiler监测生成的SQL,EF居然生成了下面的SQL:

没赋值的字段,EF生成SQL的时候居然都置为NULL,管你有没有设默认值!

可能是EF比较娇情,我字段允许为NULL,EF生成SQL时就置为null了。那我把字段设为不允许为NULL,EF又会生成怎样的SQL呢?

字段不允许Null值的情况

Users表结构如下:

这一次CreateDate不允许为NULL了,使用SQL理所当然地一切正常:

insert into Users(Username,Password) values('admin','123456')

那再看下用EF:

Users user = new Users();user.Username = "admin";user.Password = "123456";TestEntities entity = new TestEntities();entity.Users.Add(user);int result = entity.SaveChanges();

结果。。。

”从 datetime2 数据类型到 datetime 数据类型的转换产生一个超出范围的值。“

EF很傲娇地抛了个异常!!为什么会抛出这个异常?这个异常神马意思?于是照妖镜SQL Server Profiler又出场了:

日期类型的字段如果不赋值会给默认值“0001-01-01 00:00:00”,这和生成SQL无关,是在生成SQL之前CLR进行的初始化操作,同理String类型的字段如果不赋值会给默认值String.Empty!

但是,那也不至于抛异常啊,顶多字段值存储为“0001-01-01 00:00:00”呗。

于是查MSDN发现datetime类型的日期范围仅支持1750-01-01 00:00:00至9999-12-31 23:59:59.997,而0001-01-01 00:00:00不在此范围内,所以抛出异常。原来如此。

datetime和datetime2支持的日期范围如下:

结论

如此看来无论字段有没有默认值,使用EF插入时都没有任何实际作用了。因此如果你使用EF建议干脆直接写死算了。比如user.CreateDate = DateTime.Now。

ps.这里只是举个例子,事实上,在实际项目中不建议写成DateTime.Now,因为数据库系统时间和服务器时间一般是不同的,笔者实际项目中其实是封装了一个方法叫GetSQLServerSystemDateTime。

除此之外是否还有别的方法呢?

方法是有的,打开EF的.edmx文件,找到<EntityType Name="表名">节点,把如下节点:

改成:

即可。

StoreGeneratedPattern是一个枚举:
  • None 一个值,指示它不是服务器生成的属性。这是默认值。如果没有StoreGeneratedPattern属性,其值就默认为None。
  • Identity 执行插入时生成一个值,但在执行更新时保持不变。
  • Computed 执行插入和更新时都将生成一个值。

笔者经过实际测试发现:

  1. 如果将StoreGeneratedPattern值设置为Identity,只要一修改CreateDate字段就会抛异常;
  2. 如果把StoreGeneratedPattern值设置为Computed不会抛异常,但值仍然没有被修改,即使你写了user.CreateDate = "xxx"。

所以如果你使用EF,我还是建议采用第一种方式,直接在程序中赋值user.CreateDate = xx.GetSQLServerSystemDateTime()。这种方法省事且安全。修改edmx文件太麻烦,况且每增加一个datetime类型的字段都要修改一次edmx文件。

最后感谢大家提出的建议,欢迎大家亲自测试并把结果反馈给我。

 

延伸阅读

转载地址:http://jmsmx.baihongyu.com/

你可能感兴趣的文章
ORACLE系列脚本1:救命的应急会话处理脚本
查看>>
计算/etc/目录所有*.conf配置文件所占总空间大小
查看>>
python——ip加入nginx黑名单
查看>>
自动化技术初出茅庐,ansible使用解析。
查看>>
Microsoft Azure Site Recovery (2) 配置虚拟机保护
查看>>
Apache CouchDB安装及入门
查看>>
企业实际应用之同步远程yum源到本地
查看>>
一次核心线上磁盘差点爆满坑人事件...
查看>>
Oracle 添加RAC数据库集群节点(二)
查看>>
AIX 5 ftp 文件传输
查看>>
企业Java应用服务器之JBoss7.1与Apahce整合
查看>>
oculus,一种未来的交流方式
查看>>
选择优异Linux培训机构的几点建议
查看>>
Rsync镜像同步工具的安装配置
查看>>
python脚本判断一个数是否为素数的几种方法
查看>>
血性的青,尿性的春——《中国合伙人》影评
查看>>
关于python调用zabbix api接口的自动化实例 [结合saltstack]
查看>>
Exchange更改默认的OST、PST路径
查看>>
移动应用UI设计中的48dp定律
查看>>
Wijmo 更优美的jQuery UI部件集:从wijwizard和wijpager开始
查看>>