oracle sql TO_CHAR函数在某些情况下添加了尾随空白

Pie*_*ard 3 sql oracle to-char

我有一个带有数字列的表,我正在尝试使用oracle sql函数to_char进行格式化.根据列的显示方式,结果会得到尾随空白.

SELECT 
  '[',TO_CHAR(ABS(Balance), 'FM0000000000000000V00'),']',
  '[' || TO_CHAR(ABS(Balance),'FM0000000000000000V00') || ']'
FROM Accounts
Run Code Online (Sandbox Code Playgroud)

会得到以下结果:

[;000000000749460366 ;];[000000000749460366]
[;000000008751094792 ;];[000000008751094792]
[;000000000000050696 ;];[000000000000050696]
[;000000000000000000 ;];[000000000000000000]
Run Code Online (Sandbox Code Playgroud)

在这两种情况下我都期待相同的结果,但我不明白为什么会有区别.我有没有理由得到这些差异?

非常感谢,Pierre-Yves

Ale*_*ole 5

它没有为添加尾随空格- 如果你检查它的长度仍然是18,如果是,那么连接版本也会有空格.列的元数据决定最大可能长度为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;];[###################]
Run Code Online (Sandbox Code Playgroud)

17位数值不能以提供的格式显示,它代替哈希值; 但有趣的是其中有19个,实际上平衡值很大.

FM修改禁止为正值的主要空间; 但是如果你有一个负16位数值,你仍然需要19位数才能显示它,因为减号会出现.知道由于ABS()调用,值不能为负,但TO_CHAR()函数不知道,因此它仍然必须允许列宽中的减号.

如果你真的从应用程序运行查询/转换,例如通过JDBC运行,那么你得到的字符串(少于17位,后面的字符串ABS())将是18个字符,并且不会有(不存在)尾随空间.

您的客户端(或至少SQL*Plus和SQL Developer)根据它认为最宽的值可以格式化列宽.你可以覆盖:

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;];[###################]
 ;#                 ; ; ;          ; ;                     
Run Code Online (Sandbox Code Playgroud)

...这使得非空间消失,但​​是如果/当原始值超过16位时会更加混乱,因为它将额外的哈希包装到下一行.

您还可以显式地将结果CAST到您期望的长度:

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;];[###################]
Run Code Online (Sandbox Code Playgroud)