C#定位*在由管道分隔的字符串中的位置

问题描述

当第一个位置可能根本没有*时,我必须找出*的位置第二名第三名。

位置由管道分隔|
因此

否*通配符为

`ABC|DEF|GHI` 

但是,虽然可能是1种情况,但其他3种是

    string testPosition1 = "*|DEF|GHI";
    string testPosition2 = "ABC|*|GHI";
    string testPosition3 = "ABC|DEF|*";

我收集的数据比应使用IndexOf的数据多,但似乎应该合并| (管道)知道位置(不只是长度,因为在3个位置中的每个位置值都可能长或短。所以我只想结束知道*是在第一,第二还是第三位置(或根本不知道) )

因此,我正在这样做,但是我不会知道它是在第一个还是第二个管道之前

if(testPosition1.IndexOf("*") > 0) 
{
    // Look for pipes?  
}

解决方法

有很多方法可以解决这个问题。最容易读懂的可能实际上是用困难的方式(例如,扫描字符串以找到第一个'*'个字符,并跟踪您在此过程中看到了多少个'|'个字符)。

也就是说,这可能是类似的可读性和简洁性:

int wildcardPosition = Array.IndexOf(testPosition1.Split('|'),"*");

如果找不到则返回-1,否则返回{0}开头的'|'分隔字符串段中包含通配符字符串的索引。

这仅在通配符正好是一个字符字符串"*"的情况下有效。如果需要在此基础上支持其他变体,则仍然需要拆分字符串,但是可以遍历数组以查找所需的条件。

,

您可以尝试使用linq在管道字符处分割字符串,然后获取仅包含*的元素的索引。

+---+--------+
| id|myvalues|
+---+--------+
|  1|    AAAA|
|  2|    BBBB|
|  2|    CCCC|
+---+--------+

因此,第一行开始在管道上拆分字符串,以创建字符串数组。该序列将传递到Select扩展名,该扩展名枚举通过字符串文本(k)和索引(i)的序列。使用这两个参数,我们可以构建具有两个属性(文本和索引)的匿名对象序列。 FirstOrDefault从此序列中提取文本等于*的对象,然后我们可以打印该对象的属性索引。

,

其他答案很好(可能更好),但是这是另一种方法,老式的 for循环和try-get模式

public bool TryGetStar(string input,out int index)
{
   var split = input.Split('|');
   for (index = 0; index < split.Length; index++)
      if (split[index] == "*")
         return true;
   return false;
}

或者如果您要处理较大的 strings 并尝试保存分配。您可以完全删除Split并使用单个解析 O(n)

public bool TryGetStar(string input,out int index)
{
   index = 0;
   for (var i = 0; i < input.Length; i++)
      if (input[i] == '|') index++;
      else if (input[i] == '*') return true;
   return false;
}

注意:如果考虑性能,则还可以使用unsafe pointers Span<Char>,它们会带来少量效率。

,

尝试DotNETFiddle

testPosition.IndexOf("*") - testPosition.Replace("|","").IndexOf("*")

找到通配符(“ *”)的索引,并查看删除通配符(“ |”)字符后其移动的距离。结果是从零开始的索引。

,

从问题中可以看到以下代码段:

.

如果您现在位于if(testPosition1.IndexOf("*") > 0) { } 语句中,请确保星号存在。

从那时起,一种有效的解决方案可能是检查前两个字符和后两个字符。

if

这假定不能存在一个以上的if (testPosition1.IndexOf("*") > 0) { if (testPosition1[0] == '*' && testPosition[1] == '|') { // First position. } else if (testPosition1[testPosition.Length - 1] == '*' && testPosition1[testPosition.Length - 2] == '|') { // Third (last) position. } else { // Second position. } } ,并且还假定如果存在一个*,则只能被管道包围。例如,假设*之类的输入无效。

如果要删除此假设,可以对字符串进行一次遍历,并跟踪必要的信息。