Вот специальное решение, которое я написал для этого.К сожалению, есть повторяющаяся логика, так как я использую это в постоянном вычисляемом столбце и, следовательно, не могу использовать переменные или скалярные функции (для параметров, которые будут использоваться в качестве переменных).
DECLARE @Y INT = 2000, @M INT = 2, @D INT = 29, @T FLOAT = 135559.999
SELECT IIF
(
--If the Year is out-of-bounds.
@Y <= 0 OR @Y > 9999
--Or the Month is out-of-bounds.
OR @M <= 0 OR @M > 12
--Or the Day is out-of-bounds (Accounts for leap years).
OR @D <= 0 OR @D > DAY(EOMONTH(DATETIME2FROMPARTS(@Y, @M, 1, 0, 0, 0, 0, 3)))
--Or the Time is less than 0
OR @T < 0
--Or the Hour is out-of-bounds.
OR ROUND(@T / 10000, 0, 1) >= 24
--Or the Minute is out-of-bounds.
OR ROUND(@T / 100, 0, 1) - ROUND(@T / 10000, 0, 1) * 100 >= 60
--Or the Second is out-of-bounds.
OR ROUND(@T, 0, 1) - ROUND(@T / 100, 0, 1) * 100 >= 60,
--NULL is returned
NULL,
--Otherwise, the Date Time components are parsable into a DATETIME2.
DATETIME2FROMPARTS
(
--Year
@Y,
--Month
@M,
--Day
@D,
--Hour
ROUND(@T / 10000, 0, 1),
--Minute
ROUND(@T / 100, 0, 1) - ROUND(@T / 10000, 0, 1) * 100,
--Second
ROUND(@T, 0, 1) - ROUND(@T / 100, 0, 1) * 100,
--Millisecond (multiplied by 1000 to use the first 3 decimal places).
(@T - ROUND(@T, 0, 1)) * 1000,
--Precision (3 is specified since only 3 decimal places become part of the integer for the fraction parameter above).
3
)
)
Если вам нужна точность в миллисекундах, отличная от3 знака после запятой, где ваша точность в миллисекундах равна x
, измените 3
на x
и * 1000
в строке выше на * POWER(10, x)
.