私は、SwiftでSystem Configurationフレームワークを使用して、macOSでネットワークインターフェイスの設定を読み込もうとしています。私は実際にCFDictionary
であるCFPropertyList
を得ています。各辞書エントリにはCFArray
が含まれています。 CFShow
と私は期待されたデータを得たことを確認することができました。私が実際に行うことができないのは、辞書の値にアクセスすることです。 CFGetTypeId
では、私はCFArrayGetTypeID()
によって返されたものと同じ値を得ていません。ここでCFPropertyListから値を読み取る
は、私は、プロパティリストからIPアドレスを取得しようとしたものです:次のように
import SystemConfiguration
func readIP() {
let cfName = "Demo" as CFString
let dynamicStore = SCDynamicStoreCreate(nil, cfName, nil, nil)
let key = "State:/Network/Interface/en0/IPv4" as CFString
guard let plist: CFPropertyList = SCDynamicStoreCopyValue(dynamicStore, key) else {return}
print("CFShow(plist):")
CFShow(plist)
print("Key: \(key):")
print("Value: \(plist)")
let typeIdPList = CFGetTypeID(plist)
let typeIdDictionary = CFDictionaryGetTypeID()
print("typeIdPList: \(typeIdPList)")
print("typeIdDictionary: \(typeIdDictionary)")
guard typeIdPList == typeIdDictionary else {return}
let cfDict: CFDictionary = plist as! CFDictionary
let rawPointerToKeyAddresses = Unmanaged.passUnretained(kSCPropNetIPv4Addresses).toOpaque()
var rawPointerToValue: UnsafeRawPointer?
guard CFDictionaryGetValueIfPresent(cfDict, rawPointerToKeyAddresses, &rawPointerToValue) == true else {return}
let anyRef: CFTypeRef = rawPointerToValue as CFTypeRef
print("Value:")
CFShow(anyRef)
let typeIdValue = CFGetTypeID(anyRef)
let typeIdArray = CFArrayGetTypeID()
print("typeIdValue: \(typeIdValue)")
print("typeIdArray: \(typeIdArray)")
let cfArray: CFArray = anyRef as! CFArray
let typeId = CFGetTypeID(cfArray)
print("typeId: \(typeId)")
let desc = CFCopyDescription(anyRef)
let typeDesc = CFCopyTypeIDDescription(CFGetTypeID(anyRef))
print("CFTypeRef description: \(desc!)")
print("CFTypeRef type description: \(typeDesc!)")
}
出力は次のようになります。
CFShow(plist):
<CFBasicHash 0x610000069500 [0x7fffc4383da0]>{type = immutable dict, count = 3, entries =>
0 : Addresses = <CFArray 0x610000069440 [0x7fffc4383da0]>{type = immutable, count = 1, values = (
0 : <CFString 0x610000028980 [0x7fffc4383da0]>{contents = "192.168.139.24"}
)}
1 : <CFString 0x610000044c20 [0x7fffc4383da0]>{contents = "BroadcastAddresses"} = <CFArray 0x610000069480 [0x7fffc4383da0]>{type = immutable, count = 1, values = (
0 : <CFString 0x610000044c50 [0x7fffc4383da0]>{contents = "192.168.139.255"}
)}
2 : <CFString 0x7fffc429d300 [0x7fffc4383da0]>{contents = "SubnetMasks"} = <CFArray 0x6100000694c0 [0x7fffc4383da0]>{type = immutable, count = 1, values = (
0 : <CFString 0x6100000289a0 [0x7fffc4383da0]>{contents = "255.255.255.0"}
)}
}
Key: State:/Network/Interface/en0/IPv4:
Value: {
Addresses = (
"192.168.139.24"
);
BroadcastAddresses = (
"192.168.139.255"
);
SubnetMasks = (
"255.255.255.0"
);
}
typeIdPList: 18
typeIdDictionary: 18
Value:
0x0000610000069440
typeIdValue: 1
typeIdArray: 19
typeId: 1
CFTypeRef description: 0x0000610000069440
CFTypeRef type description: CFType
ポインタのキャストを行うことのもう一つの(等価)の方法があることCFDictionary'と 'NSDictionaryの'の間の架橋'はフリーダイヤルです。スウィフト辞書に橋渡しするのにはオーバーヘッドがあり、場合によっては問題になるかもしれません。説明されたユースケースではないかもしれません。なぜなら、ビュー要素のみが存在し、更新はまれですからです。 – Bokeh
@Bokeh:CFPropertyListは既にサブスクリプトをサポートしているようですので、 'guard let plist = SCDynamicStoreCopyValue(dynamicStore、key)else {return}'は大文字と小文字を区別しません。 - そして、私は、スウィフトブリッジのコストが適切であるとは思っていません。 Apple Swiftブログのhttps://developer.apple.com/swift/blog/?id=37の例を比較してください。 –
実際には、Core Foundationコレクションとrawポインタを使ってすべての面倒をする方法を理解したいと思います。私はそれらを何度も見つけ、問題が発生するたびに来ました。それ以外の場合は、パフォーマンス上の理由からCFインターフェイスを直接使用する必要があります。 – Bokeh