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

Posted

技术标签:

【中文标题】oracle sql TO_CHAR 函数在某些情况下添加尾随空白【英文标题】:oracle sql TO_CHAR function is adding a trailing blank in some case 【发布时间】:2016-09-05 08:59:16 【问题描述】:

我有一个带有数字列的表,我正在尝试使用 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]

我希望两种情况下的结果相同,但我不明白为什么会有差异。我有这些差异有什么原因吗?

非常感谢, 皮埃尔-伊夫

【问题讨论】:

【参考方案1】:

它不会在 中添加尾随空格 - 如果您检查长度,它仍然是 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;];[###################]

一个 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;];[###################]
 ;#                 ; ; ;          ; ;                     

...这会使非空格消失,但如果/当原始值超过 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;];[###################]

【讨论】:

【参考方案2】:
SELECT 
  '[', lpad(ABS(Balance),18,0) ,']',
  '[' || lpad(ABS(Balance),18,0) || ']'
FROM Accounts 

这显示了相同的结果。

【讨论】:

以上是关于oracle sql TO_CHAR 函数在某些情况下添加尾随空白的主要内容,如果未能解决你的问题,请参考以下文章

将 Oracle SQL 转换为 OBIEE Evaluate/TO_CHAR 函数

Oracle SQL TO_CHAR 函数返回值

oracle to_char 语法

如何在不使用 to_date /to_char 函数的情况下比较 oracle sql 中的日期?

在Oracle SQL中计算工作日(无函数或过程)

oracle中to_char 的用法求解