如何解决SQL 时间戳插入添加额外的零
我创建了下表 --
CREATE TABLE placed_order
(
bill_number NUMBER(6) NOT NULL,customer_id NUMBER(6),order_time TIMESTAMP,paymentid NUMBER(6),total_bill_amount NUMBER(10,2),scheduled_delivery_time TIMESTAMP,address_id NUMBER(6),PRIMARY KEY (bill_number),FOREIGN KEY (customer_id) REFERENCES customer(customer_id),FOREIGN KEY (paymentid) REFERENCES payment_information(paymentid),FOREIGN KEY (address_id) REFERENCES delivery_address(address_id)
);
当我向表中插入以下值时 --
INSERT INTO placed_order
VALUES (123456,1,'01-Feb-21 12:00:00 AM',34.56,'03-Feb-21 12:00:00 PM',1);
时间存储格式如下--
123456 1 01-FEB-21 12.00.00.000000000 AM 1 34.56 03-FEB-21 12.00.00.000000000 PM 1
当我尝试运行语句以对 SQL 进行操作(例如平均值)时,这会导致问题。例如,即使我想使用 to_date 将其转换为日期 --
select (TO_DATE(actual_delivery_time)) from delivery;
我收到以下错误 --
ORA-01830:日期格式图片在转换整个输入字符串之前结束
01830. 00000 - “日期格式图片在转换整个输入字符串之前结束”
我尝试检查 nls_database_parameters 中的时间戳格式,结果如下 -- NLS_TIMESTAMP_FORMAT DD-MON-RR HH.MI.SSXFF AM
如何确定附加零是什么以及为什么要附加它们?
解决方法
A TIMESTAMP
是一种二进制数据格式,包含 7-20 个字节(世纪、世纪年、月、日、时、分、秒,小数秒最多 6 个字节,小数秒最多 7 个字节)时区信息的字节);它没有格式。
'01-Feb-21 12:00:00 AM'
是 NOT TIMESTAMP
数据类型;它是一个字符串文字,您依赖于从字符串到时间戳的隐式转换。 Oracle 将执行此操作,您的查询将有效:
INSERT INTO placed_order
VALUES (
123456
1,TO_TIMESTAMP(
'01-Feb-21 12:00:00 AM',( SELECT value FROM NLS_SESSION_PARAMETERS WHERE parameter = 'NLS_TIMESTAMP_FORMAT' )
),1,34.56,TO_TIMESTAMP(
'03-Feb-21 12:00:00 PM',1
);
但是,如果 NLS_TIMESTAMP_FORMAT
与您的字符串格式不匹配,则转换将失败。
查询应该怎么写?
你应该:
- 显式使用
TO_TIMESTAMP( '01-Feb-21 12:00:00 AM','DD-MON-RR HH12:MI:SS AM' )
而不是依赖隐式转换 - 使用时间戳文字
TIMESTAMP '2021-02-01 00:00:00'
;或 - 使用日期文字
DATE '2021-02-01'
。
例如:
INSERT INTO placed_order (
bill_number,customer_id,order_time,paymentid,total_bill_amount,scheduled_delivery_time,address_id
) VALUES (
123456
1,TIMESTAMP '2021-02-01 00:00:00',TIMESTAMP '2021-02-03 12:00:00',1
);
如何格式化 TIMESTAMP
值?
如果您想用某种格式显示时间戳(请记住,TIMESTAMP
不以任何格式存储),那么您想使用 TO_CHAR
对其进行转换到一个可以有格式的字符串。
如果要将 TIMESTAMP
格式化为 YYYY-MM-DD
字符串,请使用:
SELECT TO_CHAR(actual_delivery_time,'YYYY-MM-DD') FROM delivery
例如,即使我想使用 to_date 将其转换为日期
TO_DATE
不会将 TIMESTAMP
转换为 DATE
;它将字符串转换为 DATE
。如果要将 TIMESTAMP
转换为 DATE
数据类型,请使用:
SELECT CAST(actual_delivery_time AS DATE) FROM delivery
(注意:DATE
数据类型也是二进制数据类型,不以任何格式存储;但是,您用来访问数据库的用户界面可能会选择显示使用 NLS_DATE_FORMAT
会话参数的日期。)
为什么显示小数秒?
我尝试检查 nls_database_parameters 中的时间戳格式,结果如下 -- NLS_TIMESTAMP_FORMAT DD-MON-RR HH.MI.SSXFF AM
谁能帮忙弄清楚额外的零是什么以及为什么要附加它们?
您正在显示二进制数据类型;您正在使用的用户界面试图提供帮助,而不是显示字节(人们可能难以理解),而是使用 NLS_TIMESTAMP_FORMAT
将二进制数据隐式转换为格式化字符串。
假设您的 NLS_TIMESTAMP_FORMAT
是 DD-MON-RR HH.MI.SSXFF AM
,那么 FF
格式模型将显示小数秒;在这种情况下,您的小数秒为 0
,它将显示达到时间戳的最大精度。
更改 NLS_TIMESTAMP_FORMAT
,它将改变(某些)用户界面显示 TIMESTAMP
的方式(但是,它不会改变数据的存储方式)。>
是否可以在没有小数秒的情况下存储?
如果您想更改数据的存储方式,使其没有小数秒,则:
- 为您使用的
TIMESTAMP
数据类型提供小数秒的显式精度并使用TIMESTAMP(0)
;或 - 使用
DATE
数据类型。
这两种数据类型都将使用 7 字节存储值,并将存储世纪、世纪年、月、日、小时、分钟和秒。两者差别不大;一个区别是时间戳将使用 NLS_TIMESTAMP_FORMAT
隐式格式化,日期使用 NLS_DATE_FORMAT
会话参数进行隐式格式化。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。