0

我有一张数据表,其中包含多种货币的收入(我们称之为这张TRANSACTION_TABLE表,列如下:

TRANSACTION_NAME
TRANSACTION_VALUE
CURRENCY

,以及另一个带有汇率 ( EXCHANGE_RATE) 的表,其列如下:

FROM_CURRENCY (e.g. JPY)
TO_CURRENCY (e.g. USD)
EXCHANGE_RATE (x)

该表至少包含转换为美元的每种货币,但并未详尽列出非美元TO_CURRENCY价值的汇率。

我想要实现的是一个查询,它将交易转换为任何货币,即使EXCHANGE_RATE表中没有明确规定,首先将货币转换为美元,然后从美元转换为目标货币。

例如 1000 日元兑英镑:

  1. 查找 JPY 到 USD 的汇率 - 计算 =1000 * EXCHANGE_RATE = 9
  2. 查找英镑兑美元汇率 - 计算 =9 \ EXCHANGE_RATE = 7

目前,我已经为TRANSACTION_TABLEon完成了左连接,EXCHANGE_RATE但我不知道下一步该去哪里。

任何帮助将不胜感激。

到目前为止,我构建的查询(非常基本)如下,我是一个新手 SQL 用户。我首先构建了这个查询以转换为美元,它工作正常(因为我的汇率表包含所有货币到美元的值) - 但在将目标货币设置为英镑时它显然会失败,因为它只会返回空值。

SELECT TRANSACTION_NAME,
             SUM (TRANSACTION_VALUE * EXCHANGE_RATE)
                 AS "REVENUE GBP"
        FROM TRANSACTION_TABLE S
             LEFT JOIN EXCHANGE_RATE C ON S.CURRENCY = C.FROM_CURRENCY AND C.TO_CURRENCY = 'GBP'
ORDER BY TRANSACTION_NAME
4

2 回答 2

3

如果您的EXCHANGE_RATE表格中包含了美元,那么您将不会有超过两个“跳跃”来进行转换。最多,您将转换为美元,然后从美元转换为任何值。鉴于此,我只会为所有可能的情况编写代码,而不是尝试像CONNECT BY.

我认为“所有可能的情况”是:

  • 交易已经使用目标货币
  • 交易以可直接兑换为目标货币的货币进行
  • 交易必须先转换为美元,然后再从美元转换为目标货币。

这是一个可以做到这一点的查询。这些WITH子句只是为了给它一些数据——它们不会成为你解决方案的一部分,因为你有实际的表。

WITH rates ( from_currency, to_currency, exchange_rate ) AS 
( SELECT 'JPY', 'USD', 0.009 FROM DUAL UNION ALL
  SELECT 'GBP', 'USD', 1.31 FROM DUAL UNION ALL
  SELECT 'CNY', 'USD', 0.15 FROM DUAL UNION ALL
  SELECT 'JPY', 'CNY', 0.06 FROM DUAL),
txns ( transaction_name, transaction_value, currency ) AS
 ( SELECT 'txn 1 in JPY', 1000, 'JPY' FROM DUAL UNION ALL
   SELECT 'txn 2 in GBP', 1000, 'GBP' FROM DUAL UNION ALL
   SELECT 'txn 3 IN CNY', 1000, 'CNY' FROM DUAL UNION ALL
   SELECT 'txn 4 IN unknown', 1000, 'XXX' FROM DUAL),
params ( target_currency ) AS 
 ( SELECT 'CNY' FROM DUAL )
SELECT t.transaction_name,
       t.transaction_value base_value,
       t.currency base_currency,
       t.transaction_value * CASE WHEN t.currency = params.target_currency THEN 1
            WHEN r1.from_currency IS NOT NULL THEN r1.exchange_rate
            ELSE r2usd.exchange_rate / r2tar.exchange_rate END converted_value,
        params.target_currency converted_currency
FROM   params CROSS JOIN 
       txns t 
       LEFT JOIN rates r1 ON r1.from_currency = t.currency AND r1.to_currency = params.target_currency
       LEFT JOIN rates r2usd ON r2usd.from_currency = t.currency AND r2usd.to_currency = 'USD'
       LEFT JOIN rates r2tar ON r2tar.from_currency = params.target_currency AND r2tar.to_currency = 'USD'
于 2018-07-02T13:34:57.493 回答
2

我建议采取额外的步骤来扩展您的兑换表,其中使用UDS作为转账货币额外定义的汇率。

此查询添加通过美元计算的新费率。这是一个简单的内连接约束,因此计算是通过“美元”进行的,并且从货币和到货币不同。该WHERE子句限制了已知的组合。

select  er1.FROM_CURRENCY, er2.TO_CURRENCY,  er1.EXCHANGE_RATE * er2.EXCHANGE_RATE EXCHANGE_RATE
from exchange_rates er1
join exchange_rates er2
on er1.TO_CURRENCY = 'USD' and er2.FROM_CURRENCY = 'USD'  and er1.FROM_CURRENCY != er2.TO_CURRENCY
where (er1.FROM_CURRENCY, er2.TO_CURRENCY)
not in (select FROM_CURRENCY,   TO_CURRENCY from exchange_rates)

您可以定义一个物理的新表或视图,甚至仅将其作为UNION ALL原始表的子查询和此查询的结果来执行。

您的最终查询使用此扩展汇率表而不是原始汇率表。

这是我测试过的样本数据

create table exchange_rates
as
select 'GBP' FROM_CURRENCY,  'USD' TO_CURRENCY, 1.31  EXCHANGE_RATE from dual union all
select 'EUR' FROM_CURRENCY,  'USD' TO_CURRENCY, 1.16 EXCHANGE_RATE from dual union all
select 'AUD' FROM_CURRENCY,  'USD' TO_CURRENCY, .73 EXCHANGE_RATE from dual union all
select 'USD' FROM_CURRENCY,  'GBP' TO_CURRENCY, .76  EXCHANGE_RATE from dual union all
select 'USD' FROM_CURRENCY,  'EUR' TO_CURRENCY, .86 EXCHANGE_RATE from dual union all
select 'USD' FROM_CURRENCY,  'AUD' TO_CURRENCY, 1.36 EXCHANGE_RATE from dual union all
select 'GBP' FROM_CURRENCY,  'EUR' TO_CURRENCY, 1.12 EXCHANGE_RATE from dual;
于 2018-07-02T14:58:07.397 回答