oracle sql TO_CHAR 函数在某些情况下添加尾随空白
oracle sql TO_CHAR function is adding a trailing blank in some case
我有一个 table,其中有一个数字列,我正在尝试使用 oracle sql 函数 to_char 对其进行格式化。根据列的显示方式,结果是否会出现尾随空白。
SELECT
'[',TO_CHAR(ABS(Balance), 'FM0000000000000000V00'),']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts
会得到以下结果:
[;000000000749460366 ;];[000000000749460366]
[;000000008751094792 ;];[000000008751094792]
[;000000000000050696 ;];[000000000000050696]
[;000000000000000000 ;];[000000000000000000]
我希望这两种情况的结果相同,但我不明白为什么会有差异。有什么原因让我有这些差异吗?
非常感谢,
皮埃尔-伊夫
SELECT
'[', lpad(ABS(Balance),18,0) ,']',
'[' || lpad(ABS(Balance),18,0) || ']'
FROM Accounts
这显示了相同的结果。
它没有在 值 中添加尾随 space - 如果您检查长度,它仍然是 18,如果是,则连接版本将具有一个 space 也是。该列的元数据决定最大可能长度为 19,如果原始余额值超过 16 位,这似乎是正确的:
set colsep ;
with accounts(balance) as (
select 749460366 from dual
union all select 1234567890123456 from dual
union all select 12345678901234567 from dual
)
SELECT
'[',TO_CHAR(ABS(Balance), 'FM0000000000000000V00') as balance,']',
'[',length(TO_CHAR(Balance, 'FM0000000000000000V00')) as length,']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts;
';BALANCE ;';'; LENGTH;';'['||TO_CHAR(ABS(BALA
-;-------------------;-;-;----------;-;---------------------
[;000000074946036600 ;];[; 18;];[000000074946036600]
[;123456789012345600 ;];[; 18;];[123456789012345600]
[;###################;];[; 19;];[###################]
提供的格式无法显示 17 位数的值,它用哈希值代替;但有趣的是,其中有 19 个,无论实际余额有多大。
FM
修饰符抑制正值的前导space;但是如果你有一个负的 16 位数字,你仍然需要 19 位数字来显示它,因为减号会出现。 你因为ABS()
调用知道这个值不能为负,但是TO_CHAR()
函数不知道,所以它仍然必须考虑列宽中的减号。
如果您真的 运行 来自某个应用程序的 query/conversion,例如在 JDBC 之上,那么您返回的字符串(少于 17 位数字,在 ABS()
之后)将是 18 个字符,并且不会有(不存在的)尾随 space。
您的客户(或者至少,SQL*Plus 和 SQL 开发人员)正在根据其认为可能的最宽值来格式化列宽。您可以覆盖它:
column balance format A18
with accounts(balance) as (
select 749460366 from dual
union all select 1234567890123456 from dual
union all select 12345678901234567 from dual
union all select 123456789012345678 from dual
)
SELECT
'[',TO_CHAR(ABS(Balance), 'FM0000000000000000V00') as balance,']',
'[',length(TO_CHAR(Balance, 'FM0000000000000000V00')) as length,']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts;
';BALANCE ;';'; LENGTH;';'['||TO_CHAR(ABS(BALA
-;------------------;-;-;----------;-;---------------------
[;000000074946036600;];[; 18;];[000000074946036600]
[;123456789012345600;];[; 18;];[123456789012345600]
[;##################;];[; 19;];[###################]
;# ; ; ; ; ;
[;##################;];[; 19;];[###################]
;# ; ; ; ; ;
... 这使得非 space 消失,但更加混乱 if/when 原始值超过 16 位,因为它将额外的散列包装到下一行。
您也可以将结果显式转换为您期望的长度:
clear columns
with accounts(balance) as (
select 749460366 from dual
union all select 1234567890123456 from dual
union all select 12345678901234567 from dual
union all select 123456789012345678 from dual
)
SELECT
'[',CAST(TO_CHAR(ABS(Balance), 'FM0000000000000000V00') as varchar2(18)) as balance,']',
'[',length(TO_CHAR(Balance, 'FM0000000000000000V00')) as length,']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts;
';BALANCE ;';'; LENGTH;';'['||TO_CHAR(ABS(BALA
-;------------------;-;-;----------;-;---------------------
[;000000074946036600;];[; 18;];[000000074946036600]
[;123456789012345600;];[; 18;];[123456789012345600]
[;##################;];[; 19;];[###################]
[;##################;];[; 19;];[###################]
我有一个 table,其中有一个数字列,我正在尝试使用 oracle sql 函数 to_char 对其进行格式化。根据列的显示方式,结果是否会出现尾随空白。
SELECT
'[',TO_CHAR(ABS(Balance), 'FM0000000000000000V00'),']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts
会得到以下结果:
[;000000000749460366 ;];[000000000749460366]
[;000000008751094792 ;];[000000008751094792]
[;000000000000050696 ;];[000000000000050696]
[;000000000000000000 ;];[000000000000000000]
我希望这两种情况的结果相同,但我不明白为什么会有差异。有什么原因让我有这些差异吗?
非常感谢, 皮埃尔-伊夫
SELECT
'[', lpad(ABS(Balance),18,0) ,']',
'[' || lpad(ABS(Balance),18,0) || ']'
FROM Accounts
这显示了相同的结果。
它没有在 值 中添加尾随 space - 如果您检查长度,它仍然是 18,如果是,则连接版本将具有一个 space 也是。该列的元数据决定最大可能长度为 19,如果原始余额值超过 16 位,这似乎是正确的:
set colsep ;
with accounts(balance) as (
select 749460366 from dual
union all select 1234567890123456 from dual
union all select 12345678901234567 from dual
)
SELECT
'[',TO_CHAR(ABS(Balance), 'FM0000000000000000V00') as balance,']',
'[',length(TO_CHAR(Balance, 'FM0000000000000000V00')) as length,']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts;
';BALANCE ;';'; LENGTH;';'['||TO_CHAR(ABS(BALA
-;-------------------;-;-;----------;-;---------------------
[;000000074946036600 ;];[; 18;];[000000074946036600]
[;123456789012345600 ;];[; 18;];[123456789012345600]
[;###################;];[; 19;];[###################]
提供的格式无法显示 17 位数的值,它用哈希值代替;但有趣的是,其中有 19 个,无论实际余额有多大。
FM
修饰符抑制正值的前导space;但是如果你有一个负的 16 位数字,你仍然需要 19 位数字来显示它,因为减号会出现。 你因为ABS()
调用知道这个值不能为负,但是TO_CHAR()
函数不知道,所以它仍然必须考虑列宽中的减号。
如果您真的 运行 来自某个应用程序的 query/conversion,例如在 JDBC 之上,那么您返回的字符串(少于 17 位数字,在 ABS()
之后)将是 18 个字符,并且不会有(不存在的)尾随 space。
您的客户(或者至少,SQL*Plus 和 SQL 开发人员)正在根据其认为可能的最宽值来格式化列宽。您可以覆盖它:
column balance format A18
with accounts(balance) as (
select 749460366 from dual
union all select 1234567890123456 from dual
union all select 12345678901234567 from dual
union all select 123456789012345678 from dual
)
SELECT
'[',TO_CHAR(ABS(Balance), 'FM0000000000000000V00') as balance,']',
'[',length(TO_CHAR(Balance, 'FM0000000000000000V00')) as length,']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts;
';BALANCE ;';'; LENGTH;';'['||TO_CHAR(ABS(BALA
-;------------------;-;-;----------;-;---------------------
[;000000074946036600;];[; 18;];[000000074946036600]
[;123456789012345600;];[; 18;];[123456789012345600]
[;##################;];[; 19;];[###################]
;# ; ; ; ; ;
[;##################;];[; 19;];[###################]
;# ; ; ; ; ;
... 这使得非 space 消失,但更加混乱 if/when 原始值超过 16 位,因为它将额外的散列包装到下一行。
您也可以将结果显式转换为您期望的长度:
clear columns
with accounts(balance) as (
select 749460366 from dual
union all select 1234567890123456 from dual
union all select 12345678901234567 from dual
union all select 123456789012345678 from dual
)
SELECT
'[',CAST(TO_CHAR(ABS(Balance), 'FM0000000000000000V00') as varchar2(18)) as balance,']',
'[',length(TO_CHAR(Balance, 'FM0000000000000000V00')) as length,']',
'[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts;
';BALANCE ;';'; LENGTH;';'['||TO_CHAR(ABS(BALA
-;------------------;-;-;----------;-;---------------------
[;000000074946036600;];[; 18;];[000000074946036600]
[;123456789012345600;];[; 18;];[123456789012345600]
[;##################;];[; 19;];[###################]
[;##################;];[; 19;];[###################]