为什么这个函数用位运算符转换成这种方式?

时间:2013-10-11 13:35:01

标签: c# bitwise-operators

我正在为一个设备编写一个控件类,直到我需要将ARGB颜色转换为其格式。起初,我写了这个函数(有效):

private static int convertFormat(System.Drawing.Color c)
{
    String all;

    int a = (int)((float)c.A / 31.875);

    if (a == 0)
        a = 1;

    all = a.ToString() + c.B.ToString("X").PadLeft(2, '0') + c.G.ToString("X").PadLeft(2, '0') + c.R.ToString("X").PadLeft(2, '0');

    int num = int.Parse(all, System.Globalization.NumberStyles.AllowHexSpecifier);

    return num;
}

但它太丑了我想写一个更优雅的解决方案。 所以我做了一些以获得正确的值,尝试0到50之间的所有组合。 它奏效了,我最终得到了这个:

private static int convertFormatShifting(System.Drawing.Color c)
{
    int alpha = (int)Math.Round((float)c.A / 31.875);

    int a = Math.Max(alpha,1);

    return (a << 24) | (c.B << 48) | (c.G << 40) | (c.R << 32);
}

哪个有效!

但是现在,我希望有人能够解释为什么这些是正确的转移价值。

2 个答案:

答案 0 :(得分:9)

最不容易混淆的移位值应如下:

return (a << 24) | (c.B << 16) | (c.G << 8) | (c.R << 0);
// Of course there's no need to shift by zero  ^^^^^^^^

您的值工作的原因是移位运算符mod是右侧,操作数的位长度位于左侧。换句话说,以下所有变化都是相同的:

c.G << 8
c.G << 40
c.G << 72
c.G << 104
...

答案 1 :(得分:0)

根据Wikipedia,RGBA格式遵循以下惯例:

semantics alpha | red | green | blue
bits      31-24 |23-16|15 -  8|7 - 0

这与你的班次类似。你正在向左移动位并用一点点或者连接它们。然后看看你期望返回的函数是什么(int中颜色的顺序是什么。)