객체가 VB.NET에서 캐스팅되는 방식과 같이 Objective-C에서 객체를 캐스팅하는 방법이 있습니까?
예를 들어, 다음을 시도하고 있습니다.
// 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 클래스에 존재하지만 SelectionListViewController가 FieldEditViewController에서 상속하더라도 FieldEditViewController에는 없기 때문에 컴파일러 오류가 발생합니다.
이것은 의미가 있지만 ‘list’속성에 액세스 할 수 있도록 myEditController를 SelectionListViewController로 캐스팅하는 방법이 있습니까?
예를 들어 VB.NET에서는 다음을 수행합니다.
CType(myEditController, SelectionListViewController).list = listOfItems
도와 주셔서 감사합니다!
답변
Objective-C는 C의 상위 집합이므로 형변환은 C에서와 같이 작동합니다.
myEditController = [[SelectionListViewController alloc] init];
((SelectionListViewController *)myEditController).list = listOfItems;
답변
((SelectionListViewController *)myEditController).list
더 많은 예 :
int i = (int)19.5f; // (precision is lost)
id someObject = [NSMutableArray new]; // you don't need to cast id explicitly
답변
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]];
– 컬렉션 유형 안전 엑스 코드 7의 PS는 훨씬 더 타입 캐스팅 이상이 제네릭을
답변
물론 구문은 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];
답변
포함을위한 캐스팅은 C ++ 프로그래머의 제외를위한 캐스팅만큼 중요합니다. 유형 캐스팅은 객체를 모든 유형으로 캐스팅 할 수 있고 결과 포인터가 nil이 아니라는 점에서 RTTI와 동일하지 않습니다.