2

:表名

create table table_name
(
given_dates timestamp,
set_name varchar
);

插入记录

insert into table_name values('2001-01-01'),('2001-01-05'),('2001-01-10'),
                 ('2001-01-15'),('2001-01-20'),('2001-01-25'),
                 ('2001-02-01'),('2001-02-05'),('2001-02-10'),
                 ('2001-02-15');

现在我想更新某些日期的 set_name 。

例如

我想像这样更新表:

given_dates    set_name 
----------------------
2001-01-01      s1
2001-01-05      s1
2001-01-10      s2
2001-01-15      s2
2001-01-20
2001-01-25
2001-02-01
2001-02-05
2001-02-10
2001-02-15

注意given_datesandset_name是传递参数,因为它们是动态的。如上图我可以通过2套,s1,s2也可以根据要求通过4套。

所以我需要动态 case 语句来更新set_name.

给定两个参数

declare p_dates varchar := '2001-01-01to2001-01-05,2001-01-10to2001-01-15';

declare p_sets varchar := 's1,s2';

那么我可以通过使用以下静态脚本来做到这一点:

静态更新声明

update table_name
SET set_name = 
CASE  
when given_dates between '2001-01-01' and '2001-01-05' then 's1'
when given_dates between '2001-01-10' and '2001-01-15' then 's2'
else '' 
end;

上面的 update 语句完成了工作,但是是静态的。

就像更新表的相同方式一样,我只想准备 case 语句,它应该是动态的,可以根据参数的(p_dates,p_sets)变化而变化。

问题

  1. 如何拆分给定的日期p_dates?(我to在两个日期之间有关键字。)
  2. 如何拆分给定的集合p_sets?(我在两个 set_name 之间有 ',' 逗号。)
  3. p_dates拆分and后如何准备动态case语句p_sets

这个问题与使用 SQL Server 2008 R2 的动态案例语句有关,这与 Microsoft SQL Server 相同。

4

3 回答 3

3

清洁设置:

CREATE TABLE tbl (
  given_date date
, set_name varchar
);

使用单数术语作为单个值的列名。
数据类型显然是date而不是timestamp.

要将文本参数转换为有用的表格:

SELECT unnest(string_to_array('2001-01-01to2001-01-05,2001-01-10to2001-01-15', ',')) AS date_range
     , unnest(string_to_array('s1,s2', ',')) AS set_name;

“Parallel unnest”很方便,但有一些警告。Postgres 9.4添加了一个干净的解决方案,Postgres 10最终清理了这个行为。见下文。

动态执行

准备好的声明

准备好的语句只对创建会话可见并随之消亡。根据文档:

准备好的语句仅在当前数据库会话期间持续。

PREPARE 每个会话一次

PREPARE upd_tbl AS
UPDATE tbl t
SET    set_name = s.set_name
FROM  (
   SELECT unnest(string_to_array($1, ',')) AS date_range
        , unnest(string_to_array($2, ',')) AS set_name
   ) s
WHERE t.given_date BETWEEN split_part(date_range, 'to', 1)::date
                       AND split_part(date_range, 'to', 2)::date;

或者使用客户提供的工具来准备声明。
使用任意参数执行 n 次:

EXECUTE upd_tbl('2001-01-01to2001-01-05,2001-01-10to2001-01-15', 's1,s4');

服务器端功能

函数对所有会话都是持久的和可见的。

CREATE FUNCTION 一次

CREATE OR REPLACE FUNCTION f_upd_tbl(_date_ranges text, _names text)
  RETURNS void AS
$func$
UPDATE tbl t
SET    set_name = s.set_name
FROM  (
   SELECT unnest(string_to_array($1, ',')) AS date_range
        , unnest(string_to_array($2, ',')) AS set_name
   ) s
WHERE  t.given_date BETWEEN split_part(date_range, 'to', 1)::date
                        AND split_part(date_range, 'to', 2)::date
$func$  LANGUAGE sql;

调用n次:

SELECT f_upd_tbl('2001-01-01to2001-01-05,2001-01-20to2001-01-25', 's2,s5');

SQL小提琴

卓越的设计

使用数组参数(仍然可以作为字符串字面量提供)、daterange类型(均为 pg 9.3)和新的并行unnest()(pg 9.4)。

CREATE OR REPLACE FUNCTION f_upd_tbl(_dr daterange[], _n text[])
  RETURNS void AS
$func$
UPDATE tbl t
SET    set_name = s.set_name
FROM   unnest($1, $2) s(date_range, set_name)
WHERE  t.given_date <@ s.date_range
$func$  LANGUAGE sql;

<@是“元素包含在”运算符中。

称呼:

SELECT f_upd_tbl('{"[2001-01-01,2001-01-05]"
                  ,"[2001-01-20,2001-01-25]"}', '{s2,s5}');

细节:

于 2015-01-29T17:09:04.283 回答
0

String_to_array

declare p_dates varchar[] := string_to_array('2001-01-01,2001-01-05,
2001-01-10,2001-01-15*2001-01-01,2001-01-05,2001-01-10,2001-01-15','*');
declare p_sets varchar[]  := string_to_array('s1,s2',',');
declare p_length integer=0;
declare p_str  varchar[];
declare i integer;
select array_length(p_dates ,1) into p_count;

for i in 1..p_count loop

  p_str  := string_to_array( p_dates[i],',')   

  execute 'update table_name
  SET set_name = 
  CASE  
  when given_dates between'''|| p_str  [1] ||''' and '''|| p_str  [2] 
  ||''' then ''' || p_sets[1] ||'''
  when given_dates between '''|| p_str  [3] ||''' and '''
  || p_str  [4] ||''' then ''' || p_sets[2] ||'''
  else '''' 
  end';
end loop;
于 2015-01-19T07:04:41.233 回答
0

现在我们可以使用datemultirange.

create or replace function f_upd_tbl_multirange(_dr datemultirange , _n text[])
    returns void as
$func$
UPDATE tbl t
SET    set_name = s.set_name
FROM   unnest($1,$2) s(date_range,set_name)
WHERE  t.given_date <@ s.date_range
$func$ language sql;

运行。

SELECT f_upd_tbl_multirange(
    '{[''2022-01-01'',''2022-01-05''],[''2022-02-06'',''2022-02-25'']}', '{s2,s5}');
于 2022-02-21T13:37:38.950 回答