isa指针:
在Objective-C中,任何类的定义都是对象。类和类的实例(对象)没有任何本质上的区别。任何对象都有 isa
指针。
isa
是一个Class 类型的指针。 每个实例对象有个 isa
的指针,他指向对象的类,而Class里也有个 isa
的指针, 指向 meteClass
(元类)。元类保存了类方法的列表。当类方法被调用时,先会从本身查找类方法的实现,如果没有,元类会向他父类查找该方法。同时注意的是:元类也是类,它也是对象。元类也有 isa
指针,它的 isa
指针最终指向的是一个根元类。根元类的 isa
指针指向本身,这样形成了一个封闭的内循环。
const 含义
-
面试题
const int a; int const a; const int *a; int const *a; int * const a; int const * const a;
- 前两个的作用是一样:a 是一个常整型数
- 第三、四个意味着 a 是一个指向常整型数的指针(整型数是不可修改的,但指针可以)
- 第五个的意思:a 是一个指向整型数的常指针(指针指向的整型数是可以修改的,但指针是不可修改的)
- 最后一个意味着:a 是一个指向常整型数的常指针(指针指向的整型数是不可修改的,同时指针也是不可修改的)
-
合理地使用关键字const可以使编译器保护那些不希望被改变的参数,防止其被无意的代码修改,减少bug。
- 欲阻止一个变量被改变,可以使用 const 关键字。在定义该 const 变量时,通常需要对它进行初始化,因为以后就没有机会再去改变它了;
- 对指针来说,可以指定指针本身为 const,也可以指定指针所指的数据为 const,或二者同时指定为 const;
- 在一个函数声明中,const 可以修饰形参,表明它是一个输入参数,在函数内部不能改变其值;
- 对于类的成员函数,若指定其为 const 类型,则表明其是一个常函数,不能修改类的成员变量;
- 对于类的成员函数,有时候必须指定其返回值为 const 类型,以使得其返回值不为“左值”。
static关键字
- 函数体内
static
变量的作用范围为该函数体,不同于auto
变量,该变量的内存只被分配一次,因此其值在下次调用时仍维持上次的值; - 在模块内的
static
全局变量可以被模块内所用函数访问,但不能被模块外其它函数访问; - 在模块内的
static
函数只可被这一模块内的其它函数调用,这个函数的使用范围被限制在声明它的模块内; - 在类中的
static
成员变量属于整个类所拥有,只会初始化一次,并且在程序退出时才会回收内存; - 在类中的
static
成员函数属于整个类所拥有,这个函数不接收this
指针,因而只能访问类的static
成员变量。
-
static
用途- 限制变量的作用域
- 设置变量的存储域
volatile关键字
volatile
的变量是说这变量可能会被意想不到地改变,这样,编译器就不会去假设这个变量的值了。精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。下面是
- volatile 变量的几个例子
- 并行设备的硬件寄存器(如:状态寄存器)
- 一个中断服务子程序中会访问到的非自动变量(Non-automatic variables)
- 多线程应用中被几个任务共享的变量
- 一个参数既可以是 const 还可以是 volatile 吗?解释为什么。
是的。一个例子是只读的状态寄存器。它是 volatile 因为它可能被意想不到地改变。它是 const 因为程序不应该试图去修改它 - 一个指针可以是 volatile 吗?解释为什么。
是的。尽管这并不很常见。一个例子是当一个中服务子程序修该一个指向一个buffer的指针时。
self. 跟 self-> 区别?
self. 是调用 get 方法或者 set 方法 self 是当前本身,是一个指向当前对象的指针 self-> 是直接访问成员变量
懒加载
懒加载——也称为延迟加载,只在用到的时候才去初始化,比如控制器的view,在第一次用到view时才会调用loadView方法进行创建,所谓懒加载,写的是其get方法。我觉得最好也最简单的一个列子就是tableView中图片的加载显示了。一个延时载,避免内存过高,一个异步加载,避免线程堵塞。
注意:如果是懒加载的话则一定要注意先判断是否已经有了,如果没有那么再去进行实例化
- 懒加载的好处:
- 不必将创建对象的代码全部写在viewDidLoad方法中,代码的可读性更强
- 每个控件的 getter 方法中分别负责各自的实例化处理,代码彼此之间的独立性强,松耦合
Objective C中的selector 是什么?
你可以理解 @selector()就是取类方法的编号,他的行为基本可以等同C语言的中函数指针,只不过C语言中,可以把函数名直接赋给一个函数指针,而 Objective-C的类不能直接应用函数指针,这样只能做一个@selector语法来取.它的结果是一个SEL类型。这个类型本质是类方法的编号 (函数地址)。 方法和选择器有何不同? 答案:selector是一个方法的名字,method是一个组合体,包含了名字和实现。通过一个selector可以找到方法地址,进而调用一个方法
其他
-
[[NSMutableArray alloc]init]
和[NSMutableArray array]
-
[[NSMutableArray alloc]init]
alloc
分配内存,init
初始化,需要手动释放; -
[NSMutableArray array]
不需要手动release
,遵循autoreleasepool
机制 - 在ARC(自动引用计数)中两种方式并没什么区别
-
new
与alloc/init
两种方式创建对象现在基本上一样,区别就是使用new
只能默认init
进行初始化,alloc
方式可以使用其它的init
开头的方法进行初始化。-
子线程要主动开启 runloop
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ [self performSelector:@selector(fireBlock:) withObject:^{ NSLog(@"hello world"); } afterDelay:0.3]; });
上面的代码片段原有的目的是异步延迟0.3秒后输出Hello world。但是运行之后发现不会输出Hello world。
原因是:非主线程的NSRunLoop默认没有开启,而- (void)performSelector:(SEL)aSelector withObject:(nullable id)anArgument afterDelay:(NSTimeInterval)delay;
函数内部是通过NSTimer定时器实现,在NSRunLoop没有开启的情况下,NSTimer不会得到正常运行。 -
类的加载和初始化
-
+ (void)load;
调用:程序一启动的时候就会把所有的类加载进内存
作用:加载类的时候调用 -
+ (void)initialize;
调用:当第一次使用这个类或者子类的时候调用
作用:初始化类
-
代码布局还是 storyboard ?
对于复杂的、动态生成的界面,建议使用手工编写界面。
对于需要统一风格的按钮或UI控件,建议使用手工用代码来构造。方便之后的修改和复用。
对于需要有继承或组合关系的 UIView 类或 UIViewController 类,建议用代码手工编写界面。
对于那些简单的、静态的、非核心功能界面,可以考虑使用 xib 或 storyboard 来完成。-
有些图片加载的比较慢怎么处理?你是怎么优化程序的性能的?
- 图片的下载放在异步的线程
- 图片的下载过程使用占位图片
- 如果图片较大,可以考虑多线程断点下载
-
Foundation对象与Core Foundation对象有什么区别
- Foundation对象是OC的,Core Foundation对象是C对象
- 数据类型之间的转换
ARC:__bridge_retained
、__bridge_transfer
非ARC:__bridge
写 个“标准”宏,这个宏输出两个参数并返回较大的 #define MIN(X,Y) ((X)>(Y)?(Y):(X))
iPhone OS中有没有垃圾回收? 没有
self.name=“object” 和 name=“object”有什么区别?
前者实际上是调用了set 方法给变量赋值, 后者是直接给变量赋值Objective-c中有私有方法吗?私有变量呢?
没有私有法,但可以将方法直接实现在.m文件中不在.h 件中声明时,外部也不能访问。
有私有变量Objective-c中有多重继承么?不是的话有声明替代?
没有多继承,可以通过协议模拟多继承多态性
答案:不同对象以自己的方式响应相同的消息的能力叫做多态。意思就是假设生物类(life)都用有一个相同的方法-eat;那人类属于生物,猪也属于生物,都继承了life后,实现各自的eat,但是调用是我们只需调用各自的eat方法。
多态。 主要是将数据类型的确定由编译时,推迟到了运行时。比如运行的时候修改某个方法的实现,或者枚举某个对象都有哪些属性等等。也就是不同的对象以自己的方式响应了相同的消息(响应了eat这个选择器)。因此也可以说,运行时机制是多态的基础。
多态,子类指针可以赋值给父类。关于多态,继承和封装基本最好都有个自我意识的理解。
站在编程的角度来说,就是以C语言的视角看待OC中的对象、类、方法等等 也就是可以理为类、对象、方法被映射成了一些结构体、函数、指针等等 比如你要枚举某个对象有哪些属性,就可以调运行时里的一些函数。 JSonModel、YYModel就是基于运行时实现的,其通过枚举model里有哪些属性、及属性的类型,去对应的json(数组和字典)来找相应的字段,进而完成解析-
obj-c的优缺点
- 优点
- Cateogies
- Posing
- 运行时特性
- 指标计算
- 弹性讯息传递
- 不是一个过度复杂的 C 衍生语言
- Objective-C 与 C++ 可混合编程
- 缺点:
- 不支援命名空間,对于命名冲突,可以使用长命名法或特殊前缀解决,如果是引入的第三方库之间的命名冲突,可以使用link命令及flag解决冲突
- 不支持运算符重载
- 不支持多重继承
- 使用动态运行时类型,所有的方法都是函数调用,所以很多编译时优化方法都用不到。(如内联函数等),性能低劣。
- 优点
静态语言:你写的源代码编译之后完全变成了机器码 效率高
动态语言:你写的源代码没有完全编译成机器码全局变量和局部变量在内存中是否有区别?如果有,是什么区别?
全局变量储存在静态数据库,局部变量在堆栈-
id 声明的变量有什么特性?
- 没有 * 号
- 动态数据类型
- id声明的变量能指向任何OC对象(设置是nil),而不关心其具体类型
- 在运行时检查其具体类型
- 可以对其发送任何(存在的)消息
-
苹果的安全机制:
- 没经过用户同意,你不能随便获取用户信息。
- 所有的程序都在沙盒里运行,B程序不能进入A程序的运行范围。
- 如果跟钱有关,比如说支付宝,这些底层的实现都是保密的,只提供接口供开发者调用,这样的话安全性得到保障。
- 如果要防止代码被反编译,可以将自己的代码中的.m文件封装成静态库(.a文件)或者是framework文件,只提供给其它人.h文件。这样就保证了个人代码的安全性。
- 网络登录的话跟用户名跟密码相关要发送POST请求,如果是GET请求的话密码会直接在URL中显示。然后同时要对帐号密码采用加密技术,加一句:我们公司用的是MD5,但是现在MD5有一个专门的网站来破解,为了防止这个,可以采用加盐技术。
-
写一个NSString类的实现
+ (id)stringWithCString: (c*****t char*)nullTerminatedCString encoding: (NSStringEncoding)encoding { NSString *obj; obj = [self allocWithZone: NSDefaultMallocZone()]; obj = [obj initWithCString: nullTerminatedCString encoding: encoding]; return AUTORELEASE(obj); }
什么是平衡二叉树?
左右子树都是平衡二叉树且左右子树的深度差值的绝对值不大于1局部变量能否和全局变量重名?
能,局部会屏蔽全局。要用全局变量,需要使用"::"
局部变量可以与全局变量同名,在函数内引用这个变量时,会用到同名的局部变量,而不会用到全局变量。对于有些编译器而言,在同一个函数内可以定义多个同名的局部变量,比如在两个循环体内都定义一个同名的局部变量,而那个局部变量的作用域就在那个循环体内如何引用一个已经定义过的全局变量?
答:可以用引用头文件的方式,也可以用extern关键字,如果用引用头文件方式来引用某个在头文件中声明的全局变量,假定你将那个变写错了,那么在编译期间会报错,如果你用extern方式引用时,假定你犯了同样的错误,那么在编译期间不会报错,而在连接期间报错。NSString 和 NSMutableString 有什么区别:
NSString 相当于一个 const char* 不可以改变。
NSMutableString 相当于 char* 可以改变内部的内容。图片操作:
imageNamed: 优点是当加载时会缓存图片。图片使用频繁的使用比较好,一般用于加载小图片。
iamgeWithContentsFile: 大图片。每次调用,会占缓存。
imageWithContentsOfFile:仅加载图片,图像数据不会缓存。因此对于较大的图片以及使用情况较少时,那就可以用该方法,降低内存消耗。类实例(成员)变量的
@protected
,@private
,@public
声明各有什么含义?
@protected
:受保护的,该实例变量只能在该类和其子类内访问,其他类内不能访问。
@private
:私有的,该实例变量只能在该类内访问,其他类内不能访问。
@public
:共有的,该实例变量谁都可以访问。@dynamic 和 @synthesize
@dynamic: 的意思是告诉编译器,属性的获取与赋值方法由用户自己实现, 不自动生成。对于只读属性需要提供 setter,对于读写属性需要提供 setter 和 getter。
@synthesize: 意思是,除非开发人员已经做了,否则由编译器生成 getter 和 setter 属性声明。UIscrollVew 到了什么设计模式?还能再foundation库中找到类似的吗?
组合模式(composition):所有的container view都用了这个模式
观察者模式(observer):所有的UIResponder都用了这个模式。
模板模式(Template):所有datasource和delegate接口都是模板模式的典型应用-
_btn.frame.origin.y = 10
错误
原因:OC语法规定不允许直接修改某个对象的结构体属性的成员。_btn 是个对象,frame是个结构体。
对象和结构体是不一样的,结构体是C语言中的,里面可以定义许多属性,但是不能定义方法,而对象是即可以定义属性又可以定义方法的,是典型的面向对象语法。
如何改变对象中结构体属性的成员:// 先取出结构体 CGRect frame = _btn.frame; // 修改结构体 frame.origin.y -= 10; // 将修改后的结构体重新赋值回去 _btn.frame = frame;
或者
// 先取出y值 CGFloat y = _btn.frame.origin.y; // 修改y值 y -= 10; // 重新设置_btn的y值,其他属性和_btn保持不变 _btn.frame = CGRectMake(_btn.frame.origin.x, y, _btn.frame.size.width, _btn.frame.size.height);
-
如果想让同一个控件同时即改变位置的移动,又放大。这样设置是无效果的。这样操作是创建新的transform然后赋值,给按钮的transform,第二次赋值的会把之前赋值的给覆盖,所以会达不到想要的效果。
_btn.transform = CGAffineTransformMakeTranslation(0, 100); _btn.transform = CGAffineTransformMakeScale(1.2, 1.2);
解决方法:
_btn.transform = CGAffineTransformMakeTranslation(0, 100); // 在之前的transform情况下,继续添加缩放的形变。 _btn.transform = CGAffineTransformScale(_btn.transform, 1.2, 1.2);
-
四舍五入问题
float i = 1.7; // 会自动四舍五入,不保留小数 NSLog(@"%0.f",i); // 打印结果2 // 强转类型不会四舍五入 int j = (int)i; NSLog(@"%d",j); // 打印结果1
-
id、nil代表什么?
- id 和 void *并非完全一样。在上面的代码中,id是指向struct objc_object的一个指针,这个意思基本上是说,id是一个指向任何一个继承了Object(或者NSObject)类的对象。需要注意的是id是一个指针,所以你在使用id的时候不需要加星号。比如id foo=nil定义了一个nil指针,这个指针指向NSObject的一个任意子类。而id *foo=nil则定义了一个指针,这个指针指向另一个指针,被指向的这个指针指向NSObject的一个子类。
- nil和C语言的NULL相同,在objc/objc.h中定义。nil表示一个Objctive-C对象,这个对象的指针指向空(没有东西就是空)。
首字母大写的Nil和nil有一点不一样,Nil定义一个指向空的类(是Class,而不是对象)。
常见的object-c的数据类型有那些,和C的基本数据类型有什么区别?如:NSInteger和int
答:object-c的数据类型有NSString,NSNumber,NSArray,NSMutableArray,NSData等等,这些都是class,创建后便是对象,而C语言的基本数据类型int,只是一定字节的内存空间,用于存放数值;NSInteger是基本数据类型,并不是NSNumber的子类,当然也不是NSObject的子类。NSInteger是基本数据类型Int或者Long的别名(NSInteger的定义typedef long NSInteger),它的区别在于,NSInteger会根据系统是32位还是64位来决定是本身是int还是Long。