"OC基础"这个分类的文章是我在自学Stephen G.Kochan的《Objective-C程序设计第6版》过程中的笔记。
17、Foundation框架的数组是有序的对象集合。一般来说这些对象元素会是同一种类型,但是也可以不同。
18、不可变数组是由NSArray类处理的,可变数组是有NSMutableArray类处理的。NSMutableArray类是NSArray类的子类。
19、NSArray类常见方法:
(1)、NSArray *xxx = [NSArray arrayWithObject: @”a”, @”b”, nil];
这个方法可以接收可变数量的对象作为数组元素,最后一个值要指定为nil;
(2)、NSArray *xxx = @[@”a”, @”b”];
这个方法和(1)效果相同,可以不用加“nil”;
(3)、[xxx objectAtIndex: i];
相当于:xxx[i];
(4)、[xxx setObject: o forIndex: i];
相当于:xxx[i] = o;
20、两种输出数组内容的方法的区别:
...
for(i=0; i<10; ++i) {
NSLog(@”%@”, xxx[i]);
}
...
NSLog(@”%@”, xxx);
...
使用第一种方法是多次调用NSLog方法,每次显示一个元素;使用第二种方法,编译器也会显示所有元素,不过元素之间会用逗号隔开并且过行,同时整个数组会用括号括起来。
21、可以使用[xxx addObject: ...]方法往NSMutableArray类数组的末尾添加元素。
22、@”i”和@(i)的区别:
@”i”是字符串i,@(i)是数字i,注意@(i)里面的i必须是数字,在代码中直接写@(i)会报错。
23、定义NSMutableArray数组对象的时候,需要先调用初始化方法,才能往数组里添加元素。至少要调用一下以下方法:
NSMutableArray xxx = [NSMutableArray array];
如果不调用这个方法,即使写了往数组里添加元素的语句,数组里的内容也仍然都是(null)。
24、成员对象?修改成员对象的值对原先对象的影响。
成员对象即是包含在一个对象内部的对象,一般来说为成员对象赋值都只是赋了指针,改变赋值对象的值会影响到成员对象的值。
可参照第八章第6。
26、在for循环中使用in关键字可以遍历数组的元素。使用块enumerateObjectsUsingBlock:也可以遍历。
27、重载init函数的时候,如果涉及到在初始化的过程中顺便要给实例变量(不是属性)赋值,那么在赋值的语句中,直接写“实例变量=形参”即可,不要使用点语法。
比如:
-(instancetype) initWith: (NSString *) theBookName {
self = [super init];
if(self) {
bookName = theBookName; //实例变量这里不要使用点语法
book = [NSMutableArray array]; //可以顺便给参数列表没有的内容赋值
}
return self;
}
还有另一种方法,可以写成“self->bookName = theBookName”;
28、removeObjectIdenticalTo:方法和removeObject:方法的区别:
(1)、removeObjectIdenticalTo:方法是通过“==”来判断,只要“对象1==对象2”为真,那么removeObjectIdenticalTo:方法就会把两个都删除( “对象1==对象2”为真则表示对象1和对象2指向了内存中同一块区域);
(2)、removeObject:方法是通过“isEqual:”来判断,只要“对象1 isEqual:对象2”为真,那么removeObject:方法就会把两个都删除( “对象1 sEqual:对象2”为真则表示对象1和对象2内容相同);
(3)、“isEqual:”比较的是内容,而“==”只是简单的内存地址比较。一般情况下“==”为真的 “isEqual”就为真,但是“isEqual”为真就不一定“==”也为真。
(4)、看以下例子:
...
NSString *str1 = [[NSString alloc] init];
NSString *str2 = [[NSString alloc] init];
NSString *str3 = [str1 stringByAppendingFormat:@"字符串"];
NSString *str4 = [str2 stringByAppendingFormat:@"字符串"];
NSMutableArray *muArray = [NSMutableArray arrayWithCapacity:6];
[muArray addObject:@"对象"];
[muArray addObject:str3];
[muArray addObject:str4];
for (NSObject *object in muArray) {
NSLog(@"数组内容:%@", object);
}
if ([str3 isEqual:str4]) {
NSLog(@"str1 isEqual str2"); //这个会显示
}
if (str3 == str4) {
NSLog(@"str1 == str2"); //这个不会显示
}
//[muArray removeObject:str3]; 这个会删除str3和str4
[muArray removeObjectIdenticalTot:str3]; //这个只会删除str3
for (NSObject * object in muArray) {
NSLog(@"内容对象:%@", object);
}
...
29、看以下代码:
- (void) removeCard: (NSString *) theName {
AddressCard *ac;
for (ac in book) { //book是一个NSMutableArray数组
if([ac.name compare: theName] == NSOrderedSame) {
[book removeObject: ac];
NSLog(@”%@删除成功!”, ac.name);
return;
} else {
NSLog(@”没找到”);
return;
}
}
}
这段代码有两个错误:
(1)、如果要删除的对象不是数组里的第一个对象,那么就会输出“没找到”,检查代码可以发现是因为只查找一次就return了,没有继续查找下去;
(2)、一旦有一个对象删除成功了,for in继续循环下去就会报错,报NSGenricException错误。原因是不能在for in循环中修改所遍历的数组,无论你是add或remove都不可以,如果要修改的话,要使用for(;;;)格式。
(3)、使用for(;;;)格式删除数组的内容的话,还有一点一定要注意,比如以下语句:
for(i=0; 1<[array count]; i++) {
if(...) {
[array removeObject: array[i]];
}
}
这个语句有可能在删除的时候会漏了内容,比如有两个紧挨在一起的数组元素是符合删除条件的,那么只会删掉第一个,第二个会被漏过。原因在于:删除掉第一个可删除的元素之后,后面的所有元素的下标就会自减1,即是全部前移了一位,那么如果删除的是第i个元素的话,下一个元素就会自动顶替了i的位置,而for循环下一个要删除的是(i+1),这就造成了紧挨着的第二个可删除元素会被漏掉。
解决方法是在执行了删除方法后,加上一句“i--;”。
30、“==”运算符只判断thing1和thing2的指针数值,即是地址,而不是判断它们所指的内容。有时我们想检查两个对象thing1和thing2是否同一个对象,这时就应该使用运算符“==”,如果是想查看等价性(即这两个对象内容是否相同),那么请使用isEqualToString: 。
31、使用下面的语句可以使程序读取键盘输入的字符串,但是不安全,因为gets()函数会无限地接收键盘输入的内容,甚至内容大到会覆盖掉有存储内容的内存它也不会检测出来:
/*控制台输入字符串并打印*/
char str[50] = {0};
printf("输入名字并回车:");
scanf("%s",str);
NSString *lastName = [NSString stringWithUTF8String:str];
NSLog(@"lastName=%@",lastName);
32、系统提示breakpoint 1.1错误,其实不是错误,是因为在提示的位置被设了断点。
33、sortUsingSelector:(SEL)方法:
(1)、这个方法是对应NSMutableArray对象进行排序,由于NSMutableArray对象是可变的,所以可以直接调用这个方法,调用完后数组就会是排序完的了;
(2)、参数(SEL)必须是一个比较方法,即是要返回NSComparisonResult类的对象,并且这个(SEL)方法必须是这个可变数组里的元素的比较方法,而整个sortUsingSelector:方法是给这个可变数组本身来调用的;
(3)、(SEL)方法可以是在另一个文件中,只要sortUsingSelector:(SEL)方法有import这个文件即可;
(4)、正常排序的结果是从小到大,(即是0到9,从a到z,从大写到小写),如果要反过来,必须要修改(SEL)方法,让方法的比较结果应该返回NSOrderedAscending的换成返回NSOrderedDescending,应该返回NSOrderedDescending的返回NSOrderedAscending;
(5)、以下是一个sortUsingSelector:(SEL)方法的例子:
//AddressCard.m文件
...
(NSComparisonResult) compareName: (AddressCard *) card {
return [self.name compare: card.name];
}
//AddressBook.m文件,已import了AddressCard.m文件
...
(void) sort {
[book sortUsingSelector: @selector(compareName:)]; //book是AddressBook的实例
//变量,是一个NSMutableArray
}
...
这个例子的排序结果是正序;
(6)、如果要倒序,修改compareName:方法如下即可:
(NSComparisonResult) compareName: (AddressCard *) card {
return [card.name compare:self.name];
}
34、使用块来排序:
还有另一种更简单的使用块来排序的方法,以33的内容为例,也可以在AddressBook.m中使用以下代码来排序:
-(void) sort {
[book sortUsingComparator:^(AddressCard obj1, AddressCard obj2) {
return [[obj1 name] compare: [obj2 name]];
}];
}
使用块来排序有两个好处:
(1)、可以在sort方法里面直接定义数组元素的比较方法,不需要另外去添加一个比较方法;
(2)、如果要修改数组元素的比较方法,只需修改sort方法即可,不用动别的部分。
35、关于sortArrayUsingSelector:方法:
sortArrayUsingSelector:方法和sortUsingSelector:方法的功能是一样的,只不过sortArrayUsingSelector:方法是NSArray类的方法,有一个(NSArray *)的返回值,而sortUsingSelector:方法是NSMutableArray类的方法,没有返回值。
同理,使用块来排序的方法中:sortUsingComparator:是NSMutableArray类的方法,sortArrayUsingComparator:是NSArray类的方法。
36、类声明的时候要用*,而id类型声明的时候不用*,其他声明时不用*的有:
NSInteger(typedef的int);
NSComparisonResult;
NSRange(typedef的结构,包含了location和length属性,用来表示字符串的一部分)
37、关于NSValue类:
(1)、NSValue类可以把结构转化为对象,对象才可以存储在数组内;
(2)、将结构转化为对象,称为“包装”,逆处理称为“展开”。
38、“词典”是由“键-对象”组成的数据集合。键必须是单值的,一般是字符串,也可以是其他对象类型。和键关联的值可以是任何类型,但是不能为nil。
39、注意以下语句:
...
NSMutableDictionary *md = [NSMutableDictionary dictionary];
[md setObject: @”xxxxxxxxxx”
forKey:@”a”];
NSLog(@”a is %@”, [md objectForKey: @”a”]);
...
分别展示了如何定义、赋值、取值。
40、另外还有以下简便语法:
md[key] = object
相当于:
[md setObject:object forKey:key];
md[key]
相当于
[md objectForKey: key];
注意key和object仍然必须是@“...”的字符串书写格式,或者遵照其他类型对象的书写格式。
41、关于不可变词典对象NSDictionary的两种定义方式:
... //第一种
NSDictoinary *nsd =
[NSDictionary dictionaryWithObjectsAndKeys:
@”xxx”, //对象
@”a”, //键,这两个构成一组
@”xxxxxx”,
@”b”,
nil];
...//第二种
NSDictionary *nsd = @{
@”a”:@”xxxxx”, //键和对象,构成一组
@”b”:@”xxxxxxxx”};
42、NSMutableDictionary类是NSDictionary类的子类。