zoukankan      html  css  js  c++  java
  • OC基础13:数字、字符串和集合2

      "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类的子类。

  • 相关阅读:
    流复制-pg_basebackup (有自定义表空间)
    流复制-pg_basebackup (没有自定义表空间)
    PG 更新统计信息
    PG修改参数方法
    Postgres的索引01
    Postgres基础操作
    PostgreSQL安装
    SQL拦截器
    没对象的快自己写一个吧!带你了解一下python对象!
    喜欢看电影来哦!教你如果使用Python网络爬虫爬取豆瓣高分电影!
  • 原文地址:https://www.cnblogs.com/shayneyeorg/p/4748135.html
Copyright © 2011-2022 走看看