C# 关于位运算的学习笔记
一、理解什么是位运算
程序中的所有内容在计算机内存中都是以二进制的形式储存的(即:0或1),简单来说位运算就是直接对在内存中的二进制数的每位进行运算操作。
二、学习前先了解一下有哪些运算,运算符都怎么使用的,如下图所示。
对于上面的运算符做个基础的认识,接下来我们详细看一下每个运算符到底该怎么使用呢?
三、在讲位运算之前先讲一下进制转换,方便下文讲解位运算。
2进制、8进制、16进制、32进制、64进制等转换成10进制计算方法我得出一个公式:(^表示次方,如:2^2,即2的2次方,8^5即8的5次方)
每位数字转换成10进制时=进制数^(次方)数字索引位(从0开始计算)*数字
计算示例:(注意黑粗体字)
2进制数:10100=2^0*0+2^1*0+2^2*1+2^3*0+2^4*1=0+0+4+0+16=20
8进制数:24=8^0*4+8^1*2=4+16=20
16进制数:0x14(注意0x是用来表示16进制数的意思,不是数字本身的内容)=16^0*4+16^1*1=4+16=20
至于各进制之间的转换,比如:2进制转换成16进制,如果想自己手算,一般都是先转成10进制,然后将数字进行与进制数相除,直到得出余数小于或等于进制数(或0),当然作为程序员的我们,应该使用现有的方法,如下:
Convert.ToString(数字,进制数)
如:Convert.ToString(10,2)=01010,Convert.ToString(10,8)=12 ,Convert.ToString(13,16)=0x0d
int i10 = 66; int i16 = 0x5A; Console.WriteLine("示例一:"); Console.Write("10进制【66】转成2、8、16进制结果:{0}、{1}、{2}\n", Convert.ToString(i10, 2), Convert.ToString(i10, 8), Convert.ToString(i10, 16)); Console.Write("16进制【0x5A】转成2、8、10进制结果:{0}、{1}、{2}\n", Convert.ToString(i16, 2), Convert.ToString(i16, 8), Convert.ToString(i16, 10)); Console.ReadLine();
输出如下图:
在这里我要另外讲一下,使用上方Convert.ToString(数字,进制数)转换的时候,转二进制时,左边为0时,0会自动去掉,但是有的时候我们又不希望去掉,因此我想到一种方法是补零。
使用 public String PadLeft(int totalWidth, char paddingChar);进行补零,因为我需要的是得到一个八位的二进制数,所以是PadLeft(8, '0')
padleft 返回一个新字符串,该字符串通过在此实例中的字符左侧填充指定的 Unicode 字符来达到指定的总长度,从而使这些字符右对齐
/// <summary> /// 将十进制字符串转二进制 /// </summary> /// <param name="hexString">返回字符串</param> /// <returns></returns> public static string HexStringByStr(string hexString) { string result = string.Empty; int d = Convert.ToInt32(hexString); //十进制转二进制字符串 result = Convert.ToString(d, 2).PadLeft(8, '0'); return result; }
四、接下来看一下位运算,看一下每一个运算符是怎么使用的
#region 位运算与运算 int a = 11; //11的二进制是0000 1011 int b = 13; //13的二进制是0000 1101 Console.WriteLine("a 二进制:"+Convert.ToString(a, 2).PadLeft(8,'0')+ " b 二进制:" + Convert.ToString(b, 2).PadLeft(8,'0')); Console.WriteLine("a & b 二进制输出:"+Convert.ToString(a & b,2).PadLeft(8,'0')); //11 & 13的结果就是0000 1001 即结果是9 Console.Read(); #endregion
结果输出:
#region 位运算或运算 int a = 16; int b = 22; Console.WriteLine("a 二进制:" + Convert.ToString(a, 2).PadLeft(8, '0') + " b 二进制:" + Convert.ToString(b, 2).PadLeft(8, '0')); Console.WriteLine("a | b 二进制输出:" + Convert.ToString(a | b, 2).PadLeft(8, '0')); Console.Read(); #endregion
结果输出:
#region 位运算异或运算 int a = 16; int b = 22; Console.WriteLine("a 二进制:" + Convert.ToString(a, 2).PadLeft(8, '0') + " b 二进制:" + Convert.ToString(b, 2).PadLeft(8, '0')); Console.WriteLine("a ^ b 二进制输出:" + Convert.ToString(a ^ b, 2).PadLeft(8, '0')); Console.Read(); #endregion
结果输出:
#region 位运算左移运算 int a = 16; int b = 22; Console.WriteLine("a 二进制:" + Convert.ToString(a, 2).PadLeft(8, '0') + " b 二进制:" + Convert.ToString(b, 2).PadLeft(8, '0')); Console.WriteLine("a << 2 二进制输出:" + Convert.ToString(a << 2, 2).PadLeft(8, '0')); Console.WriteLine("b << 2 二进制输出:" + Convert.ToString(b << 2, 2).PadLeft(8, '0')); Console.Read(); #endregion
#region 位运算右移运算 int a = 16; int b = 22; Console.WriteLine("a 二进制:" + Convert.ToString(a, 2).PadLeft(8, '0') + " b 二进制:" + Convert.ToString(b, 2).PadLeft(8, '0')); Console.WriteLine("a >> 2 二进制输出:" + Convert.ToString(a >> 2, 2).PadLeft(8, '0')); Console.WriteLine("b >> 2 二进制输出:" + Convert.ToString(b >> 2, 2).PadLeft(8, '0')); Console.Read(); #endregion
输出结果:
值高八位、低八位
sampleData[4] = (byte)((Int16)(Math.Round(head , 0)) >> 8); //航向角的高8位
sampleData[5] = (byte)((Int16)(Math.Round(head , 0)) & 0xff); //航向角的低八位
将byte转16进制字符串
/// <summary> /// 将byte转16进制字符串 /// </summary> /// <param name="byt"></param> /// <param name="startIndex"></param> /// <param name="lenth"></param> /// <param name="sep"></param> /// <returns></returns> public static string getHexStrFromByte(byte[] byt, int startIndex, int lenth,string sep="") { StringBuilder temp = new StringBuilder(lenth*(2+sep.Length)); for (int i = startIndex; i < startIndex + lenth; i++) { if (i != startIndex) temp.Append(sep); temp.Append(byt[i].ToString("X2")); } return temp.ToString(); }
Console.WriteLine(getHexStrFromByte(sampleData, 0, sampleData.Length, " "));
ok,今天关于位运算的讲解就到这里了,有疑问的欢迎指正。