str
swift自带api
let a: UInt8 = 9
let str = String(a, radix: 2)
print("str: \(str)")
// 输出:str: 1001
// 它输出的字符串只包括有效二进制位
二进制——>八进制
00001001——>11
转换原理:(因为是小端模式,二进制的有效位是从右到左的)二进制每三位一组(不够三位补0),分别计算对应的十进制值,计算结果的组合就是一个八进制数。
1、分组:000(不足三位补0),001,001
2、每组分别计算十进制值:0*2^0 + 0*2^1 + 0*2^2 = 0,1*2^0 + 0*2^1 + 0*2^2 = 1,1*2^0 + 0*2^1 + 0*2^2 = 1
3、每组的结果从左到右组合:0,1,1,最终的八进制数字为:11
swift代码实现:
func octal(_ v: String) -> String {
guard v.count > 0 else { return "" }
var res = ""
var idx = 0
var sum = 0
for i in v.indices.reversed() {
if let a = Int(String(v[i])) {
sum += a*Int(pow(2, Double(idx)))
if idx >= 2 {
res.insert(Character("\(sum)"), at: res.startIndex)
idx = 0
sum = 0
}else {
idx += 1
return res
print("----o: \(octal("00001011"))")
// 输出:----o: 13
swift自带api:
let a = 0b0000_1011
let str = String(a, radix: 8)
print("str: \(str)")
// 输出:str: 13
八进制——>二进制
11——>00001001
原理:(利用二进制 转 八进制 反推导可知,对八进制的每位分别进行转3位的二进制数,然后将每位的结果拼接)1->001,1->001,最后拼接得到二进制:001001
swift实现代码:
func octalToBinary(_ v: String) -> String {
guard v.count > 0 else { return "" }
var res = ""
for i in v.indices {
if let a = Int(String(v[i])) {
var n = a
var s = ""
for _ in 0..<3 {
s.insert(Character("\(n % 2)"), at: s.startIndex)
n /= 2
res.append(s)
return res
print("----b: \(octalToBinary("11"))") // 八进制:11转二进制
// 输出:----b: 001001
swift自带api
let a = 0o11
let str = String(a, radix: 2)
print("str: \(str)")
// 输出:str: 1001
二进制——>十六进制
00101011——>2b
原理:二进制每四位一组(不够四位补0),分别计算对应的十进制值,计算结果的组合就是一个八进制数
1、分组:0010,1001
2、分别计算每组的十进制值:0*2^0 + 1*2^1 + 0*2^2 + 0*2^3 = 2,1*2^0 + 0*2^1 + 0*2^2 + 1*2^3 = 11(因为在16进制的表示中是从0到f的,因此11对应的是b)
3、组合结果:2b
swift代码实现:
func hex(_ v: String) -> String {
guard v.count > 0 else { return "" }
var res = ""
var idx = 0, sum = 0
for i in v.indices.reversed() {
if let a = Int(String(v[i])) {
sum += (a*Int(pow(2, Double(idx))))
if idx >= 3 {
res.insert(Character(String(format: "%x", sum)), at: res.startIndex)
idx = 0
sum = 0
}else {
idx += 1
if sum > 0 {
res.insert(Character("\(sum)"), at: res.startIndex)
return res
print("----hex: \(hex("101011"))") // 二进制:101011转十六进制
// 输出:----hex: 2b
swift自带api:
let a = 0b0010_1011
let str = String(a, radix: 16)
print("str: \(str)")
// 输出:str: 2b
十六进制——>二进制
2b——>00101011
原理:(根据二进制 转 十六进制 可推导出,对十六进制的每位分别进行转4位的二进制数,然后将每位的结果拼接)2——>0010,b——>1011
swift实现代码:
func hexToBinary(_ v: String) -> String {
guard v.count > 0 else { return "" }
var res = ""
let map = ["1": 1, "2": 2, "3": 3, "4": 4, "5": 5, "6": 6, "7": 7, "8": 8, "9": 9, "a": 10, "b": 11, "c": 12, "d": 13, "e": 14, "f": 15]
for i in v.indices {
if let a = map[String(v[i])] {
var n = a
var s = ""
for _ in 0..<4 {
s.insert(Character("\(n % 2)"), at: s.startIndex)
n /= 2
res.append(s)
return res
print("----b: \(hexToBinary("2b"))") // 十六进制:2b转二进制
// 输出:----b: 00101011
swift自带api
let a = 0x2b
let str = String(a, radix: 2)
print("str: \(str)")
// 输出:str: 101011
八进制——>十进制
0o1101——>577
原理:(类似于二进制 转 十进制,因为是小端模式,分别从右往左对八进制每位进行加权计算,然后将所有加权的值累加即为 十进制结果)
1*8^0 + 0*8^1 + 1*8^2 + 1*8^3 = 557
swift实现代码:
func octalToDecimal(_ v: String) -> Int {
guard v.count > 0 else { return 0 }
var idx = 0, sum = 0
for i in v.indices.reversed() {
if let a = Int(String(v[i])) {
sum += (a*Int(pow(8, Double(idx))))
idx += 1
return sum
print("----d: \(octalToDecimal("1101"))") // 八进制:1101转十进制
// 输出:----d: 577
swift自带api:
let a = 0o1101
let str = String(a, radix: 10)
print("str: \(str)")
// 输出:str: 577
十进制——>八进制
577——>1101
原理:(根据八进制 转 十进制 可推导出,对十进制数 除以 8,余数作为二进制有效位,商>0,继续对商进行上面的操作,直到商==0为止)
swift实现代码:
func decimalToOctal(_ v: Int) -> String {
guard v > 0 else { return "0" }
var str = ""
var a = v
while a > 0 {
str.insert(Character("\(a % 8)"), at: str.startIndex)
a /= 8
return str
print("---o: \(decimalToOctal(577))") // 十进制:577 转 八进制
// 输出:---o: 1101
swift自带api:
let a = 577
let str = String(a, radix: 8)
print("str: \(str)")
// 输出:str: 1101
八进制——>十六进制
1271——>2b9
原理:(中转法)现将八进制 转成 二进制 或 十进制,然后通过二进制 或 十进制 再转 十六进制。
swift自带api:
let a = 0o1271
let str = String(a, radix: 16)
print("str: \(str)")
// 输出:str: 2b9
十六进制——>八进制
2b9——>1271
原理:(中转法)现将十六进制 转成 二进制 或 十进制,然后通过二进制 或 十进制 再转 八进制。
swift自带api:
let a = 0x2b9
let str = String(a, radix: 8)
print("str: \(str)")
// 输出:str: 1271
以上就是全部的进制之间的转化过程,接下来介绍一下对二进制操作常用到的位运算符
“&”运算符
两个二进制对应位都为1,该位的结果才为1,否则为0。如下:
第一个二进制
第二个二进制
&运算后的结果
&运算符在日常开发中用处比较多,举个最长用到的例子,在OC开发中,我们定义枚举,经常会利用&来判断一个包含多个枚举值的情况:
NS_ENUM(NSInteger, MyType) {
MyType1 = 1 << 0,
MyType2 = 1 << 1,
MyType3 = 1 << 2,
enum MyType type = MyType1 | MyType2;
if ((type & MyType1) == MyType1) {
NSLog(@"-----type中包含MyType1枚举值");
}else {
NSLog(@"-----type中不包含MyType1枚举值");
if ((type & MyType2) == MyType2) {
NSLog(@"-----type中包含MyType2枚举值");
}else {
NSLog(@"-----type中不包含MyType2枚举值");
if ((type & MyType3) == MyType3) {
NSLog(@"-----type中包含MyType3枚举值");
}else {
NSLog(@"-----type中不包含MyType3枚举值");
// 输出:
// -----type中包含MyType1枚举值
// -----type中包含MyType2枚举值
// -----type中不包含MyType3枚举值
我们也可以利用&运算符来判断一个数是奇数还是偶数,只要二进制位第0位=1,那这个数必定是奇数,如果=0,那就是偶数。为什么?根据上面介绍的二进制转十进制可知:第一位的值要么是0(0*2^0),要么是1(1*2^0),而后面位数对应的值都是2的倍数(偶数),因此一个偶数+1=奇数。这样我们只需要判断第一位是1还是0,就可以判断出这个数是奇还是偶 。如何判断二进制第一位是1还是0呢,我们可以让这个数与1进行&运算,即x & 0000 0001 = 1(奇数),x & 0000 0001 = 0(偶数),如下:
let x = 123
if x & 1 == 0 {
print("偶数")
}else {
print("奇数")
“|”运算符
两个二进制对应位只要有一个为1,该位的结果就为1,否则为0。如下:
第一个二进制
第二个二进制
|运算后的结果
enum MyType type = MyType1 | MyType2;
if ((type & MyType1) == MyType1) {
NSLog(@"-----type中包含MyType1枚举值");
// 删除枚举值 MyType1
type &= ~MyType1;
if ((type & MyType1) == 0) {
NSLog(@"枚举MyType1从type变量中移除");
我们也可以通过该运算符计算一个数的相反数,如下:(一个数x的相反数=~x + 1)
let x = -123
print("x: \(~x + 1)") // 计算x的相反数
// 输出:x: 123
“<<”左移运算符
对二进制的每位进行左移,右侧空出的位用0补,运算符后面跟着移动的位数,例如:x << 2,对x进行左移2位。
一个二进制
由于二进制每位之间都相差2^n倍数,因此对一个数进行左移运算,相当于这个数乘以2^n(n:移动的位数)
例如:3 << 1:相当于:3 * 2^1 = 6。因此要求一个数的2的幂数,可以这样计算:1 << n:相当于:1 * 2^n
“>>”右移运算符
与左移运算符相反,它是对对二进制的每位进行右移,左侧空出的位用0补,运算符后面跟着移动的位数,例如:x >> 2,对x进行右移2位。
右移运算相当于这个数除以2^n(n:移动的位数)
一个二进制
左移与右移运算也在开发中常用到,例如我们可以通过对一个数右移一位来实现除以2的操作。
另外还有用于取指定位的值,例如:要取一个八位无符号的二进制的前4位的数值,如下:
1、二进制:1111 0000
2、对二进制右移4位:0000 1111
3、计算十进制数值:1*2^0 + 1*2^1 + 1*2^2 + 1*2^3 = 15
在取一个图片中指定像素点的色值的时候,我们就会用到移位算法。一个色值是包括r、g、b、a四个分量,根据不同的颜色通道的mode,它们的排列顺序也不同。一般以r、g、b、a顺序居多。颜色的每个分量采用8位二进制数表示,因此一个色值的位数可以是32位。根据它们的排列顺序,我们就可以通过移位运算,来分别取出每个分量值。例如:
----------------x----------------
---r---- ---g--- ---b--- ---a---
00000100 00001001 00000010 00000001
r:x >> 24
g:(x << 8) >> 24
b:(x << 16) >> 24
a:(x << 24) >> 24
其实对于上面的取值还有一些技巧,比如在取b分量的值时,我们可以通过x&00000000_00000000_11111111_00000000来快速取值,这样就不需要移位了。一般我们会采用十六进制来表示这样的一串二进制数。我们知道十六进制中最大值是F,而每个十六进制数占4位,因此8位的最大值就是八个1,用十六进制表示就是:FF。
下面将展示不通过移位方法来去各个分量的值。
r:x & 0xFF000000
g:x & 0x00FF0000
b:x & 0x0000FF00
a:x & 0x000000FF
这种计算方法会减少很多移位的步骤,因此这样计算的效率肯定比移位方法要高。
“>>>”(无符号右移)运算符(某些开发语言中不支持该运算符,例如:swift、OC)
这个运算符跟>>运算符类似,都是将二进制位右移,而>>>在移动时不会考虑符号位,会将符号位用0补。而>>在移动的时候会考虑符号位,并不会用0来补充符号位。因此通过>>>运算后得到的值肯定是一个正数。
1、关于如何完整打印出二进制位的字符串,下面是问题描述:
Int8:代表8位二进制位的整型,其二进制位字符串完整输出应该是长度为8的字符串,例如:00001111。
知识点,关于Int(Int8、Int16、Int32、Int64)的属性介绍:
let a = Int8(12) // 拿Int8为例
print("非完整的二进制字符串:\(String(a, radix: 2))") // 二进制字符串:1100
print("bit位数:\(a.bitWidth), ") // 8
print("bit位左侧(头部)0的个数:\(a.leadingZeroBitCount)") // 4
print("bit位右侧(尾部)0的个数:\(a.trailingZeroBitCount)") // 2
print("非0比特位的个数:\(a.nonzeroBitCount)") // 2
综上知识点,我们可以打印完整的二进制位字符串的方法如下:
let a = Int8(12) // 拿Int8为例
let bitStr = String(repeating: "0", count: a.leadingZeroBitCount) + String(a, radix: 2)
print("完整的二进制字符串:\(bitStr)") // 00001100
2、如果获取下一个2的幂数,问题描述如下:
2的幂数列表为:2^0、2^1、2^2、2^3、2^4...,即:1 、2 、4 、8 、16...
给出一个数,返回这个数的下一个2的幂数,例如:1->1、2->2、3->4、4->4、5->8、6->8、7->8、8->8、9->16、10->6、11->16....
// 例如:给出一个数n
let n = 6
let nextPowerOf2 = 1 << (n.bitWidth - (n - 1).leadingZeroBitCount)
print("nextPowerOf2: \(nextPowerOf2)") // 8