将A转换为1,B转换为2 ... Z转换为26,然后将AA转换为27,AB转换为28(在Excel中将列索引转换为列引用)。

26

有没有算法或逻辑可以将A转换为1,B转换为2,...,Z转换为26,然后AA转换为27,AB转换为28等。

换句话说,将Excel中的列索引转换为列引用。

5个回答

38

以下是一个简单的 LINQ 表达式:

static int TextToNumber(this string text) {
    return text
        .Select(c => c - 'A' + 1)
        .Aggregate((sum, next) => sum*26 + next);
}

这个测试

Console.WriteLine(" A -> " + "A".TextToNumber());
Console.WriteLine(" B -> " + "B".TextToNumber());
Console.WriteLine(" Z -> " + "Z".TextToNumber());
Console.WriteLine("AA -> " + "AA".TextToNumber());
Console.WriteLine("AB -> " + "AB".TextToNumber());

将产生以下输出:

 A -> 1
 B -> 2
 Z -> 26
AA -> 27
AB -> 28

更新: 这里是同样的代码,但目标是.NET 2.0:

static int TextToNumber(string text) {
    int sum = 0;
    foreach (char c in text) {
        sum = sum*26 + c - 'A' + 1;
    }
    return sum;
}

2
如果使用一个折叠函数,你可以在不将字符串转换为字符数组的情况下进行操作,因为字符串的行为就像一个 IEnumerable<char>。例如:s.Select(c => c - 'A' + 1).Aggregate((sum, next) => sum * 26 + next),这样做也是+1的。 - cfern
1
很好的解决方案,但是在Visual Studio 2005中不支持Linq。 - Thunder
@Merritt 我想应该包含 System.Linq 命名空间就足够了。 - Thomas Freudenberg
我的错:由于某种原因,它实际上被Intellisense隐藏了... https://dev59.com/v3RC5IYBdhLWcg3wS_AF - Merritt
很好的回答,那么NumberToText方法在两种语言中怎么样? - Pierre
显示剩余2条评论

33

看看这些

/// <summary>
/// 1 -> A<br/>
/// 2 -> B<br/>
/// 3 -> C<br/>
/// ...
/// </summary>
/// <param name="column"></param>
/// <returns></returns>
public static string ExcelColumnFromNumber(int column)
{
    string columnString = "";
    decimal columnNumber = column;
    while (columnNumber > 0)
    {
        decimal currentLetterNumber = (columnNumber - 1) % 26;
        char currentLetter = (char)(currentLetterNumber + 65);
        columnString = currentLetter + columnString;
        columnNumber = (columnNumber - (currentLetterNumber + 1)) / 26;
    }
    return columnString;
}

/// <summary>
/// A -> 1<br/>
/// B -> 2<br/>
/// C -> 3<br/>
/// ...
/// </summary>
/// <param name="column"></param>
/// <returns></returns>
public static int NumberFromExcelColumn(string column)
{
    int retVal = 0;
    string col = column.ToUpper();
    for (int iChar = col.Length - 1; iChar >= 0; iChar--)
    {
        char colPiece = col[iChar];
        int colNum = colPiece - 64;
        retVal = retVal + colNum * (int)Math.Pow(26, col.Length - (iChar + 1));
    }
    return retVal;
}

1
你使用decimal作为columnNumber和currentLetterNumber的类型,有什么原因吗? - Matthias
1
不,这只是我最初的实现方式。你可以将其更改为整数。快速测试过,看起来没问题。 - Adriaan Stander

3

如果您愿意在客户端上完成操作,则以下是JavaScript代码:

<script type="text/javascript" lang="javascript">
function Alphabet2Numeric(mystr) {
    mystr = mystr.toUpperCase(); //Hence the ASCII code 64 down there
    var sum = 0;
    for (var i = 0; i < mystr.length; i++) {
        sum = sum * 26 + mystr.charCodeAt(i) - 64; //returns 1 for 'a' and 2 for 'b' so on and so forth.
    }
    return sum;
}
</script>

0

同样的问题,不同的语言:PHP。

function charToInt($char)
{
   $array = array_flip(range(a, z));
   return $array[$char] + 1;
}

echo charToInt('c');

outputs: 3

0

从A到ZZ:((字符数-1)*26) + 最后一个字符的ASCII码%65 + 第一个字符的ASCII码%65 * (26^(字符长度-1)) -1 [仅在结果为正数时加上] 输入图像描述


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接