如何在Objective-C中转换对象

Bil*_*lly 120 objective-c

有没有办法在objective-c中转换对象,就像在VB.NET中转换对象的方式一样?

例如,我正在尝试执行以下操作:

// create the view controller for the selected item
FieldEditViewController *myEditController;
switch (selectedItemTypeID) {
    case 3:
        myEditController = [[SelectionListViewController alloc] init];
        myEditController.list = listOfItems;
        break;
    case 4:
        // set myEditController to a diff view controller
        break;
}

// load the view
[self.navigationController pushViewController:myEditController animated:YES];
[myEditController release]; 
Run Code Online (Sandbox Code Playgroud)

但是我收到编译器错误,因为'list'属性存在于SelectionListViewController类中,但不存在于FieldEditViewController上,即使SelectionListViewController继承自FieldEditViewController.

这是有道理的,但有没有办法将myEditController转换为SelectionListViewController,以便我可以访问'list'属性?

例如在VB.NET中,我会这样做:

CType(myEditController, SelectionListViewController).list = listOfItems
Run Code Online (Sandbox Code Playgroud)

谢谢您的帮助!

Jim*_*uls 209

请记住,Objective-C是C的超集,因此类型转换就像在C中一样工作:

myEditController = [[SelectionListViewController alloc] init];
((SelectionListViewController *)myEditController).list = listOfItems;
Run Code Online (Sandbox Code Playgroud)

  • 或者"记住,Objective-C的工作方式与Java类似,只记得在指向Obj-C对象的变量中添加星号." (20认同)
  • 很好的答案。您可以通过将演员表和分配分成两行来使其更加清晰。 (2认同)

Sij*_*der 11

((SelectionListViewController *)myEditController).list
Run Code Online (Sandbox Code Playgroud)

更多例子:

int i = (int)19.5f; // (precision is lost)
id someObject = [NSMutableArray new]; // you don't need to cast id explicitly
Run Code Online (Sandbox Code Playgroud)

  • 一般来说这是正确的; 您不需要在消息表达式中转换id.但是当使用点语法来访问和设置属性时,必须使用具体类型,而不仅仅是id,因此编译器知道实际生成的方法调用.(对于具有相同名称的属性,它可能不同.) (7认同)

Ale*_*nin 8

Objective-C中的类型转换很简单:

NSArray *threeViews = @[[UIView new], [UIView new], [UIView new]];
UIView *firstView = (UIView *)threeViews[0];
Run Code Online (Sandbox Code Playgroud)

但是,如果第一个对象不是UIView并且您尝试使用它会发生什么:

NSArray *threeViews = @[[NSNumber new], [UIView new], [UIView new]];
UIView *firstView = (UIView *)threeViews[0];
CGRect firstViewFrame = firstView.frame; // CRASH!
Run Code Online (Sandbox Code Playgroud)

它会崩溃.并且很容易在这种情况下找到这样的崩溃,但如果这些行在不同的类中并且第三行在100种情况下仅执行一次该怎么办呢.我打赌你的客户会发现这次崩溃,而不是你!一个看似合理的解决方案是提前崩溃,如下所示:

UIView *firstView = (UIView *)threeViews[0];
NSAssert([firstView isKindOfClass:[UIView class]], @"firstView is not UIView");
Run Code Online (Sandbox Code Playgroud)

那些断言看起来不太好,所以我们可以用这个方便的类别来改进它们:

@interface NSObject (TypecastWithAssertion)
+ (instancetype)typecastWithAssertion:(id)object;
@end


@implementation NSObject (TypecastWithAssertion)

+ (instancetype)typecastWithAssertion:(id)object {
    if (object != nil)
        NSAssert([object isKindOfClass:[self class]], @"Object %@ is not kind of class %@", object, NSStringFromClass([self class]));
    return object;
}

@end
Run Code Online (Sandbox Code Playgroud)

好得多:

UIView *firstView = [UIView typecastWithAssertion:[threeViews[0]];
Run Code Online (Sandbox Code Playgroud)

PS对于集合类型安全Xcode 7比类型转换更好 - 泛型


And*_*ant 5

当然,语法与 C 完全相同 -NewObj* pNew = (NewObj*)oldObj;

在这种情况下,您可能希望考虑将此列表作为参数提供给构造函数,例如:

// SelectionListViewController
-(id) initWith:(SomeListClass*)anItemList
{
  self = [super init];

  if ( self ) {
    [self setList: anItemList];
  }

  return self;
}
Run Code Online (Sandbox Code Playgroud)

然后像这样使用它:

myEditController = [[SelectionListViewController alloc] initWith: listOfItems];
Run Code Online (Sandbox Code Playgroud)