有没有办法找到与DateTime格式字符串匹配的第一个字符串

本文关键字:字符串 字符 串匹配 第一个 格式 DateTime 有没有 | 更新日期: 2023-09-27 18:21:19

给定一个日期-时间格式字符串,是否有标准的方法来找到与该格式匹配的第一个子字符串?

例如,给定。。。

d-MMM-yy H:mm:ss

还有一些文字。。。

"blah 1 2 3 7-Jul-13 6:15:00 4 5 6 blah"

我希望它能退回

"7-Jul-13 6:15:00"

现在我可以通过解析找到这个字符串,但我想知道是否有库支持这样做?

有没有办法找到与DateTime格式字符串匹配的第一个字符串

这可能不是最有效的,但这似乎是一个有趣的问题,所以我想试试这个方法。

它使用DateTime格式字符串,并通过用.替换任何字母,用''s替换空白来生成Regex字符串模式。然后,它根据该模式创建一个Regex对象,并尝试在输入句子中找到第一个匹配项。

如果存在匹配,则将其传递到DateTime.TryParseExact调用中。我确信可以进行改进,但这可能有助于提供一个不需要硬编码Regex或输入句子格式的技术的总体概念。

string inputSentence = "blah 1 2 3 7-Jul-13 6:15:00 4 5 6 blah";
string dtformat = "d-MMM-yy H:mm:ss";
//convert dtformat into regex pattern
StringBuilder sb = new StringBuilder();
foreach (char c in dtformat)
{
    if (Char.IsLetter(c))
    {
       if (char.ToUpperInvariant(c) == 'D' || char.ToUpperInvariant(c) == 'H' || char.ToUpperInvariant(c) == 'S')            
          sb.Append(".{1,2}");
       else
          sb.Append(".");
    }
    else if(Char.IsWhiteSpace(c))        
       sb.Append("''s");
    else
       sb.Append(c);
}

string dtPattern = sb.ToString();
Regex dtrx = new Regex(dtPattern);
//get the match using the regex pattern
var dtMatch = dtrx.Match(inputSentence);
if(dtMatch != null)
{
    string firstString = dtMatch.Value.Trim();
    //try and parse the datetime from the string
    DateTime firstMatch;
    if (DateTime.TryParseExact(dstr, dtformat, null, DateTimeStyles.None, out firstMatch))
    {
       Console.WriteLine("Parsed");
    }
    else
    {
       Console.WriteLine("Could not parse");
    }
}

也许是这样的:

通过解析文本中的每个单词找到格式字符串的每个部分,并将2个单词组合起来创建DateTime

string test = "blah 1 2 3 7-Jul-13 6:15:00  4 5 6 blah";
int formatPart = 0;
bool dateFound = false;
string format = "d-MMM-yy H:mm:ss";
DateTime myDateTime = DateTime.MinValue;
foreach (var item in test.Split(' '))
{
    DateTime dummy;
    if (DateTime.TryParseExact(item, format.Split(' ')[formatPart], null, DateTimeStyles.NoCurrentDateDefault, out dummy))
    {
        if (myDateTime == DateTime.MinValue)
        {
            formatPart++;
            myDateTime = dummy;
            dateFound = myDateTime.Date != DateTime.MinValue.Date;
            continue;
        }
        // If date was found first, add time, else add date
        myDateTime = dateFound
         ? myDateTime.Add(new TimeSpan(dummy.Hour, dummy.Minute, dummy.Second))
         : dummy.Add(new TimeSpan(myDateTime.Hour, myDateTime.Minute, myDateTime.Second));
        break;
    }
}

测试:

Input: "blah 1 2 3 7-Jul-13 6:15:00  4 5 6 blah"
Format: "d-MMM-yy H:mm:ss"
Input: "blah 1 2 3 6:15:00 7-Jul-13 4 5 6 blah"
Format: "H:mm:ss d-MMM-yy"
Input: "blah 1 2 3 6:15:00 7-7-2013 4 5 6 blah"
Format: "H:mm:ss d-M-yyyy"
Input: "blah 1 2 3 07-07-2013 6:15:00  4 5 6 blah"
Format: "dd-MM-yyyy H:mm:ss" 

您可以尝试NodaTime:

var input = "blah 1 2 3 7-Jul-13 6:15:00 4 5 6 blah";
var pattern = "d-MMM-yy H:mm:ss";
var nodaPattern = NodaTime.Text
    .LocalDateTimePattern
    .Create(pattern, System.Globalization.CultureInfo.CurrentUICulture);
for (int i = 0; i < input.Length - pattern.Length; i++)
{
    var result = nodaPattern.Parse(input.Substring(i, pattern.Length));
    if (result.Success)
    {
        Console.WriteLine(result.Value);
        break;
    }
}

我不知道除了DateTime.TryParse(或者RegEx)之外还有什么可以在.NET.中执行此操作的

我会设置一个流标记器,只将候选标记对传递到DateTime.TryParse中(基于最小字符串长度的某种组合,可能是子字符串检查标记0上的一对破折号和标记1上的一对联冒号,等等。确切的检查取决于您支持的日期/时间格式)

您可以尝试一个正则表达式,如:

^[0-9]+-[a-z A-z]+-[0-9]+'s[0-9]+:[0-9]+:[0-9]+

你可以试试这个:

string original = "blah 1 2 3 7-jul-13 6:15:00 4 5 6 blah";
Match mc = Regex.Match(original, @"'s*(?<date>'d+[-/][A-Z]{3}[-/]'d+)'s*([01]?[0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9]", RegexOptions.IgnoreCase);
if (mc.Success)
{
    string datetime = mc.Groups[0].Value;
}