Swift 基本知识之十四 Sizeof 和 sizeofValue 解析

喜欢写 C 的读者可能会经常和 sizeof 打交道,不论是分配内存,I/O 操作,还是计算数组大小的时候基本都会用到。这个在 C 中定义的运算符可以作用于类型或者某个实际的变量,并返回其在内存中的尺寸 size_t (这是和平台无关的一个整数类型)。在 Cocoa 中,我们也有一部分 API 需要涉及到类型或者实例的内存尺寸,这时候就可以使用 sizeof 来计算。一个常见的用例是在从一个数组生成 NSData 的时候需要传入数据长度。因为在 Objective-C 中 sizeof 这个 C 运算符被保留了,因此我们可以直接这么做:

    char bytes = {1,2,3};
    NSData *data = [NSData dataWithBytes:&bytes length:sizeof(bytes)];
    NSLog(@"%@",data);
    //在C语言中sizeof有两个版本,即可以计算类型的长度,又可以接受某个值
    int a = sizeof(int);
    int b = sizeof(bytes);
    NSLog(@"a==%d,b==%d",a,b);
    
与传统的 C 或者 Objective-C 里的运算符的存在形式不同,在 Swift 中,为了保证类型安全,sizeof 经过了一层包装。在 Swift 中 sizeof 不再是运算符,而是一个只能接受类型的方法。我们还可以找到一个接受具体值,并返回其尺寸的方法:sizeofValue。另外,这样个方法返回的都是看起来比较友好和直接的 Int,而不再是 size_t:

func sizeof<T>(_: T.Type) -> Int
func sizeofValue<T>(_: T) -> Int

虽然 sizeofValue 接受的是具体值,但是它和 C 时代的接受具体值的版本的 sizeof 行为并不相同。Swift 的 sizeofValue 所返回的是这个值实际的大小,而并非其内容的大小。具体来说,如果我们在 Swift 中想表示上面的 bytes 的话,我们会将其类型写为 [CChar]。在 C 或者 Objective-C 中,对 bytes 做 sizeof 返回的是整个数组内容在内存中占据的尺寸,每个 char 为 1,而数组元素为 3,因此这个值为 3。而在 Swift 中,我们如果直接对 bytes 做 sizeofValue 操作的话,将返回 8,这其实是在 64 位系统上一个引用的长度:

// C
char bytes[] = {1, 2, 3};  
sizeof(bytes);  
// 3

// Swift
var bytes: [CChar] = [1,2,3]  
sizeofValue(bytes)  
// 8
所以,我们不能简单地用 sizeofValue 来获取长度,而需要进行一些计算。上面的生成 NSData 的方法在 Swift 中书写的话,等效的代码应该是下面这样的:

var bytes: [CChar] = [1,2,3]  
let data = NSData(bytes: &bytes, length:sizeof(CChar) * bytes.count) 
在看一个例子:
enum MyEnum: UInt16 {
    case A = 0
    case B = 1
    case C = 65535
}

print("UInt16 =\(sizeof(UInt16))")
print("MyEnum = \(sizeof(MyEnum))")
print("MyEnum.A = \(sizeofValue(MyEnum.A))")
print("MyEnum.A.rawValue=\(sizeofValue(MyEnum.A.rawValue))")
结果:

这里写图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

图解AI

你的鼓励是我最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值