将VARCHAR2转换为数字

时间:2013-05-29 11:32:40

标签: sql oracle

我有一个名为duration的列的表。其数据类型为VARCHAR2

我想对列duration求和。

00:56:30
02:08:40
01:01:00

总共=> 04:05:10

如何使用ANSI SQL或Oracle SQL执行此操作?

4 个答案:

答案 0 :(得分:10)

您可以使用SUBSTR,然后SUM分隔小时,分钟和秒,最后使用NUMTODSINTERVAL函数将其转换为INTERVAL类型。

SELECT NUMTODSINTERVAL (SUM (total_secs), 'second')
  FROM (SELECT   SUBSTR (duration, 1, 2) * 3600
               + SUBSTR (duration, 4, 2) * 60
               + SUBSTR (duration, 7, 2) total_secs
          FROM user_tab);

答案 1 :(得分:7)

我认为最好先将字符串转换为INTERVAL,然后将这些值添加为日期值。有点像:

select   to_dsinterval('0 00:56:30') 
       + to_dsinterval('0 02:08:40') 
       + to_dsinterval('0 01:01:00') myinterval from dual;

MYINTERVAL
-------------------
+000000000 04:06:10

答案 2 :(得分:4)

FOR ORACLE

select 
numtodsinterval(sum(
  to_char(to_date(duration, 'HH24:MI:SS'), 'HH24') * 3600 + 
  to_char(to_date(duration, 'HH24:MI:SS'), 'MI') * 60+
  to_char(to_date(duration, 'HH24:MI:SS'), 'SS')
  ), 'second'
) as SUMTOTAL
from tbl;

第二次查询

select 
numtodsinterval(hr+mn+sc, 'second')
from 
(
select 
sum(to_char(to_date(duration, 'HH24:MI:SS'), 'HH24') * 3600) as hr,
sum(to_char(to_date(duration, 'HH24:MI:SS'), 'MI') * 60) as mn,
sum(to_char(to_date(duration, 'HH24:MI:SS'), 'SS'))as sc
from tbl) tmp

FIDDLE

在SQL FIDDLE中返回对象的示例。试试你的机器

FOR MYSQL

试试这个

select sec_to_time(sum(time_to_sec(duration))) from tbl

FIDDLE

| SEC_TO_TIME(SUM(TIME_TO_SEC(DURATION))) |
-------------------------------------------
|          January, 01 1970 04:06:10+0000 |

答案 3 :(得分:2)

为了好玩,我编写了自己的聚合函数,可以对间隔求和(参见@ Yasir的帖子)。这可以修改为在内部进行varchar到区间转换,但我现在会尽可能地保持它简单。)

首先创建对象类型规范:

CREATE OR REPLACE TYPE SumInterval 
AS OBJECT (

runningSum INTERVAL DAY(9) TO SECOND(9),

STATIC FUNCTION ODCIAggregateInitialize
  ( actx IN OUT SumInterval
  ) RETURN NUMBER,

MEMBER FUNCTION ODCIAggregateIterate
  ( self  IN OUT SumInterval,
    val   IN       DSINTERVAL_UNCONSTRAINED
  ) RETURN NUMBER,

MEMBER FUNCTION ODCIAggregateTerminate
  ( self             IN   SumInterval,
    returnValue  OUT DSINTERVAL_UNCONSTRAINED,
    flags           IN   NUMBER
  ) RETURN NUMBER,

MEMBER FUNCTION ODCIAggregateMerge
  (self  IN OUT SumInterval,
   ctx2 IN      SumInterval
  ) RETURN NUMBER

);

对象体:

CREATE OR REPLACE TYPE BODY SumInterval AS

STATIC FUNCTION ODCIAggregateInitialize
  ( actx IN OUT SumInterval
  ) RETURN NUMBER IS 
  BEGIN
    IF actx IS NULL THEN
      actx := SumInterval (INTERVAL '0 0:0:0.0' DAY TO SECOND);
    ELSE
      actx.runningSum := INTERVAL '0 0:0:0.0' DAY TO SECOND;
    END IF;
    RETURN ODCIConst.Success;
  END;

MEMBER FUNCTION ODCIAggregateIterate
  ( self  IN OUT SumInterval,
    val   IN     DSINTERVAL_UNCONSTRAINED
  ) RETURN NUMBER IS
  BEGIN
    self.runningSum := self.runningSum + val;
    RETURN ODCIConst.Success;
  END;

MEMBER FUNCTION ODCIAggregateTerminate
  ( self        IN  SumInterval,
    ReturnValue OUT DSINTERVAL_UNCONSTRAINED,
    flags       IN  NUMBER
  ) RETURN NUMBER IS
  BEGIN
    returnValue := self.runningSum;
    RETURN ODCIConst.Success;
  END;

MEMBER FUNCTION ODCIAggregateMerge
  (self IN OUT SumInterval,
   ctx2 IN     SumInterval
  ) RETURN NUMBER IS
  BEGIN
    self.runningSum := self.runningSum + ctx2.runningSum;
    RETURN ODCIConst.Success;
  END;

END;

最后使用此对象类型的函数:

CREATE OR REPLACE FUNCTION sum_interval( x DSINTERVAL_UNCONSTRAINED) 
RETURN DSINTERVAL_UNCONSTRAINED  PARALLEL_ENABLE
AGGREGATE USING SumInterval;

现在您可以使用新的“sum_interval”,如下所示:

with x as (
select to_dsinterval('0 00:56:30') as duration from dual
union
select to_dsinterval('0 02:08:40') as duration from dual
union
select to_dsinterval('0 01:01:00') as duration from dual
)
select sum_interval(duration)
from x;

输出:

SUM_INTERVAL(DURATION)
+00 04:06:10.000000

这也很好,它可以类似于典型的聚合函数使用。例如,我们可以使用组函数:

with x as (
select 'FL' as state, to_dsinterval('0 00:56:30') as duration from dual
union
select 'FL' as state, to_dsinterval('0 02:08:40') as duration from dual
union
select 'GA' as state, to_dsinterval('0 01:01:00') as duration from dual
)
select state, sum_interval(duration)
from x
group by state;

输出:

STATE   SUM_INTERVAL(DURATION)
FL  +00 03:05:10.000000
GA  +00 01:01:00.000000