再次使用Swift,我错过了什么吗?
我有这一行:
self.itemDescription?.setContentCompressionResistancePriority(UILayoutPriorityRequired, forAxis: UILayoutConstraintAxis.Vertical);
但是Xcode给了我一个错误:
Undefined symbols for architecture i386: "_UILayoutPriorityRequired"
另一个斯威夫特的怪癖?
答案 0 :(得分:25)
解决方案:将UILayoutPriorityRequired
替换为1000
self.itemDescription?.setContentCompressionResistancePriority(1000, forAxis: UILayoutConstraintAxis.Vertical);
这不是错误。这是对Objective-C库如何导入Swift的误解。应该理解Swift如何从Objective-C向Swift导入代码(甚至从Apple UIKIt Libraries)。
UILayoutPriority是一个浮点数。在Objective-C中,为我们预先定义了几个值。预定义的值似乎是枚举。我们可能期望在Swift中可以使用相同的枚举。
文档似乎暗示了一个枚举:
SWIFT(文档)
typealias UILayoutPriority = Float
目标-C(文件)
enum {
UILayoutPriorityRequired = 1000,
UILayoutPriorityDefaultHigh = 750,
UILayoutPriorityDefaultLow = 250,
UILayoutPriorityFittingSizeLevel = 50,
};
typedef float UILayoutPriority;
但是在Xcode中,如果你要求查看其中一个枚举值(例如UILayoutPriorityRequired)的定义,你会发现它们实际上是在头文件中定义为常量浮点数。
OBJECTIVE-C(头文件)
typedef float UILayoutPriority;
static const UILayoutPriority UILayoutPriorityRequired NS_AVAILABLE_IOS(6_0) = 1000; // A required constraint. Do not exceed this.
static const UILayoutPriority UILayoutPriorityDefaultHigh NS_AVAILABLE_IOS(6_0) = 750; // This is the priority level with which a button resists compressing its content.
static const UILayoutPriority UILayoutPriorityDefaultLow NS_AVAILABLE_IOS(6_0) = 250; // This is the priority level at which a button hugs its contents horizontally.
因此,尽管我们可能会将预定义的布局优先级视为枚举值(如文档所示),但布局优先级并未真正定义为枚举;它们被定义为常量浮点数。
对于任何知道C编程语言的人来说,存在不匹配的提示是C enum可能只包含int
个值。以下是合法的,将编译:
enum myEnum {
JGCEnum_one = 1, // this is O.K.
JGCEnum_two,
JGCEnum_three
} JGCEnum;
但是我们不能将浮动定义为C枚举的值。以下内容无法编译:
enum myEnum {
JGCEnum_one = 1.5, // compile error
JGCEnum_two,
JGCEnum_three
} JGCEnum;
Objective-C枚举与C枚举相同(Swift枚举不同)。重要的是要知道我们是否正在处理实际的整数或浮点数。因为可以使用NS_ENUM宏定义整数,然后可以将其作为Swift枚举导入Swift。
iBook说
Swift导入作为Swift枚举的任何标有NS_ENUM宏的C样式枚举。这意味着枚举值名称的前缀在导入Swift时会被截断,无论它们是在系统框架中定义还是在自定义代码中定义。
摘自:Apple Inc. “Using Swift with Cocoa and Objective-C.” iBooks
这意味着如果使用NS_ENUM宏将UILayoutPriority 定义为整数,则它将作为Swift枚举导入Swift。这是UILayoutConstraintAxis的情况。
SWIFT(文档)
enum UILayoutConstraintAxis : Int {
case Horizontal
case Vertical
}
目标-C(文件)
enum {
UILayoutConstraintAxisHorizontal = 0,
UILayoutConstraintAxisVertical = 1
};
typedef NSInteger UILayoutConstraintAxis;
查看Objective-C头文件确认文档说的内容。
目标-C(头文件)
//
// UIView Constraint-based Layout Support
//
typedef NS_ENUM(NSInteger, UILayoutConstraintAxis) {
UILayoutConstraintAxisHorizontal = 0,
UILayoutConstraintAxisVertical = 1
};
因此,至少有两种方法可以知道您在Objective-C中使用的预定义值是否在Swift中可用:
还有一种方法可以查看您习惯使用的typedef是常量还是枚举。在代码中,测试以查看常量的地址是否存在。常量有一个内存地址,而枚举没有。请参阅下面的代码。
// this line will compile and run just fine.
// UILayoutPriorityDefaultHigh is a constant and has a memory address
// the value will be true if the device is running iOS 6.0 or later
// and false otherwise
BOOL predefinedValueIsAvailable = (NULL != &UILayoutPriorityDefaultHigh);
// this line will not compile
// UILayoutConstraintAxisHorizontal is an enum (NOT a constant)
// and does not have a memory address
predefinedValueIsAvailable = (NULL != &UILayoutConstraintAxisHorizontal);
答案 1 :(得分:8)
要记住的一件事是,即使var items = {
"8f348f-b4d0-66bf-b0cj-a31b9c45c": "Pete",
"ff5764-7863-6pc9-b72f-195f2h962": "Dave"
}
var item = {
key: "8f348f-b4d0-66bf-b0cj-a31b9c45c",
value: "Pete"
}
delete items[item.key];
console.log(items);
是UILayoutPriority
的别名,您也可以使用Swift,并且可以使用扩展为这些值创建语义常量:
Float
然后你的代码可以读作:
extension UILayoutPriority {
static var Low: Float { return 250.0 }
static var High: Float { return 750.0 }
static var Required: Float { return 1000.0 }
}
我已经在我的一个项目中完成了这项工作并认为它可能有用。