解决 Entity Framework 6.0 decimal 类型精度问题
Ø 前言
本文主要解决 EF 中对于 MSSQL 数据库的 decimal 类型经度问题,经实验该问题仅在 CodeFirst 模式的情况下发生,话不多说直接看代码。
1. 假设我们有一张 Customer 数据表,主要探究:Longitude、Latitude、LonLatSum 这三个字段。
1) 结构如下:
CREATE 的数据)
1) C# 代码如下:
using (MyTestingEntities context = new MyTestingEntities())
{
Customer entity = context.Customers.Attach(new Customer() { Id = 1 });
entity.Longitude = 123.1256789f; //123.125679
entity.Latitude = 456.1295678d; //456.1295678
entity.LonLatSum = (decimal)(entity.Longitude + entity.Latitude); //579.255246816113M
context.Configuration.ValidateOnSaveEnabled = false;
result = context.SaveChanges() > 0;
}
2) 生成SQL:
exec 位小数。
2. Latitude:float 类型(对应 C# 中的 double 类型),保留了7位小数。
3. LonLatSum:decimal 类型(对应 C# 中的 decimal 类型),也保留了7位小数。
4. OK 这是正常的。
3. 然后,我们再使用 Code Frirst 的方式对数据更新(更新 Id 为2的数据)
1) C# 代码如下:
using (MyTestingContext context = new MyTestingContext())
{
Customer entity = context.Customer.Attach(new Customer() { Id = 2 });
entity.Longitude = 123.1256789f; //123.125679
entity.Latitude = 456.1295678d; //456.1295678
entity.LonLatSum = (decimal)(entity.Longitude + entity.Latitude); //579.255246816113M
result = context.SaveChanges() > 0;
}
return result;
2) 生成SQL:
exec )和小数位数(2位)的方式生成了,结果 SQL 的类型声明是这样:decimal(18,2)。
3) 搞清楚了问题,下面我们就来解决这个问题吧。
5. 解决问题
1) 创建一个 DecimalPrecisionAttribute 特性类
/// <summary>
/// 用于指定 decimal 类型的精确度与小数保留位数。
/// </summary>
[AttributeUsage(AttributeTargets.Property, Inherited = false, AllowMultiple = false)]
public class DecimalPrecisionAttribute : Attribute
{
private byte _precision;
/// <summary>
/// 精确度。
/// </summary>
public byte Precision
{
get { return _precision; }
set { _precision = value; }
}
private byte _scale;
/// <summary>
/// 小数保留位数。
/// </summary>
public byte Scale
{
get { return _scale; }
set { _scale = value; }
}
/// <summary>
/// 根据指定的精确度与小数保留位数,初始化 DecimalPrecisionAttribute 的实例。
/// </summary>
/// <param name="precision">精确度。</param>
/// <param name="scale">小数保留位数。</param>
public DecimalPrecisionAttribute(byte precision, byte scale)
{
this.Precision = precision;
this.Scale = scale;
}
}
2) 再创建一个 DecimalPrecisionAttributeConvention 类(表示 DecimalPrecisionAttribute 的一种约定)
1. 该类继承于 System.Data.Entity.ModelConfiguration.Conventions.PrimitivePropertyAttributeConfigurationConvention 类。
2. 并实现 Apply 抽象方法,通俗点说:该方法在“生成 SQL”时被调用,对于打了 DecimalPrecisionAttribute 标记的实体属性,精度将根据 configuration.HasPrecision() 方法中的设置去生成。
/// <summary>
/// 表示 DecimalPrecisionAttribute 的一种约定。
/// </summary>
public class DecimalPrecisionAttributeConvention
: PrimitivePropertyAttributeConfigurationConvention<DecimalPrecisionAttribute>
{
public override void Apply(ConventionPrimitivePropertyConfiguration configuration, DecimalPrecisionAttribute attribute)
{
if (attribute.Precision < 1 || attribute.Precision > 38)
{
throw new InvalidOperationException("Precision must be between 1 and 38.");
}
if (attribute.Scale > attribute.Precision)
{
throw new InvalidOperationException("Scale must be between 0 and the Precision value.");
}
configuration.HasPrecision(attribute.Precision, attribute.Scale);
}
}
3) 在数据上下文的 OnModelCreating() 方法将该 DecimalPrecisionAttributeConvention(约定)加入数据约定集合中。
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.Conventions.Add(new DecimalPrecisionAttributeConvention());
}
4) 最后一步,将需要设置小数位数的属性打上 DecimalPrecision 标记,例如:
[DecimalPrecision(18, 7)]
public decimal LonLatSum { get; set; }
5) 好了之后再次运行代码,生成 SQL 如下:
exec sp_executesql N'UPDATE [dbo].[Customer]
SET [Longitude] = @0, [Latitude] = @1, [LonLatSum] = @2
WHERE ([Id] = @3)
',N'@0 real,@1 float,@2 decimal(18,7),@3 int',@0=123.12567901611328,@1=456.12956780000002,@2=579.2552468,@3=2
6) 结果如下:
7) OK,这样就与 DB First 生成的 SQL 没什么区别了。
解决 Entity Framework 6.0 decimal 类型精度问题的更多相关文章
- 在Entity Framework 4.0中使用 Repository 和 Unit of Work 模式
[原文地址]Using Repository and Unit of Work patterns with Entity Framework 4.0 [原文发表日期] 16 June 09 04:08 ...
- Entity Framework 教程——Entity Framework中的实体类型
Entity Framework中的实体类型 : 在之前的章节中我们介绍过从已有的数据库中创建EDM,它包含数据库中每个表所对应的实体.在EF 5.0/6.0中,存在POCO 实体和动态代理实体两种. ...
- 开发 ASP.NET vNext 续篇:云优化的概念、Entity Framework 7.0、简单吞吐量压力测试
继续上一篇<开发 ASP.NET vNext 初步总结(使用Visual Studio 2014 CTP1)>之后, 关于云优化和版本控制: 我本想做一下MAC和LINUX的self-ho ...
- Entity Framework 5.0系列之Code First数据库迁移
我们知道无论是"Database First"还是"Model First"当模型发生改变了都可以通过Visual Studio设计视图进行更新,那么对于Cod ...
- Entity Framework 5.0
今天 VS2012 .net Framework 4.5 Entity Framework 5.0 三者共同发布了. ( EF5 Released ) 在介绍新特性之前,先与大家回顾一下EF版 ...
- 云优化的概念、Entity Framework 7.0、简单吞吐量压力测试
云优化的概念.Entity Framework 7.0.简单吞吐量压力测试 继续上一篇<开发 ASP.NET vNext 初步总结(使用Visual Studio 2014 CTP1)>之 ...
- [EF2]Sneak Preview: Persistence Ignorance and POCO in Entity Framework 4.0
http://blogs.msdn.com/b/adonet/archive/2009/05/11/sneak-preview-persistence-ignorance-and-poco-in-en ...
- 精进不休 .NET 4.5 (12) - ADO.NET Entity Framework 6.0 新特性, WCF Data Services 5.6 新特性
[索引页][源码下载] 精进不休 .NET 4.5 (12) - ADO.NET Entity Framework 6.0 新特性, WCF Data Services 5.6 新特性 作者:weba ...
- 浅析Entity Framework Core2.0的日志记录与动态查询条件
前言 Entity Framework Core 2.0更新也已经有一段时间了,园子里也有不少的文章.. 本文主要是浅析一下Entity Framework Core2.0的日志记录与动态查询条件 去 ...
随机推荐
- sql 语句查所有父级
常见问题,给一个记录ID,查出它的所有父级,直到顶级 使用SMSS,sql server,找到一个办法. 思路是分两步,先循环找到所有父级的ID,再用IN查出所有父级 列说明 ID=PK Pare ...
- Mysql数据库操作笔记
如果数据库表字段存在,则删除该表 drop table if exists `table_name` 创建数据库表语句 create table `table_name`( `id` ) not n ...
- luogu3759 不勤劳的图书管理员 (树状数组套线段树)
交换的话,只有它们中间的书会对答案产生影响 树状数组记位置,套线段树记书的编号 它对应的页数和书的个数 然后就是减掉中间那些原来是逆序对的,再把交换以后是逆序对的加上 别忘了考虑这两个自己交换以后是不 ...
- PHP使用自定义key实现对数据加密解密
// 加密 function encryptStr($str, $key){ $block = mcrypt_get_block_size('des', 'ecb'); $pad = $block - ...
- CAS与ABA问题产生和解决
乐观锁和悲观锁 Synchronized属于悲观锁,悲观地认为程序中的并发情况严重,所以严防死守.CAS属于乐观锁,乐观地认为程序中的并发情况不那么严重,所以让线程不断去尝试更新. 性能对比: Syn ...
- struts2 UI标签 和 主题
四.Struts2的UI标签和主题 1.Struts2中UI标签的优势 自动的数据回显和错误提示功能 自带的简单样式和排版 2.表单标签的通用属性 说明:UI标签中value的取值一般都是字符串. 2 ...
- python学习笔记—Day1
1. python使用<变量名>=<表达式>的方式对变量进行赋值 a=1; python中数分为整数和浮点数 字符串的定义一定要用引号,单引号和双引号是等价的 三引号用来输入包 ...
- toString()和toLocaleString()有什么区别
偶然之间用到这两个方法 然后在数字转换成字符串的时候,并没有感觉这两个方法有什么区别,如下: 1 2 3 4 5 6 7 8 var e=123 e.toString() "123& ...
- decorator 装饰页面,根据不同设备自动切换移动和pc站
package com.thinkgem.jeesite.modules.sys.interceptor; import javax.servlet.http.HttpServletRequest; ...
- Vue(小案例)底部tab栏和顶部title栏的实现
---恢复内容开始--- 一.前言 1.底部tab栏实现 2.顶部title栏实现 二.主要内容 1.底部tab栏实现(将底部导航提取到公共的组件中) 具体效果:当点击切换不同的tab的时候,对应 ...