Sed - 替换文件的第二行中出现的所有字符串

时间:2017-06-16 23:58:57

标签: sql-server regex bash sed

我有一个用于修改sql文件(test.sql)的bash脚本。

sql文件如下:

select count(*) from (
select 
I.ID,
I.START_DATE, 
I.END_DATE
from MY_TABLE I with (nolock)
where I.START_DATE >= '20170101' and I.START_DATE < '20170201'
) as cnt

我想在sql文件的START_DATE子句中替换所有字符串WHERE 但仅的出现次数。

到目前为止,我已经尝试过这个:

#!/bin/bash
sed 'x; ${/START_DATE/s/START/END;p;x}; 1d' test.sql > new-test.sql

但是,这将返回以下new-test.sql文件:

select count(*) from (
select 
I.ID,
I.START_DATE, 
I.END_DATE
from MY_TABLE I with (nolock)
where I.END_DATE >= '20170101' and I.START_DATE < '20170201'
) as cnt

START_DATE子句中第二次出现的字符串WHERE未被替换。

我该如何修改我的sed表达式才能实现这个目标?

2 个答案:

答案 0 :(得分:0)

试试这样:

where

说明:

  • 仅对包含where的行进行操作(我们仅“解析”与正则表达式START_DATE匹配的行)

  • END_DATE的每次出现替换为g - 注意最后的“全球”标记-i

  • ... "Some text" ... 标志告诉sed“就地”编辑文件(无需重定向输出)。

答案 1 :(得分:0)

如果您使用GNU sed,则可以使用以下命令

sed '/WHERE/I s/START_DATE/END_DATE/g' test.sql > new-test.sql
            ^                        
            |-------------------------- case insensitive match; GNU sed only

首都I告诉sed执行不区分大小写的匹配。在处理不区分大小写的SQL命令时,这会派上用场。

如果您没有GNU sed,那么不区分大小写的匹配会更复杂一些:

sed '/[wW][hH][eE][rR][eE]/ s/START_DATE/END_DATE/g' test.sql > new-test.sql

只需在g命令末尾添加全局s选项,即可使您的解决方案正常运行。

sed 'x; ${/START_DATE/s/START/END/g;p;x}; 1d'

但是,如果文件有尾随换行符,它将会中断。我强烈建议搜索(不区分大小写)WHERE子句并运行相应的替换命令。