美文网首页
NSString的几个初始化方法区别

NSString的几个初始化方法区别

作者: 咖啡绿茶1991 | 来源:发表于2017-01-11 22:39 被阅读0次

    1、initWithFormat是实例方法

    只能经由过程 NSString* str = [[NSString alloc] initWithFormat:@"%@",@"Hello World"] 调用,然则必须手动release来开释内存资料

    2、stringWithFormat是类方法

    可以直接用 NSString* str = [NSString stringWithFormat:@"%@",@"Hello World"] 调用,内存经管上是autorelease的,不消手动显式release

    别的国外有个贴子对此有专门评论辩论并且提出了一个常见错误:label.text = [[NSString alloc] initWithFormat:@"%@",@"abc"];最后在dealloc中将label给release掉然则仍然会产生内存泄漏!

    原因在于:用label.text = ...时,实际是隐式调用的label的setText办法,这会retain label内部的字符串变量text(哪怕这个字符串的内容跟传进来的字符串内容雷同,但体系仍然当成二个不合的字符串对象),所以最后release label时,实际上只开释了label内部的text字符串,然则最初用initWithFormat生成的字符串并未开释,终极造成了泄漏。

    解决办法有二个:

    1、NSString * str = [[NSString alloc] initWithFormat:@"%@",@"abc"];label.text = str;[str release]最后在dealloc中再[label release]

    2、label.text = [NSString stringWithFormat:@"%@",@"abc"];然后剩下的工作交给NSAutoreleasePool最后,若是你不断定你的代码是否有内存泄漏题目,可以用Xcode中的Build-->Build And Analyze 做初步的搜检.

    前几天,同事提到initWithString和initWithFormat的区别问题,觉得很有意思,决定研究下,现把成果和大家分享。下面是测试代码:

    NSString * str =[[NSString alloc] initWithString:@"this is from initWithString function"];

    NSLog(@"this is from [[NSString alloc] initWithString] m_addr is %ld retainCount is %i", str, [str retainCount]);

    [str release];

    [str release];

    [str release];

    NSLog(@"this is from [[NSString alloc] initWithString] m_addr is %ld retainCount is %i", str, [str retainCount]);

    str = [[NSString alloc] initWithFormat:@"this is from initWithFormat function"];

    NSLog(@"this is from [[NSString alloc] initWithFormat] m_addr is %ld retainCount is %d", str, [str retainCount]);

    下面是LOG的结果:

    this is from [[NSString alloc] initWithString] m_addr is 12356 retainCount is 2147483647

    this is from [[NSString alloc] initWithString] m_addr is 12356 retainCount is 2147483647

    this is from [[NSString alloc] initWithFormat] m_addr is 82076688 retainCount is 1

    this is from [[NSString alloc] initWithString] m_addr is 12356 retainCount is 2147483647

    this is from [[NSString alloc] initWithString] m_addr is 12356 retainCount is 2147483647

    this is from [[NSString alloc] initWithFormat] m_addr is 78748112 retainCount is 1

    this is from [[NSString alloc] initWithString] m_addr is 12356 retainCount is 2147483647

    this is from [[NSString alloc] initWithString] m_addr is 12356 retainCount is 2147483647

    this is from [[NSString alloc] initWithFormat] m_addr is 78777072 retainCount is 1

    我将上面这段测试代码调用了三次,得到以上的LOG结果。

    顺便得出这么几条结论:

    1.从两个变量的地址看,两个变量的地址差据较大。前者的地址非常靠前。

    2.从release看,前者无论被release多少次,都不会被释放,而且的值不变,而后者只要release一次,变量即消亡。

    3.前者的releaseCount= NSIntegerMax,而NSIntegerMax ==INT_MAX ,而 UINT_MAX== (INT_MAX * 2U + 1U)。U是指无符号整型,而我们默认的int和NSInteger是有符号的,在32位系统中,NSIntegerMax=0X7FFFFFFF,对其乘以2U,即一次向左挪一位。结果为0XFFFFFFFE,再加1U为0XFFFFFFFF,即无符号的最大值。

    然后根据苹果官方对于retainCount方法的描述:retainCountReturns the receiver’s reference count. (required)- (NSUInteger)retainCountReturn ValueThe receiver’s reference count.DiscussionYou might override this method in a class to implement your own reference-counting scheme. For objects that never get released (that is, their release method does nothing), this method should return UINT_MAX, as defined in.

    此处的UINT_MAX和NSIntegerMax是一样的,都是表示所在类型的最大值。所以,initWithString这个方法初始化后的对象是不可能被release的或者说,它的release方法啥也不干。

    验证了上面的分析2.因为他不可能被release的或者说,它的release方法啥也不干,所以我们调用无数次release都没有起到预先想想的作用。

    为什么会导致这样的情况呢?

    我们再次把目光转向地址。眼尖的同学可能会看到initWithString申请的地址每次都是一样的,而initWithFormat的地址每次都不一样,这个说明什么?

    说明initWithString的地址是静态的,而initWithFormat是动态的。为什么前者是静态的,而后者是动态的?

    结合上面关于retainCount的分析,认为initWithString的地址申请是在编译是进行的,这样才能说明为什么它的地址空间如此靠前。只有在编译是进行的,他才是静态的。

    卡盟刷钻http://www.kamengshuazuanpt.com/对于initWithString生成的对象,对其进行dealloc时,程序会报错(这里就不贴LOG了)。而后者initWithFormat不会报错。这进一步验证了initWithString生成的是静态对象,而initWithFormat是动态的。

    结论:initWithString生成的对象是在编译是申请地址空间,而且在程序中不能释放,不建议使用。(当然也有可能在某种情况下会使用到这个方法,在此不多加讨论)。

    在ObjectiveC中NSString中有一个 stringWithFormat:方法

    常见的输出方式:

    NSString *height;

    height = [NSString stringWithFormat:@"Your height is %d feet, %d inches.",5,11];

    NSLog(@"%@",height);

    输出结果:

    2013-04-12 10:30:47.744 String[2161:303] Your height is 5 feet, 11 inches.

    输出多个字符的方式(以两个字符为例):

    NSString *str;

    NSString *str1 = @"123";

    NSString *str2 = @"465";

    str = [NSString stringWithFormat:@"%@,%@",str1, str2];

    NSLog(@"%@",str);

    输出结果:

    2013-04-12 10:31:48.213 String[2171:303] 123,465

    下面这种方式是错误的:

    str = [NSString stringWithFormat:@"123",@"456"];

    NSLog(@"%@",str);

    关于stringWithFormat:

    输出结果:

    2013-04-12 10:35:34.043 String[2209:303] 123

    @“456” 不会被输出;

    以上代码可以改写成:

    str = [NSString stringWithFormat:@"%@,%@",@"123",@"456"];

    NSLog(@"%@",str);

    输出结果:

    2013-04-12 10:42:39.541 String[2229:303] 123,456

    我们经常会初始化一些string使用NSString的stringWithString函数

    但使用时发现了一个stringWithString的问题,如图

    当参数是nil时,stringWithString会crash,所以使用时必须验证参数不是nil

    相比较stringWithFormat就不会crash但返回的str也不是nil而是@"(null)"

    所以再做此类操作时事先要判断参数

    相关文章

      网友评论

          本文标题:NSString的几个初始化方法区别

          本文链接:https://www.haomeiwen.com/subject/abarbttx.html