c# - 在 DateTime.Now 上应用 CRUD 操作的正确方法是什么?
问题描述
我有一个 SQL Server 表,如下所示:
CREATE TABLE [dbo].[TimePeriod](
[ID] [int] NOT NULL,
[Time] [datetime2](7) NOT NULL,
[Description] [varchar](max) NULL,
CONSTRAINT [PK_TimePeriod] PRIMARY KEY CLUSTERED
(
[ID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF,
ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]
我有一个如下的单元测试:
[TestMethod]
public void Save()
{
OrmEngine.Initalize();
DateTime dateTime = DateTime.Now;
TimePeriod item = new TimePeriod();
item.Time = dateTime;
item.Description = item.Time.Millisecond.ToString() + "=Description";
TimePeriodBllManually bll = new TimePeriodBllManually();
int newId = bll.Save(item);
TimePeriod returns = bll.Get(newId);
Assert.IsNotNull(returns);
Assert.AreEqual(item.Time, returns.Time);
Assert.AreEqual(item.Description, returns.Description);
}
SQL Server 数据:
在插入操作期间似乎正在更改时间。无论 SQL 数据类型是datetime
or 还是,结果都是一样的datetime2
。
我认为,从 C# 的角度来看,DateTime
它是一种不可变类型。但是,在这里我看到dateTime
即使在分配之后它的值也在改变。
我该如何解决这个问题?
解决方案
滴答数和毫秒数会有所不同——因此断言很可能总是失败。
如果可以忽略毫秒数,那么我们查看总秒数以进行断言:
int x = (int)DateTime.Now.TimeOfDay.TotalSeconds;
int y = (int)DateTime.Now.TimeOfDay.TotalSeconds;
Assert.AreEqual(x, y);
编辑
关注 Icepickle 评论和 user366312 额外信息 - 我做了更多挖掘工作,我可以确认该问题与 DateTime 无关。
这里的问题是在将日期时间类型存储到数据库时会丢失精度。
根据 user366312 的示例,如果我们使用以下日期时间值DateTime(2020, 7, 25, 15, 10, 20, 30);
,则在存储到 db 时不会损失精度
但是,如果我们使用来自 DateTime.Now 的值
item.Time
{25-Jul-20 12:31:11 AM}
Date: {25-Jul-20 12:00:00 AM}
Day: 25
DayOfWeek: Saturday
DayOfYear: 207
Hour: 0
Kind: Local
Millisecond: 368
Minute: 31
Month: 7
Second: 11
Ticks: 637312338713680636
TimeOfDay: {00:31:11.3680636}
Year: 2020
我们会损失精度 - 导致该returns
项目提供的内容:
returns.Time
{25-Jul-20 12:31:11 AM}
Date: {25-Jul-20 12:00:00 AM}
Day: 25
DayOfWeek: Saturday
DayOfYear: 207
Hour: 0
Kind: Unspecified
Millisecond: 367
Minute: 31
Month: 7
Second: 11
Ticks: 637312338713670000
TimeOfDay: {00:31:11.3670000}
Year: 2020
就目前而言,该问题将/可以通过更改数据库上数据类型的设计来解决。
user366312 提供的解决方案将起作用 - 但正如 Icepickle 指出的那样,我们正在降低/失去精度。
更健壮的实现是将数据库中的时间列更改为datetime2(7)
EDIT 2 CRUD 操作实现:
我创建了一个小型应用程序,以便在数据库中添加 1000 个条目。这是实现。
我的环境:
Microsoft SQL Server Management Studio 11.0.5058.
Microsoft Visual Studio Professional 2019
Version 16.6.3
VisualStudio.16.Release/16.6.3+30225.117
Microsoft .NET Framework
Version 4.8.03761
Installed Version: Professional
数据库表实现:
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
SET ANSI_PADDING ON
GO
CREATE TABLE [dbo].[TimePeriod](
[ID] [int] IDENTITY(1,1) NOT NULL,
[Description] [varchar](50) NULL,
[Time] [datetime2](7) NULL,
CONSTRAINT [PK_TimePeriod] PRIMARY KEY CLUSTERED
(
[ID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
GO
SET ANSI_PADDING OFF
GO
TimePeriodEntity 类
#region Usings
using System;
#endregion
public class TimePeriodEntity
{
#region Properties
public int ID { get; set; }
public DateTime Time { get; set; }
public string Description { get; set; }
#endregion
}
CRUD操作类
#region Usings
using System;
using System.Collections.Generic;
using System.Data;
using System.Data.SqlClient;
#endregion
public static class CRUDOperation
{
#region Fields
private const string ConnectionString = @"Data Source=.;Database=TestDB;Integrated Security=SSPI";
static DataTable DataTable;
#endregion
static CRUDOperation()
{
DataTable = new DataTable();
}
public static List<TimePeriodEntity> Read()
{
List<TimePeriodEntity> list = new List<TimePeriodEntity>();
string query = "SELECT * FROM TimePeriod";
using (SqlConnection connection = new SqlConnection(ConnectionString))
{
SqlCommand command = new SqlCommand(query, connection);
connection.Open();
using (SqlDataAdapter adaptor = new SqlDataAdapter(command))
{
adaptor.Fill(DataTable);
foreach (DataRow row in DataTable.Rows)
{
TimePeriodEntity tp = new TimePeriodEntity();
tp.ID = (int)row[0];
tp.Description = row[1].ToString();
tp.Time = Convert.ToDateTime(row[2]);
list.Add(tp);
}
}
}
return list;
}
public static void Create(DateTime time)
{
string query = "INSERT INTO TimePeriod(Description,Time) VALUES(@param2,@param3)";
using (SqlConnection connection = new SqlConnection(ConnectionString))
{
SqlCommand command = new SqlCommand(query, connection);
connection.Open();
command.Parameters.Add("@param2", SqlDbType.VarChar).Value = $"{time.Millisecond} = Description";
command.Parameters.Add("@param3", SqlDbType.DateTime2).Value = time;
command.CommandType = CommandType.Text;
command.ExecuteNonQuery();
}
}
}
添加到数据库中的行采用以下格式:
ID Description Time
606 142 = Description 2020-07-29 08:37:38.1420528
这是我的结果的一个子集:
推荐阅读
- angular - 通过路由重复页面导致后退按钮出现问题
- html - 如何避免过大的 DOM 大小?
- git - Gitlab + Jira 集成与多个分支的合并/拉取请求的自动问题转换
- css - 如何在复杂的弹性盒中不使用绝对
- automated-tests - 具有大量控件的页面的 Specflow
- c++ - C++ 98 中的部分模板专业化?
- php - 我收到错误消息:“无法加载请求的文件”。如何解决问题?
- python - 有没有办法在 excel 和 power point 之间自动执行复制粘贴操作?
- google-apps-script - 如何解决共享 Google Apps 脚本库的“此应用被阻止”错误?
- java - 当状态码为 400 时,volley 返回空响应