我有 2 个日期时间选择器,允许用户选择开始日期和结束日期。所以在选择日期之后,用户必须点击一个按钮来运行 sql 查询。因此根据所选日期过滤输出。
但是当我如下声明日期时出现错误(下面的代码放在按钮点击事件下):
Dim startdate As DateTime = Format(DateTimePicker1.Value, "dd/MM/yyyy")
Dim enddate As DateTime = Format(Microsoft.VisualBasic.DateAdd(DateInterval.Day, 1, DateTimePicker2.Value), "dd/MM/yyyy")
query = "Select * from records where recorddate between '" & startdate & "' and '" & enddate & "'"
错误在这里:
Dim startdate As DateTime = Format(DateTimePicker1.Value, "dd/MM/yyyy")
错误是:
Conversion from string "13/02/2016" to type 'Date' is not valid.
我记录的样本日期是这样的:
12/2/2016 7:28:26 PM
我不知道我在这里错过了什么。请帮忙。谢谢
最佳答案
您的代码中存在多个问题。
Dim startdate As DateTime = Format(DateTimePicker1.Value, "dd/MM/yyyy")
如果您将鼠标悬停在 Format
上,Intellisense 会告诉您它返回一个字符串。由于无法将字符串结果分配给声明为 DateTime
的变量,因此会出现错误。如果你打开 Option Strict
,编译器会告诉你这些。
其次,查询还将日期转换为字符串:
"...between '" & startdate & "' and '" & enddate & "'"
标记不是通用的 SQL 分隔符,它们是一种指定文本/字符串数据的方式。 SQL 参数更安全,可以防止意外的数据类型更改以及 SQL injection attacks .它们还使代码更易于阅读。如果您的数据库中有名称列,请尝试插入一个名称,例如 Tim O'Brien
、D'Angelo Barksdale
或 Betty's Cupcake Factory
.查询将崩溃。 SQL 参数可以防止这种情况。
我不知道是哪个数据库,所以我猜是 Access。这无关紧要,因为 DB Provider 的工作方式大同小异:
' Dim startdate As DateTime = DateTimePicker1.Value
Dim enddate As DateTime = DateTimePicker2.Value.AddDays(1)
Dim SQL = "Select * from records where recorddate between @p1 AND @p2"
Using dbCon As New OleDbConnection(connstr)
Using cmd As New OleDbCommand(SQL, dbCon)
dbCon.Open()
cmd.Parameters.Add("@p1", OleDbType.DBDate).Value = startdate
cmd.Parameters.Add("@p2", OleDbType.DBDate).Value = DateTimePicker1.Value
dt = New DataTable
dt.Load(cmd.ExecuteReader)
End Using
End Using
注意 这假定数据库中的列类型是日期而不是字符串。 BETWEEN
子句不太可能将日期用作字符串。如果您希望它们充当日期,请将它们保存为 Date
。您不需要“为了在 SQL 中使用它而需要转换为字符串”。曾经。
DateTime
类型传递给数据库。DateTimePicker1.Value
分配给新变量,因为 .Value
是 DateTime
类型。Using
block 声明并创建 DB 对象,然后关闭并释放它们以释放资源OleDbType.DBDate
),然后 Value 集是一个实际的日期类型。对于 Access/OleDB,重要的是按照它们在 SQL 中出现的完全相同的顺序设置参数值。 OleDB
允许命名参数,但会按照它们在 SQL 中出现的顺序分配值。
要强调的是,MSDN(和其他人)鼓励使用 ?
代替其他形式("@p1"
、"@firstname"
,“@DateOfBirth”
)。尤其是在包含多个列的查询中,我喜欢 "@p1"
,因为数字有助于索引 SQL 中的相关列并有助于确保它们是有序的(视觉上)。
最后,您的 BETWEEN 子句是否按您希望的方式工作将取决于几个因素,包括该 DataType 参数。 DateTimePicker
中的日期将包括时间。为了不排除某些行,因为时间部分早于您从 DateTimePicker
获得的任何内容,您需要使用正确的行。
关于.net - 从字符串到类型 'Date' 的转换无效,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35377152/
相关文章:
azure - 是 Console.ReadKey();适合 azure webjob
vim - tmux 将选择从复制模式发送到其他 Pane (xargs 加入?)
emacs - 使 Emacs/Slime/Quicklisp/SBCL 在 Windows 中工作
intellij-idea - Intellij 理念 : Override and update
ruby-on-rails - 如何连接表以仅获取 has-many 关联的最后一条记录?
emacs - 在 emacs 中使用 orgtbl 在 latex 中垂直线条
.htaccess - 只允许通过 VPN 私有(private)访问网站