1

嘿,我想编写一个像 XNA Color 结构或 SurfaceFormat.Bgra4444 结构这样的结构,它在 8 位字节中保存 2 个半字节。

这是我目前所拥有的......

/// <summary>
/// Packed byte containing two 4bit values
/// </summary>
public struct Nibble2 : IEquatable<Nibble2>
{
    private byte packedValue;

    public byte X
    {
        get { return 0; }
        set { }
    }

    public byte Y
    {
        get { return 0; }
        set { }
    }

    /// <summary>
    /// Creates an instance of this object.
    /// </summary>
    /// <param name="x">Initial value for the x component.</param>
    /// <param name="y">Initial value for the y component.</param>
    public Nibble2(float x, float y)
    {
        packedValue = 0;
    }

    /// <summary>
    /// Creates an instance of this object.
    /// </summary>
    /// <param name="vector">Input value for both components.</param>
    public Nibble2(Vector2 vector)
    {
        packedValue = 0;
    }

    public static bool operator ==(Nibble2 a, Nibble2 b) { return a.packedValue == b.packedValue; }
    public static bool operator !=(Nibble2 a, Nibble2 b) { return a.packedValue != b.packedValue; }

    public override string ToString()
    { return packedValue.ToString("X : " + X + " Y : " + Y, CultureInfo.InvariantCulture); }

    public override int GetHashCode()
    {return packedValue;}

    public override bool Equals(object obj)
    {
        if (obj is Nibble2)
            return Equals((Nibble2)obj);
        return false;
    }

    public bool Equals(Nibble2 other)
    {return packedValue == other.packedValue;}
}

如您所见,我的属性和构造函数没有被暗示。因为这是我遇到麻烦的部分。

感谢您的任何帮助,您可以提供。

4

3 回答 3

4

基本上,您只需要记住什么是高半字节和低半字节。只需用二进制 1111(十进制 15)屏蔽即可获得低半字节。byte可以通过右移 4来获得高半字节(因为它是无符号的)。其余的只是位数学。

// assume x is the low nibble
public byte X
{
    get { return (byte)(packedValue & 15); }
    set { packedValue = (packedValue & 240) | (value & 15); }
}

// assume y is the high nibble
public byte Y
{
    get { return (byte) (packedValue >> 4); }
    set { packedValue = (value << 4) | (packedValue & 15); }
}

但是,我无法帮助您:

public Nibble2(float x, float y)
{
    packedValue = 0;
}

因为那是 64 位,并且您希望将其放入 8 位。您需要具体地说明您想要对这些值做什么。

于 2012-07-23T07:23:24.140 回答
1

考虑以下字节值:

10101100 

要读取一个字节的高位,您应该将字节向右移动:

10101100 (original)
01010110 (shifted one bit)
00101011  
00010101 
00001010 (shifted four bits)

您可以使用移位字节return (byte)(packedValue >> 4);

要仅读取低位,您只需使用 AND 操作消除高位:

10101100 
00001111 AND
--------
00001100

您可以通过使用对值执行此 AND 操作return (byte)(packedValue & 0xf);

可以通过清除目标半字节然后简单地添加输入值来设置这些值(如果设置高半字节,则向左移动):

packedValue = (byte)((packedValue & 0xf0) + (lowNibble & 0xf));
packedValue = (byte)((packedValue & 0xf) + (highNibble << 4));
于 2012-07-23T07:10:45.043 回答
1

如果您将输入值与这样填充的字节相加00001111,即 dec 中的 15。您保留要保存的部分。然后,您必须将 Nibble2 的左侧部分左移 4 个字节以存储在 packedValue 字节中。

private byte x = 0;
private byte y = 0;


public byte X
{
    get { return x; }
    set { x = value}
}

public byte Y
{
    get { return y; }
    set { y = value }
}

private byte packedValue 
{
    get { return (x & 15 << 4) | (y & 15); }
}
于 2012-07-23T07:10:49.287 回答