iOS常见问题(3)

一、发现不少人在给成员变量初始化的时候,容易进错一个方法去初始化。

 1 //注意这个方法只有在内存发生警告的时候才会调用。
 2 - (void)didReceiveMemoryWarning
 3 {
 4     [super didReceiveMemoryWarning];
 5     // Dispose of any resources that can be recreated.
 6     
 7 }
 8 
 9 //大部分成员属性的初始化应该在viewDidLoad里面进行。
10 - (void)viewDidLoad
11 {
12     [super viewDidLoad];
13 }

  最牛解决方法:在一开始就将didReceiveMemoryWarning这个方法删掉。删掉不会影响程序运行。

二、模拟器问题

  

  当出现这个问题的时候,原因:没有选择模拟器。

  

  解决办法:

三、对象方法和类方法问题

 1 // 创建视图的工厂方法
 2     + (UIView *)rowViewInitWithicon:(UIImage *)icon shuju:(NSString *)shuju
 3     {
 4         rowView *viewtext = [[NSBundle mainBundle]loadNibNamed:@"rowView" owner:nil options:nil][0];
 5         [viewtext.btntouxiang setBackgroundImage:icon forState:UIControlStateNormal];
 6         viewtext.mingzilablexiao.text = shuju;
 7         
 8         // 重点是这句
 9         //  这是往通知中心添加一条通知 指定通知名称为  back  当观察者self监听到 back 通知是 就调用 callback
10         [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(callback) name:@"back" object:nil];
11         
12         /*   以下是报错信息:
13          +[rowView callback]: unrecognized selector sent to class 0x79d8
14          2013-11-26 15:31:02.581 lianxirenlianxi[1266:c07] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '+[rowView callback]: unrecognized selector sent to class 0x79d8'
15          */
16         // reason: '+[rowView callback]: 看到报错原因里的+就想到没有实现callback这个类方法。
17         
18         // 由于self这个观察者是在类方法中添加的,指的是一个类,所以在调用方法的时候,他会去类方法中找有没有这个方法,不会去对象方法中找。因此我们也应该实现类方法。因此这里的self也只能调用类方法
19     
20     // 由于实现的callback为对象方法  所以会报错
21         //  解决方法  把callback  写成类方法   供观察者调用
22         return viewtext;
23     }
24     //callback方法
25     - (void)callback
26     {
27         NSLog(@"11111111111");
28     }
29 //解决方法,将callback 写成类方法 供观察者调用
30 //callback方法
31     + (void)callback
32     {
33         NSLog(@"11111111111");
34     }

四、Xib是用来描述视图长什么样子,一个项目中允许有很多xib,因此我们需要给xib绑定一个标识,即他View中对应的class是谁,就代表描述哪个class。

  

  Xib中owner的class是用来告诉xib中的View需要调用哪个对象的方法,就填谁。比如需要调用dog类中的方法,就填dog。

  

  注意:在连线选择上别连错了,

  步骤一:先考虑自己是想给视图添加控件了还是想给视图添加事件

  步骤二:添加控件就跟xib中的view连线。添加一些事件就给xib中的File’s Owner 连线。

五、代码顺序问题

 1 [UIView animateWithDuration:0.5 animations:^{
 2         CGRect  tmpFrame = sender.superview.frame;
 3         tmpFrame.origin.x = self.view.frame.size.width;
 4         sender.superview.frame = tmpFrame;
 5         sender.superview.alpha = 0;
 6     } completion:^(BOOL finished) {
 7         int index = [self.view.subviews indexOfObject:sender.superview];
 8         [sender.superview removeFromSuperview];
 9         [UIView animateWithDuration:0.2 animations:^{
10             for (int i = index; i<self.view.subviews.count; i++)
11             {
12                 UIView *chlid = self.view.subviews[i];
13                 CGRect tmp = chlid.frame;
14                 tmp.origin.y -=kViewH+1;
15                 chlid.frame =tmp;
16             }
17             
18         }];
19         // 在这判断删除按钮是否允许点中,会在动画执行完毕的时候,判断。
20         _removeIteam.enabled = self.view.subviews.count>1;
21 
22     }];
23      // 而在执行代码块之外,判断删除按钮是否允许点中是不对的,因为动画是在后台运行的,所以在执行动画的时候,就已经执行完判断语句了,而这时最后一个视图还没销毁掉,因此删除按钮永远不会不允许选中,也就不能在判断删除按钮是否允许点中。
24 // _removeIteam.enabled = self.view.subviews.count>1;

  删完最后一行之后,正确的效果:

  删完最后一行之后,错误的效果。原因,判断的位置放错了:

六、Xib描述视图的时候,已经固定好描述视图的宽高了,外界调用视图的时候,只需要设置x,y值就好了

  1.出现的问题,创建xib描述的视图时,将宽度设置为一个按钮的宽度了,导致删除按钮不能点击。

  

 1 #pragma mark 添加联系人
 2 - (IBAction)AddPerson:(UIBarButtonItem *)sender {
 3     
 4     NSString *imgName=[NSString stringWithFormat:@"01%d.png",arc4random_uniform(9)];
 5     NSString *labelName = arr[arc4random_uniform(arr.count)];
 6     RowView *rowView = [RowView rowViewWithIcon:imgName name:labelName];
 7     UIView *lastView = [self.view.subviews lastObject];
 8     int nextY = lastView.frame.origin.y + kSpace + kItemHW ;
 9     // 设置rowView的位置和尺寸
10     CGRect cg =CGRectMake(0, nextY, kItemHW, kItemHW);
11     rowView.frame=cg;
12     
13     [self.view addSubview:rowView];
14 }

  错误原因:设置rowView的宽度为kItemHW,因此会有以上图片的出现。

   错误会导致删除按钮不能点击,原因:父视图的尺寸不够,即父视图能接收事件的尺寸只有一点点,也就导致超出父视图尺寸的子视图不能监听点击事件。还有一点需要注意,将子视图添加到父视图尺寸之外的位置,只要还在屏幕上就会显示子视图,只不过它不能接收任何事件

  解决方法:

CGRect cg =CGRectMake(0, nextY,rowView.frame.size.width , kItemHW);

  这样设置就好了,因为xib里面已经设置了rowView的尺寸了,外界不需要更改视图的宽度了,直接获取视图的宽度即可。

  正确效果:

七、UIToolBar问题

  注意UIToolBar中不能使用viewWithTag这个方法,获取UIToolBar里的子视图。因为UIToolBar里的子视图都是UIBarButtonItem,而UIBarButtonItem是继承NSObject的,因此不能使用viewWithTag获取UIToolBar里的子视图,

  viewWithTag:实现原理

 1 - (UIView *)viewWithTag:(NSInteger)tag
 2 {
 3     
 4   // 1.如果当前tag和当前视图tag相同,直接返回
 5   if (self.tag == tag) return self;
 6    
 7     // 2.如果和当前视图tag不相同,遍历当前视图的所有子控件,查找对应的tag。
 8   for (UIView *view in self.subviews) {
 9    // 3.如果view不是UIView类或者UIView的子类直接返回nil
10     if (![view isKindOfClass:[UIView class]]) return nil;
11       if (tag == view.tag) {
12     //  4. 返回有相匹配的视图
13             return view;
14         }
15     }
16     // 5.如果都没有找到,返回nil.
17     return nil;
18 }

八、结构体和对象问题

1 // 这样定义是错的,结构体不是对象,声明变量是不需要加*
2 CGRect *frame = self.view.frame;
3 
4 //结构体变量正确定义:
5 CGRect frame = self.view.frame;
6 CGPoint center = self.view.center;
7 CGSize size = self.view.frame.size;

九、内存管理问题

  错误打印:

  

  正确打印:

  

  当对象被销毁,一定会调用的方法,可以用这个方法,判断对象在什么时候销毁,用这个调试。

  

  

十、创建模型的时候,尽量自定义一个工厂方法供外界调用。

  // 工厂方法,简化对象的实例化

+ (id)provinceWithName:(NSString *)name;

  工厂方法好处:简化对象的实例化,快速创建对象。

原文地址:https://www.cnblogs.com/CJDog/p/5113911.html