0

如果我需要检查字符串是否具有某些值,正则表达式会是什么样子

例如,它应该从以下字符串中提取 Duration 量(时间跨度),但前提是源字符串包含 allDurationstartbitrate字符串

Duration: 00:04:19.39, start: 0.157967, bitrate: 15636 kb/s

像这样的行应该被忽略:

Duration: 00:04:19.39, bitrate: 15636 kb/s
start: 0.157967, bitrate: 15636 kb/s
Duration: 00:04:19.39, start: 0.157967
4

3 回答 3

2

由于您正在进行数据提取,我将使用这个简单的正则表达式:

^(?=.*start:)(?=.*bitrate:).*Duration: ([\d.:]+)

时间戳可以在第一个捕获组中找到。

数据似乎是从程序日志生成的,所以我假设间距是规则的。我的正则表达式将忽略源字符串中的start,bitrate和的顺序。Duration如果您想要不区分大小写的匹配,请打开该标志。

忽略排序将使正则表达式在长字符串上变慢。我们拥有的假设越多(尤其是关于排序的假设),正则表达式就越好。

解释

^
(?=.*start:)
(?=.*bitrate:)
.*Duration: ([\d.:]+)

^锚定字符串的开头。我出于性能原因添加了这个,因为如果正则表达式引擎已经详尽地回溯了所有案例并且没有找到匹配项,则无需检查匹配项。

(?=.*start:)零宽度正前瞻。它将尝试.*start:从字符串中的当前位置匹配,如果找到则从它离开的位置继续匹配,如果没有找到则停止。它被称为零宽度,因为它实际上并不消耗字符串,而不是正则表达式的这一部分.*Duration: ([\d.:]+)

(?=.*bitrate:),和上面一样,检查bitrate:字符串中是否有前面。

.*Duration: ([\d.:]+)与实际时长相符。我不关心格式,因为我假设你得到的任何东西都是正确的,所以我只抓住最长的数字序列\d,.:.

当字符串中有多个匹配项时,使用字符的概念很重要。有时,您想在决定操作之前检查字符串是否包含某个序列。这样的检查不应该消耗 character,因为当您还没有处理当前位置的文本时,您不应该处理前面的文本。如果您使用文本,那么您可能会丢失文本中从当前位置到字符串中前面的文本的一些匹配项。

于 2013-03-01T17:40:23.413 回答
2

您想要的是一个匹配整个字符串的表达式,并为您要提取的部分捕获组,例如:

@"Duration: (\d{2}:\d{2}:\d{2}.\d{2}), start: \d+(.\d+)?, bitrate: \d+ kb/s"

包围您的()匹配组(Duration您要读取的值)。

于 2013-03-01T17:40:23.683 回答
0

如果格式和顺序始终相同,则单纯形正则表达式可能是这样的:

Duration: (.*), start: .*, bitrate: .*
于 2013-03-01T17:47:56.700 回答