Как привести объект в Objective-C - PullRequest
115 голосов
/ 27 марта 2009

Есть ли способ приведения объектов в target-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]; 

Однако я получаю ошибку компилятора, так как свойство 'list' существует в классе SelectionListViewController, но отсутствует в FieldEditViewController, хотя SelectionListViewController наследуется от FieldEditViewController.

Это имеет смысл, но есть ли способ привести myEditController к SelectionListViewController, чтобы я мог получить доступ к свойству list?

Например, в VB.NET я бы сделал:

CType(myEditController, SelectionListViewController).list = listOfItems

Спасибо за помощь!

Ответы [ 5 ]

204 голосов
/ 27 марта 2009

Помните, Objective-C является надмножеством C, поэтому приведение типов работает так же, как и в C:

myEditController = [[SelectionListViewController alloc] init];
((SelectionListViewController *)myEditController).list = listOfItems;
11 голосов
/ 27 марта 2009
((SelectionListViewController *)myEditController).list

Больше примеров:

int i = (int)19.5f; // (precision is lost)
id someObject = [NSMutableArray new]; // you don't need to cast id explicitly
7 голосов
/ 18 июля 2015

Типизация в Objective-C проста как:

NSArray *threeViews = @[[UIView new], [UIView new], [UIView new]];
UIView *firstView = (UIView *)threeViews[0];

Однако, что происходит, если первый объект не UIView и вы пытаетесь использовать его:

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

Это потерпит крах. И в этом случае легко найти такой сбой, но что, если эти строки находятся в разных классах, а третья строка выполняется только один раз в 100 случаях. Бьюсь об заклад, ваши клиенты находят этот сбой, а не вы! Возможное решение - ранний сбой , например:

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

Эти утверждения выглядят не очень хорошо, поэтому мы могли бы улучшить их с помощью этой удобной категории:

@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

Это намного лучше:

UIView *firstView = [UIView typecastWithAssertion:[threeViews[0]];

P.S. Для коллекций тип безопасности Xcode 7 гораздо лучше, чем типизирование - generics

4 голосов
/ 27 марта 2009

Конечно, синтаксис точно такой же, как C - NewObj* pNew = (NewObj*)oldObj;

В этой ситуации вы можете рассмотреть возможность предоставления этого списка в качестве параметра конструктору, например:

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

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

  return self;
}

Тогда используйте это так:

myEditController = [[SelectionListViewController alloc] initWith: listOfItems];
0 голосов
/ 14 ноября 2011

Приведение к включению так же важно, как приведение к исключению для программиста C ++. Приведение типов отличается от RTTI в том смысле, что вы можете привести объект к любому типу, и результирующий указатель не будет равен нулю.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...