Тип "MyWeights" не соответствует протоколу "NSCopying"
Я использую Metal Performance Shader для настройки нейронной сети и столкнулся с проблемой при написании класса инициализации весов: тип "MyWeights" не соответствует протоколу "NSCopying". Что вызвало ошибку и как это исправить?
PS. Я попытался исправить это, добавив функцию copy(), однако я не знаю, что возвращать или что это значит.
import Foundation
import MetalPerformanceShaders
class MyWeights: NSObject, MPSCNNConvolutionDataSource {
//Error: Type 'MyWeights' does not conform to protocol 'NSCopying'
/*
func copy(with zone: NSZone? = nil) -> Any {
return self
}
*/
let name: String
let kernelWidth: Int
let kernelHeight: Int
let inputFeatureChannels: Int
let outputFeatureChannels: Int
var data: Data?
init(name: String, kernelWidth: Int, kernelHeight: Int,
inputFeatureChannels: Int, outputFeatureChannels: Int,
useLeaky: Bool = true) {
self.name = name
self.kernelWidth = kernelWidth
self.kernelHeight = kernelHeight
self.inputFeatureChannels = inputFeatureChannels
self.outputFeatureChannels = outputFeatureChannels
}
func dataType() -> MPSDataType {
return .float32
}
func descriptor() -> MPSCNNConvolutionDescriptor {
let desc = MPSCNNConvolutionDescriptor(kernelWidth: kernelWidth,
kernelHeight: kernelHeight,
inputFeatureChannels: inputFeatureChannels,
outputFeatureChannels: outputFeatureChannels)
return desc
}
func weights() -> UnsafeMutableRawPointer {
return UnsafeMutableRawPointer(mutating: (data! as NSData).bytes)
}
func biasTerms() -> UnsafeMutablePointer<Float>? {
return nil
}
func load() -> Bool {
if let url = Bundle.main.url(forResource: name, withExtension: "dat") {
do {
data = try Data(contentsOf: url)
return true
} catch {
print("Error: could not load \(url): \(error)")
}
}
return false
}
func purge() {
data = nil
}
func label() -> String? {
return name
}
}
1 ответ
Он говорит вам точно, что делать.
Вы должны объявить, что ваш класс соответствует NSCopying
протокол, а затем вам нужно реализовать единственную функцию в этом протоколе, copy(with:)
class MyWeights: NSObject, MPSCNNConvolutionDataSource, NSCopying {
func copy(with zone: NSZone? = nil) -> Any {
return MyWeights(
name: self.name,
kernelWidth: self.kernelWidth,
kernelHeight: self.kernelHeight,
inputFeatureChannels: self.inputFeatureChannels,
outputFeatureChannels: self.outputFeatureChannels,
useLeaky: self.useLeaky)
}
//The rest of your class
}
Вы должны реализовать весь протокол NSCopying
class MyWeights: NSObject, MPSCNNConvolutionDataSource, NSCopying {
init(/* ... */) {
// your init...
super.init() // NSObject init
}
// add this method
func copy(with zone: NSZone? = nil) -> Any {
return super.copy() // NSObject copy
}
//The rest of your class
}
Протокол источника данных свертки изменился с годами, и недавно был добавлен протокол NSCopying, так что MPSCNNConvolution может сам соответствовать NSCopying. Возможно, какой-то пример кода не продвинулся вперед. MPS должен проверять доступные селекторы, чтобы это не было проблемой двоичной совместимости, но в дальнейшем ваши пользователи протокола должны соответствовать NSCopying, чтобы структура могла вести себя так, как рекламируется. Ответы выше демонстрируют, как это сделать.