xcode怎么开发cocoa+xcode opengl 性能

xcode运行OpenGL程序 - CSDN博客
xcode运行OpenGL程序
xcode运行OpenGL程序
运行环境的搭建
我是使用的xcode7.3.1,最近入门图形学,所以必须在电脑上跑一跑程序,网上写了一些关于xcode运行OpenGL程序的例子,都是相互抄袭的,还是xcode4版本,这里我贴出如何配置运行环境的图,其实只需要添加两个包,但是在Products下面那个黑框框里面添加,点击那个黑框框,就很清晰了,需要添加GLUT和OpenGL两个包。
我的第一个OpenGL程序
第一个OpenGL程序是按照书上抄的,不是显示的老掉牙的Hello World,是显示了一个窗口,画了一条线,毕竟OpenGL就是研究图形图像么,肯定先从线开始,废话不多说,上代码(这里需要注意,初学者代码一行都不可以少,因为有些代码你不懂意思,删了是无法显示的)。
#include &iostream&
#include &OpenGl/gl.h&
#include &GLUT/GLUT.h&
void init(){
glClearColor(1.0, 1.0, 1.0, 0.0);
glMatrixMode(GL_PROJECTION);
gluOrtho2D(0.0, 200.0, 0.0, 150.0);
void lineSegment(){
glClear(GL_COLOR_BUFFER_BIT);
glColor3f(1.0, 0.0, 0.0);
glBegin(GL_LINES);
glVertex2i(180, 15);
glVertex2i(10, 145);
glFlush();
int main(int argc, char** argv){
glutInit(&argc, argv);
glutInitDisplayMode(GLUT_SINGLE | GLUT_RGB);
glutInitWindowPosition(500, 100);
glutInitWindowSize(400, 300);
glutCreateWindow("An Example OpenGL Program");
glutDisplayFunc(lineSegment);
glutMainLoop();
本文已收录于以下专栏:
相关文章推荐
在Xcode中配置GLEW+GLFW环境
因为这几天一直在忙着图形学的实验,发现真的是很蛋疼,因为课程还在使...
Nehe的OpenGL教程,准备从把所有教程边学边翻写,开发平台是 Mac + Xcode + c。...
一、MAC上OpenGL环境搭建方法
参考文章:http://my.oschina.net/rockbaby/blog/102732
Mac下搭建OpenGL环境很简单,建立一个Command Li...
说起编程作图,大概还有很多人想起TC的#include 吧?
但是各位是否想过,那些画面绚丽的PC游戏是如何编写出来的?就靠TC那可怜的640*480分辨率、16色来做吗?显然是不行的。
Xcode是Mac OS X中自带的开发环境。GLUT提供了跨越平台的OpenGL实用工具。本文讲述如何在Xcode 3.2中使用GLUT来开发OpenGL应用程序。
在Xcode的启动面板中,...
Mac OS X平台通用步骤:
1. 链接GLUT.framwork和OpenGL.framwork库。
2. 导入GLEW扩展库。基于xcode7的示例:
1. 链接GLUT.framwork...
opengl 配置
Mac上搭建OpenGL环境比较简单,我使用的是操作系统是Mac OS X 10.10版本,OpenGL版本是3.0/3.1,开发环境XCode6.1,参考的书籍《OpenGL编程指南》中文第七版。
创建opengl项目步骤1,创建一个Cocoa的项目。
2,将项目中的AppDelegate.h,AppDelegate.m,MainMenu.xib文件删除。然后将main.m文件删除,改为mai...
他的最新文章
讲师:王禹华
讲师:宋宝华
您举报文章:
举报原因:
原文地址:
原因补充:
(最多只允许输入30个字)因为这几天一直在忙着图形学的实验,发现真的是很蛋疼,因为课程还在使用glut等openGL1.x版本的固定管线API,但是我感觉这和我们计算机行业追求新技术的宗旨完全不同啊,现在openGL2.x已经完全不支持固定管线了,只支持可编程管线,而且国内的资料真的很乱,想要找一些老版openGL的相关资料已经很难了,大部分都是可编程管线API的教程。所以,这篇文章就先记录一下我通过网络上的资料学习到的如何在Xcode中搭建GLEW+GLFW的环境。
首先在mac中要安装glew和glfw,这里我安装这两个工具使用的是homebrew包管理,这东西超级好用,安装命令很简单,大家可以去网络上找一下很多资料的。安装完homebrew之后,用以下两个命令安装glew和glfw:
brew install glew
brew install glfw3
我自己在安装遇到的问题就是,在安装完glew之后,可能会有个警告就是告诉你glew还没有link,这时候你只要执行:
brew link glew
就可以完成link,警告也会消失。通过brew命令安装的软件包都会在
/usr/local/Cellar
文件夹下面,在这里你就可以找到glew和glfw3两个文件夹。
接下来就是在Xcode中的配置,在Xcode中找到Peference菜单项,这个一般在File菜单项左边的那个Xcode项目中,然后在里面找到Locations项,再点击Custom Paths,添加四项,依次为:
Display Name
glew_header
glew_header
/usr/local/Cellar/glew/2.0.0/include
/usr/local/Cellar/glew/2.0.0/lib
glfw_header
glfw_header
/usr/local/Cellar/glfw3/3.2.1/include
/usr/local/Cellar/glfw3/3.2.1/lib
在这里需要修改的只有一项,就是要根据你自己安装的glew的版本和glfw3的版本修改2.0.0和3.2.1这两个版本号,其他的都是一样的。
然后创建一个新的Xcode项目(command line tool),语言选择C++。接着,在项目的Bulid Settings里面找到Header Search Paths和Library Search Paths两项,在Header Search Paths中加入
$(glew_header) $(glfw_header)
这两项,$()显示的就是刚才我们在locations中配置的文件路径,同理也在Library Search Paths中加入
$(glew_lib) $(glfw_lib)
添加之后效果图如下所示
完成了上述过程之后,还有最后一步,就是导入framework。在项目的General中找到Linked Frameworks and Libraries,点击‘+’号,添加如下三个文件
OpenGL.framework
libGLEW.2.0.0.dylib
libglfw3.3.2.dylib
添加两个dylib文件的方法是,在你没有在framework中搜索到这两个文件时,点击add other,然后点击shift+command+G进入/usr/local文件夹,然后根据我们之前说的安装glew和glfw3的路径找到这两个文件夹,在这两个文件夹中找到这两个文件,当然这两个文件可能和我图中给出的文件名不同还是因为安装的版本号不同,这个需要注意一下。
环境配置好之后我们可以测试一下,测试如下代码:
#include &iostream&
#include &GL/glew.h&
#include &GLFW/glfw3.h&
void Render(void)
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
glClear(GL_COLOR_BUFFER_BIT);
glBegin(GL_TRIANGLES);
glColor3f(1.0,0.0,0.0);
glVertex2f(0, .5);
glColor3f(0.0,1.0,0.0);
glVertex2f(-.5,-.5);
glColor3f(0.0, 0.0, 1.0);
glVertex2f(.5, -.5);
int main(int argc, const char * argv[]) {
GLFWwindow*
if(!glfwInit()){
return -1;
win = glfwCreateWindow(640, 480, &OpenGL Base Project&, NULL, NULL);
glfwTerminate();
exit(EXIT_FAILURE);
if(!glewInit())
return -1;
glfwMakeContextCurrent(win);
while(!glfwWindowShouldClose(win)){
glfwSwapBuffers(win);
glfwPollEvents();
glfwTerminate();
exit(EXIT_SUCCESS);
那么配置正确情况下,运行后会得到如下图形
环境成功配置后,就可以继续进行其他openGL资料的学习了。
文中用到的一些资料在这里统一附上链接,并表示感谢。
from:&//how-to-configure-GLEW-GLFW-on-your-mac/
With Xcode on OS X
If you are using the dynamic library version of GLFW, simply add it to the project dependencies.
If you are using the static library version of GLFW, add it and the Cocoa, OpenGL, IOKit and CoreVideo frameworks to the project as dependencies. They can all be found in&/System/Library/Frameworks.
With command-line on OS X
It is recommended that you use&&when building
from the command line on OS X. That way you will get any new dependencies added automatically. If you still wish to build manually, you need to add the required frameworks and libraries to your command-line yourself using the&-l&and&-framework&switches.
If you are using the dynamic GLFW library, which is named&libglfw.3.dylib, do:
&cc -o myprog myprog.c -lglfw -framework Cocoa -framework OpenGL -framework IOKit -framework CoreVideo
If you are using the static library, named&libglfw3.a, substitute&-lglfw3&for&-lglfw.
Note that you do not add the&.framework&extension to a framework when linking against it from the command-line.
The OpenGL framework contains both the OpenGL and GLU APIs, so there is nothing special to do when using GLU. Also note that even though your machine may have&libGL-style
OpenGL libraries, they are for use with the X Window System and will&not&work with the OS X native version of GLFW.
http://www.glfw.org/docs/latest/build_guide.html#build_link_xcode
本文已收录于以下专栏:
相关文章推荐
xcode运行OpenGL程序运行环境的搭建我是使用的xcode7.3.1,最近入门图形学,所以必须在电脑上跑一跑程序,网上写了一些关于xcode运行OpenGL程序的例子,都是相互抄袭的,还是xco...
Nehe的OpenGL教程,准备从把所有教程边学边翻写,开发平台是 Mac + Xcode + c。
一、MAC上OpenGL环境搭建方法
参考文章:http://my.oschina.net/rockbaby/blog/102732
Mac下搭建OpenGL环境很简单,建立一个Command Li...
说起编程作图,大概还有很多人想起TC的#include 吧?
但是各位是否想过,那些画面绚丽的PC游戏是如何编写出来的?就靠TC那可怜的640*480分辨率、16色来做吗?显然是不行的。
Xcode是Mac OS X中自带的开发环境。GLUT提供了跨越平台的OpenGL实用工具。本文讲述如何在Xcode 3.2中使用GLUT来开发OpenGL应用程序。
在Xcode的启动面板中,...
Mac OS X平台通用步骤:
1. 链接GLUT.framwork和OpenGL.framwork库。
2. 导入GLEW扩展库。基于xcode7的示例:
1. 链接GLUT.framwork...
由于Mac OS X10.10已经不再鼓励使用GLUT库进行OpenGL开发,缘由是因为其实在太老了其项目组已经停止更新好多年,而且不怎么安全,现在可供OpenGL开发的窗口一键式扩展库主要有,fre...
这是 Google 对 //glfw/ 的缓存。
这是该网页在 日 16:17:17 GMT 的快照。
当前页在此期间...
opengl 配置
Mac上搭建OpenGL环境比较简单,我使用的是操作系统是Mac OS X 10.10版本,OpenGL版本是3.0/3.1,开发环境XCode6.1,参考的书籍《OpenGL编程指南》中文第七版。
他的最新文章
讲师:钟钦成
讲师:宋宝华
您举报文章:
举报原因:
原文地址:
原因补充:
(最多只允许输入30个字)10个迷惑新手的Cocoa&Objective-c开发问题 - CSDN博客
10个迷惑新手的Cocoa&Objective-c开发问题
1. language background
2. runtime(运行时)
id obj=self;
if ([obj respondsToSelector:@selector(function1:)) {
if ([obj isKindOfClass:[NSArray class]] ) {
if ([obj conformsToProtocol:@protocol(myProtocol)]) {
if ([[obj class] isSubclassOfClass:[NSArray class]]) {
[obj someNonExistFunction];
id knife=[Knife grateKnife];
NSArray *monsterList=[NSArray array];
[monsterList makeObjectsPerformSelector:@selector(killMonster:) withObject:knife];
/********************************************************************
objc_msgSend(id self,
* On entry: a1 is the message receiver,
a2 is the selector
********************************************************************/
ENTRY objc_msgSend
# check whether receiver is nil
# save registers and load receiver's class for CacheLookup
sp!, {a4,v1-v3}
v1, [a1, #ISA]
# receiver is non-nil: search the cache
CacheLookup a2, LMsgSendCacheMiss
# cache hit (imp in ip) - prep for forwarding, restore registers and call
/* set nonstret (eq) */
sp!, {a4,v1-v3}
# cache miss: go search the method lists
LMsgSendCacheMiss:
sp!, {a4,v1-v3}
_objc_msgSend_uncached
LMsgSendExit:
END_ENTRY objc_msgSend
_objc_msgSend_uncached:
# Push stack frame
sp!, {a1-a4,r7,lr}
r7, sp, #16
# Load class and selector
a1, [a1, #ISA]
/* class = receiver-&isa
/* selector already in a2 */
# Do the lookup
MI_CALL_EXTERNAL(__class_lookupMethodAndLoadCache)
# Prep for forwarding, Pop stack frame and call imp
/* set nonstret (eq) */
RESTORE_VFP
sp!, {a1-a4,r7,lr}
1. 对系统整体结构认识模糊
2. thread和Reference Counting内存管理造成的问题。
3. 线程安全
4. Asynchronous(异步) vs. Synchronous(同步)
5. thread和run-loop
4. run-loop
- (void)myThread:(id)sender
NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];
while (TRUE) {
//do some jobs
//break in some condition
usleep(10000);
[pool drain];
[pool release];
NSMutableArray *targetQueue;
NSMutableArray *actionQueue;
- (void)myThread:(id)sender
NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];
while (TRUE) {
//do some jobs
//break in some condition
int n=[targetQueue count];
assert(n==[actionQueue count]);
for(int i=0;i&n;i++){
id target=[targetQueue objectAtIndex:i];
SEL action=NSSelectorFromString([actionQueue objectAtIndex:i]);
if ([target respondsToSelector:action]) {
[target performSelector:action withObject:nil];
usleep(10000);
[pool drain];
[pool release];
while (TRUE) {
//break in some condition
@interface MyNSTimer : NSObject
id target;
SEL action;
float interval;
CFAbsoluteTime lasttime;
- (void)invoke;
@implementation MyNSTimer
- (void)invoke;
if ([target respondsToSelector:action]) {
[target performSelector:action withObject:nil];
@interface MyNSRunloop : NSObject
NSMutableArray *timerQueue;
- (void)addTimer:(MyNSTimer*)t;
- (void)executeOnce;
@implementation MyNSRunloop
- (void)addTimer:(MyNSTimer*)t;
@synchronized(timerQueue){
[timerQueue addObject:t];
- (void)executeOnce;
CFAbsoluteTime currentTime=CFAbsoluteTimeGetCurrent();
@synchronized(timerQueue){
for(MyNSTimer *t in timerQueue){
if(currentTime-t.lasttime&t.interval){
t.lasttime=currentTime;
[t invoke];
@interface MyNSThread : NSObject
MyNSRunloop *runloop;
- (void)main:(id)sender;
@implementation MyNSThread
- (void)main:(id)sender
NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];
while (TRUE) {
//do some jobs
//break in some condition
[runloop executeOnce];
usleep(10000);
[pool drain];
[pool release];
MyNSTimer *timer1=[MyNSTimer scheduledTimerWithTimeInterval:1
target:obj1
selector:@selector(download1:)];
[myThread1.runloop addTimer:timer1];
MyNSTimer *timer2=[MyNSTimer scheduledTimerWithTimeInterval:2
target:obj2
selector:@selector(download2:)];
[myThread1.runloop addTimer:timer2];
5. delegate, protocol
6. event respon-der
LXTextField.h
lxtextfield
Created by xu lian on 12-03-09.
Copyright (c) 2012 Beyondcow. All rights reserved.
#import &AppKit/AppKit.h&
@interface LXTextField : NSView
NSString *stringValue;
BOOL selectAll;
@property(retain,nonatomic) NSString *stringValue;
LXTextField.m
lxtextfield
Created by xu lian on 12-03-09.
Copyright (c) 2012 Beyondcow. All rights reserved.
#import &LXTextField.h&
@implementation LXTextField
@synthesize stringValue;
- (void)awakeFromNib
selectAll = NO;
- (id)initWithFrame:(NSRect)frameRect
if( self = [super initWithFrame:frameRect] ){
selectAll = NO;
return self;
- (BOOL)acceptsFirstResponder
return YES;
- (BOOL)becomeFirstResponder
return YES;
- (BOOL)resignFirstResponder
selectAll=NO;
[self setNeedsDisplay:YES];
return YES;
- (void)setStringValue:(NSString *)string{
stringValue = string;
[self setNeedsDisplay:YES];
- (void)drawRect:(NSRect)dirtyRect
if (selectAll) {
NSRect r = NSZeroRect;
r.size = [stringValue sizeWithAttributes:nil];
[[NSColor selectedControlColor] set];
NSRectFill(r);
[stringValue drawAtPoint:NSZeroPoint withAttributes:nil];
- (IBAction)selectAll:(id)sender;
selectAll=YES;
[self setNeedsDisplay:YES];
- (IBAction)copy:(id)sender;
NSPasteboard *pasteBoard = [NSPasteboard generalPasteboard];
[pasteBoard declareTypes:[NSArray arrayWithObjects:NSStringPboardType, nil] owner:nil];
[pasteBoard setString:stringValue forType:NSStringPboardType];
- (void)mouseDown:(NSEvent *)theEvent
if ([theEvent clickCount]&=2) {
selectAll=YES;
[self setNeedsDisplay:YES];
- (void)keyDown:(NSEvent *)theEvent
- (void)keyDown:(NSEvent *)theEvent
if(condition){
do something;
[super keyDown:theEvent];
7. mem-ory management
NSLog(@&%d&,[object retainCount]);
while([object retainCount]&0){
[object release];
8. class heritage and category
9. design pattern
10. Drawing Issues
Cocoa Drawing
[anImage drawInRect:rect fromRect:NSZeroRect operation:NSCompositeSourceOver fraction:1.0];
[@&some text& drawAtPoint:NSZeroPoint withAttributes:attrs];
NSBezierPath *p=[NSBezierPath bezierPathWithRect:rect];
[[NSColor redColor] set];
本文已收录于以下专栏:
相关文章推荐
10个迷惑新手的Cocoa,Objective-c开发难点和问题
首先请谅解我可能使用很多英文,毕竟英文资料将来会是你的主要资料来源。
在你继续深入学习之前,请停下脚步弄清这些问题。如果你...
首先请谅解我可能使用很多英文,毕竟英文资料将来会是你的主要资料来源。
这篇博客将描述一些我见到的众多Cocoa开发新手遇到的问题和障碍。在你继续深入学习MacOS编程之前,请停下脚步弄清这些问题...
September, :52 AM PDT
update on 09 Fri Mar, :24 AM PDT
原文地址:http://lianxu.me/-cocoa-objc-newbie-problems/
25 Comments
language backgrou...
language background
del-e-gate, protocol
event respon-der
mem-ory man...
因为有自己的软件要开发,所以我只能忙里偷闲过来接着写这篇教程。大家莫怪。请按上面的最后更新时间算。
http://lianxu.me/blog//10-cocoa...
10个迷惑新手的Cocoa,Objective-c开发难点和问题
14 September, :52 AM PDT
Last update on 01 Web Fe...
原文地址:/newbie/basic/01.html
10个迷惑新手的Cocoa,Objective-c开发难点和问...
10个迷惑新手的Cocoa,Objective-c开发难点和问题
10个迷惑新手的Cocoa&Objective-c开发问题
NOV 14TH, 2011
languag...
10个迷惑新手的Cocoa,Objective-c开发难点和问题
September, :52 AM PDT
Last update on Fri 2...
他的最新文章
讲师:王禹华
讲师:宋宝华
您举报文章:
举报原因:
原文地址:
原因补充:
(最多只允许输入30个字)10个迷惑新手的Cocoa&Objective-c开发问题 - CSDN博客
10个迷惑新手的Cocoa&Objective-c开发问题
首先请谅解我可能使用很多英文,毕竟英文资料将来会是你的主要资料来源。
这篇教程将描述一些我见到的众多Cocoa开发新手遇到的问题和障碍。并不会手把手教你:“这个函数什么意思,哪个函数如何使用”,而是站在一定高度,统观各种技术所处的角色,让你不会迷失在各种技术细节中。在你继续深入学习MacOS编程之前,请停下脚步看清楚这些问题。如果你是新手,这个教程不要希望一次能看的非常透彻,学一定阶段反回来再看看又会有新的体会的。
1. language background
首先c语言背景,必须。 很多人问:“没有任何语言基础,我不想学c直接学objective-c。是否可以?” 这里我简单说几句,objc是c的超集,也就是说大部分objc代码其实是c、而且众多传统开源项目都是c写成的。你不学好c在unix世界里只能是个二流开发者!也许说得过于严厉,不过自己斟酌把。c++呢大概了解一下即可,因为它太庞杂了。
接着English,必须。 苹果不会把它们文档都写成中文的。“什么,有人翻译?” 等有人闲着翻译出来了的时候,大家都已经卖了很多软件了。你也是跟着人家屁股后面做开发。
2. runtime(运行时)
Objective-c是动态语言, 很多新手或者开发人员常常被runtime这个东西所迷惑。而恰恰这是一个非常重要的概念。我可以这么问:“如果让你(设计)实现一个计算机语言,你要如何下手?” 很少程序员这么思考过。但是这么一问,就会强迫你从更高层次思考以前的问题了。
注意我这句话‘设计’括起来了,稍微次要点,关键是实现。
我把实现分成3种不同的层次:
第一种是传统的面向过程的语言开发,例如c语言。实现c语言编译器很简单,只要按照语法规则实现一个LALR语法分析器就可以了,编译器优化是非常难的topic,不在这里讨论范围内,忽略。 这里我们实现了编译器其中最最基础和原始的目标之一就是把一份代码里的函数名称,转化成一个相对内存地址,把调用这个函数的语句转换成一个jmp跳转指令。在程序开始运行时候,调用语句可以正确跳转到对应的函数地址。 这样很好,也很直白,但是太死板了。Everything is predetermined.
我们希望语言更加灵活,于是有了第二种改进,开发面向对象的语言,例如c++。 c++在c的基础上增加了类的部分。但这到底意味着什么呢?我们再写它的编译器要如何考虑呢?其实,就是让编译器多绕个弯,在严格的c编译器上增加一层类处理的机制,把一个函数限制在它处在的class环境里,每次请求一个函数调用,先找到它的对象, 其类型,返回值,参数等等,确定了这些后再jmp跳转到需要的函数。这样很多程序增加了灵活性同样一个函数调用会根据请求参数和类的环境返回完全不同的结果。增加类机制后,就模拟了现实世界的抽象模式,不同的对象有不同的属性和方法。同样的方法,不同的类有不同的行为!
这里大家就可以看到作为一个编译器开发者都做了哪些进一步的思考。虽然面相对象语言有所改进,但还是死板, 我们仍然叫c++是static language.
希望更加灵活!于是我们完全把上面哪个类的实现部分抽象出来,做成一套完整运行阶段的检测环境,形成第三种,动态语言。这次再写编译器甚至保留部分代码里的sytax名称,名称错误检测,runtime环境注册所以全局的类,函数,变量等等信息等等,我们可以无限的为这个层增加必要的功能。调用函数时候,会先从这个运行时环境里检测所以可能的参数再做jmp跳转。这,就是runtime。编译器开发起来比上面更加弯弯绕。但是这个层极大增加了程序的灵活性。 例如当调用一个函数时候,前2种语言,很有可能一个jmp到了一个非法地址导致程序crash,
但是在这个层次里面,runtime就过滤掉了这些可能性。 这就是为什么dynamic langauge更加强壮。 因为编译器和runtime环境开发人员已经帮你处理了这些问题。
好了上面说着这么多,我们再返回来看objective-c的这些语句:
12345678910id obj=self;if ([obj respondsToSelector:@selector(function1:)) {}if ([obj isKindOfClass:[NSArray class]] ) {}if ([obj conformsToProtocol:@protocol(myProtocol)]) {}if ([[obj class] isSubclassOfClass:[NSArray class]]) {}[obj someNonExistFunction];看似很简单的语句,但是为了让语言实现这个能力,语言开发者要付出很多努力实现runtime环境。这里运行时环境处理了弱类型、函数存在检查工作。runtime会检测注册列表里是否存在对应的函数,类型是否正确,最后确定下来正确的函数地址,再进行保存寄存器状态,压栈,函数调用等等实际的操作。
knife=[Knife
grateKnife];
*monsterList=[NSArray
[monsterList
makeObjectsPerformSelector:@selector(killMonster:)
withObject:knife];
用c,c++完成这个功能还是比较非常麻烦的,但是动态语言处理却非常简单并且这些语句让objc语言更加intuitive。
在Objc中针对对象的函数调用将不再是普通的函数调用,[obj function1With:var1];&这样的函数调用将被运行时环境转换成objc_msgSend(target,@selector(function1With:),var1);。Objc的runtime环境是开源的,所以我们可以拿出一下实现做简单介绍,可以看到objc_msgSend由汇编语言实现,我们甚至不必阅读代码,只需查看注释就可以了解,运行时环境在函数调用前做了比较全面的安全检查,已确保动态语言函数调用不会导致程序crash。对于希望深入学习的朋友可以自行Objc-runtime源代码来阅读,这里就不再深入讲解。
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758/******************************************************************** * id&&&&&& objc_msgSend(id self, *&&&&&&&&&&SEL op, *&&&&&&&&&&...) * * On entry: a1 is the message receiver, *&&&&&&&&&& a2 is the selector ********************************************************************/&&&&&ENTRY objc_msgSend# check whether receiver is nil&&&&teq&&&& a1, #0&&&&moveq&& a2, #0&&&&bxeq&&&&lr&# save registers and load receiver's class for CacheLookup&&&&stmfd&& sp!, {a4,v1-v3}&&&&ldr&&&& v1, [a1, #ISA]&# receiver is non-nil: search the cache&&&&CacheLookup a2, LMsgSendCacheMiss&# cache hit (imp in ip) - prep for forwarding, restore registers and call&&&&teq v1, v1&&&&&&/* set nonstret (eq) */&&&&ldmfd&& sp!, {a4,v1-v3}&&&&bx&&&&&&ip&# cache miss: go search the method listsLMsgSendCacheMiss:&&&&ldmfd&& sp!, {a4,v1-v3}&&&&b&& _objc_msgSend_uncached&LMsgSendExit:&&&&END_ENTRY objc_msgSend&&&&&&.text&&&&.align 2_objc_msgSend_uncached:&# Push stack frame&&&&stmfd&& sp!, {a1-a4,r7,lr}&&&&add&&&& r7, sp, #16&&&&SAVE_VFP&# Load class and selector&&&&ldr a1, [a1, #ISA]&&&&&&/* class = receiver-&isa&&*/&&&&# MOVE&&a2, a2&&&&&&&&&&/* selector already in a2 */&# Do the lookup&&&&MI_CALL_EXTERNAL(__class_lookupMethodAndLoadCache)&&&&MOVE&&&&ip, a1&# Prep for forwarding, Pop stack frame and call imp&&&&teq v1, v1&&&&&&/* set nonstret (eq) */&&&&RESTORE_VFP&&&&ldmfd&& sp!, {a1-a4,r7,lr}&&&&bx&&ip现在说一下runtime的负面影响: 1. 关于执行效率问题。 “静态语言执行效率要比动态语言高”,这句没错。因为一部分cpu计算损耗在了runtime过程中,而从上面的汇编代码也可以看出,大概损耗在哪些地方。而静态语言生成的机器指令更简洁。正因为知道这个原因,所以开发语言的人付出很大一部分努力为了保持runtime小巧上。所以objecitve-c是c的超集+一个小巧的runtime环境。 但是,换句话说,从算法角度考虑,这点复杂度不算差别的,Big O notation结果不会有差别。( It’s not log(n) vs n2) 2. 另外一个就是安全性。动态语言由于运行时环境的需求,会保留一些源码级别的程序结构。这样就给破解带来的方便之门。一个现成的说明就是,java,大家都知道java运行在jre上面。这就是典型的runtime例子。它的执行文件.class全部可以反编译回近似源代码。所以这里的额外提示就是如果你需要写和安全有关的代码,离objc远点,直接用c去。简单理解:“Runtime is everything between your each function call.”但是大家要明白,第二点我提到runtime并不只是因为它带来了这些简便的语言特性。而是这些简单的语言特性,在实际运用中需要你从完全不同的角度考虑和解决问题。只是计算1+1,很多语言都是一样的,但是随着问题的复杂,项目的增长,静态语言和动态语言就会演化出完全不同的风景。3. thread“thread synchronization another notorious trouble!”记得上学时候学操作系统这门课,里面都会有专门一章介绍任务调度和生产者消费者的问题。 这就是为今后使用进程、线程开发打基础。概念很简单,但难点在synchronization(同步),因为死锁检测算法不是100%有效,否则就根本没有死锁这个说法了。另一个原因是往往这类错误很隐晦,静态分析很难找到。同时多线程开发抽象度较高需要经验去把握。总体来说,我见到的在这方面的问题可以分为一下几点:1. 对系统整体结构认识模糊不知道多线程开发的几个基点,看别人代码越看越糊涂的。一会NSThread,一会Grand Central Dispatch、block,一会又看到了pthread等等。Apple封装了很多线程的API, 多线程开发的基本结构入下图:Mac OS Thread Architecture可以看到在多线程开发中你可以选择这上面这4种不同的方式。Mach是核心的操作系统部分。其实这个我也不是非常熟悉,至少我还没有读到过直接使用mach做多线程的代码。pthread(POSIX Threads)是传统的多线程标准,灵活、轻巧,但是需要理论基础,开发复杂。需要注意一点,根据,在Cocoa下使用pthread需要先启动至少一个NSThread,确定进入多线程环境后才可以。是Mac OS 10.0后发布的多线程API较为高层,但是缺乏灵活性,而且和pthread相比效率低下。&是10.6后引入的开源多线程库,它介于pthread和NSThread之间。比NSThread更灵活、小巧,并且不需要像pthread一样考虑很多lock的问题。同时objective-c 2.0发布的新语法特性之一blocks,也正是根据Grand Central Dispatch需求推出的。所以在你写多线程代码或者阅读多线程代码时候,心理要先明确使用哪种技术。2. thread和Reference Counting内存管理造成的问题。线程里面的方法都要放到NSAutoreleasePool里面吗?这类问题很常见,迷惑的原因是不明白 NSAutoreleasePool 到底是干什么用的。NSAutoreleasePool跟thread其实关系并不显著,它提供一个临时内存管理空间,好比一个沙箱,确保不会有不当的内存分配泄露出来,在这个空间内新分配的对象要向这个pool做一下注册告诉:“pool,我新分配一块空间了”。当pool drain掉或者release,它里面分配过的内存同样释放掉。可见和thread没有很大关系。但是,我们阅读代码的时候经常会看到,新开线程的函数内总是以NSAutoreleasePool开始结束。这又是为什么呢!? 因为thread内恰好是最适合需要它的地方! 线程函数应该计算量大,时间长(supposed to be heavy)。在线程里面可能会有大量对象生成,这时使用autoreleasepool管理更简洁。所以这里的答案是,不一定非要在线程里放NSAutoreleasePool,相对的在cocoa环境下任意地方都可以使用NSAutoreleasePool。如果你在线程内不使用NSAutoreleasePool,要记得在内部alloc和relase配对出现保证没有内存泄露。3. 线程安全每个程序都有一个主线程(main thread),它负责处理事件响应,和UI更新。更新UI问题。很多新手会因为这个问题,导致程序崩溃或出现各种问题。而且逛论坛会看到所以人都会这么告诉你:“不要在后台线程更新你的UI”。其实这个说法不严密,在多线程环境里处理这个问题需要谨慎,而且要了解线程安全特性。首先我们需要把“UI更新”这个词做一个说明,它可以有2个层次理解,首先是绘制,其次是显示。这里绘制是可以在任何线程里进行,但是要向屏幕显示出来就需要在主线程操作了。我举个例子说明一下,例如现在我们有一个NSImageView,里面设置了一个NSImage, 这时我想给NSImage加个变色滤镜,这个过程就可以理解为绘制。那么我完全可以再另外一个线程做这个比较费时的操作,滤镜增加完毕再通知NSImageView显示一下。另一个例子就是,Twitter客户端会把每一条微博显示成一个cell,但是速度很快,这就是因为它先对cell做了offscreen的渲染,然后再拿出来显示。所以通过这点我们也可以得到进一步的认识,合理设计view的更新是非常重要的部分。很多新手写得代码片段没错,只是因为放错了地方就导致整个程序出现各种问题。根据苹果线程安全摘要说明,再其它线程更新view需要使用lockFocusIfCanDraw和unlockFocus锁定,确保不会出现安全问题。另外还要知道常用容器的线程安全情况。immutable的容器是线程安全的,而mutable容器则不是。例如NSArray和NSMutableArray。4. Asynchronous(异步) vs. Synchronous(同步)我在一个view要显示多张web图片,我想问一下,我是应该采用异步一个一个下载的方式,还是应该采用多线程同时下载的方式,还是2个都用,那种方式好呢?实际上单独用这2个方法都不好。并不是简单的用了更多线程就提高速度。这个问题同时涉及客户端和服务器的情况。处理这种类型的程序,比较好的结构应该是:非主线程建立一个队列(相当于Asynchronous任务),队列里同时启动n个下载任务(相当于Synchronous任务)。这里的n在2~8左右就够了。这个结构下可以认为队列里面每n个任务之间是异步关系,但是这n个任务之间又是同步关系,每次同时下载2~8张图片。这样处理基本可以满足速度要求和各类服务器限制。5. thread和run-looprunloop是线程里的一部分,但我觉得有必要单独拿出来写,是因为它涉及的东西比较容易误解,而说明它的文章又不多。4. run-loopthread和runloop在以前,开发者根本不太当成一个问题。因为没有静态语言里runloop就是固定的线程执行loop。而现在Cocoa新手搞不明白的太多了,因为没有从动态角度看它,首先回想一下第2点介绍的runtime概念,接着出一个思考题。现在有一个程序片段如下:
(void)myThread:(id)sender
&&&&NSAutoreleasePool
*pool=[[NSAutoreleasePool
&&&&&&&&//do
&&&&&&&&//break
in some condition
&&&&&&&&usleep(10000);
&&&&&&&&[pool
现在要求,做某些设计,使得当这个线程运行的同时,还可以从其它线程里往它里面随意增加或去掉不同的计算任务。 这,就是NSRunloop的最原始的开发初衷。让一个线程的计算任务更加灵活。 这个功能在c,
c++里也许可以做到但是非常难,最主要的是因为语言能力的限制,以前的程序员很少这么去思考。
好,现在我们对上面代码做一个非常简单的进化:
1234567891011121314151617181920212223242526NSMutableArray *targetQueue;NSMutableArray *actionQueue;- (void)myThread:(id)sender{&&&&NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];&&&&while (TRUE) {&&&&&&&&&//do some jobs&&&&&&&&//break in some condition&&&&&&&&int n=[targetQueue count];&&&&&&&&assert(n==[actionQueue count]);&&&&&&&&for(int i=0;i&n;i++){&&&&&&&&&&&&id target=[targetQueue objectAtIndex:i];&&&&&&&&&&&&SEL action=NSSelectorFromString([actionQueue objectAtIndex:i]);&&&&&&&&&&&&if ([target respondsToSelector:action]) {&&&&&&&&&&&&&&&&[target performSelector:action withObject:nil];&&&&&&&&&&&&}&&&&&&&&}&&&&&&&&&usleep(10000);&&&&&&&&&[pool drain];&&&&}&&&&&[pool release];}注意,这里没有做线程安全处理,记住Mutable container is not thread safe. 这个简单的扩展,让我们看到了如何利用runtime能力让线程灵活起来。当我们从另外线程向targetQueue和actionQueue同时加入对象和方法时候,这个线程函数就有了执行一个额外代码的能力。有人会问,哪里有runloop? 那个是nsrunloop? 看不出来啊。
&&&&//break
in some condition
这个结构就叫线程的runloop, 它和NSRunloop这个类虽然名字很像,但完全不是一个东西。以前在使用静态语言开始时候,程序员没有什么迷惑,因为没有NSRunloop这个东西。 我接着来说,这个NSRunloop是如何来得。
第二段扩展代码里面确实没有NSRunloop这个玩意儿,我们接着做第3次改进。 这次我们的目的是把其中动态部分抽象出来。
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273@interface MyNSTimer : NSObject{&&&&id target;&&&&SEL action;&&&&float interval;&&&&CFAbsoluteTime lasttime;}- (void)invoke;@end&@implementation MyNSTimer- (void)invoke;{&&&&if ([target respondsToSelector:action]) {&&&&&&&&[target performSelector:action withObject:nil];&&&&}}@end&&#!objc@interface MyNSRunloop : NSObject{&&&&NSMutableArray *timerQueue;}- (void)addTimer:(MyNSTimer*)t;- (void)executeOnce;@end&@implementation MyNSRunloop- (void)addTimer:(MyNSTimer*)t;{&&&&@synchronized(timerQueue){&&&&&&&&[timerQueue addObject:t];&&&&}}- (void)executeOnce;{&&&&CFAbsoluteTime currentTime=CFAbsoluteTimeGetCurrent();&&&&@synchronized(timerQueue){&&&&&&&&for(MyNSTimer *t in timerQueue){&&&&&&&&&&&&if(currentTime-t.lasttime&t.interval){&&&&&&&&&&&&&&&&t.lasttime=currentTime;&&&&&&&&&&&&&&&&[t invoke];&&&&&&&&&&&&}&&&&&&&&}&&&&}}@end&&#!objc@interface MyNSThread : NSObject{&&&&MyNSRunloop *runloop;}- (void)main:(id)sender;@end&@implementation MyNSThread- (void)main:(id)sender{&&&&NSAutoreleasePool *pool=[[NSAutoreleasePool alloc] init];&&&&while (TRUE) {&&&&&&&&//do some jobs&&&&&&&&//break in some condition&&&&&&&&[runloop executeOnce];&&&&&&&&usleep(10000);&&&&&&&&[pool drain];&&&&}&&&&[pool release];}@end走到这里,我们就算是基本把Runloop结构抽象出来了。例如我有一个MyNSThread实例,myThread1。我可以给这个实例的线程添加需要的任务,而myThread1内部的MyNSRunloop对象会管理好这些任务。
*timer1=[MyNSTimer
scheduledTimerWithTimeInterval:1
target:obj1
selector:@selector(download1:)];
[myThread1.runloop
addTimer:timer1];
*timer2=[MyNSTimer
scheduledTimerWithTimeInterval:2
target:obj2
selector:@selector(download2:)];
[myThread1.runloop
addTimer:timer2];
当你看懂了上面的代码也许会感叹,‘原来是这么回事啊!为什么把这么简单的功能搞这么复杂呢?’ 其实就是这么回事,把Runloop抽象出来可以使得线程任务管理更加loose coupling,给设计模式提供更大的空间。这样第三方开发者不需要过深入的涉及线程内部代码而轻松管理线程任务。另外请注意,这里MyNSRunloop, MyNSTimer等类是我写得一个模拟情况,真实的NSRunloop实现肯定不是这么简单。这里为了说明一个思想。这种思想贯穿整个cocoa framework,从界面更新到event管理。
5. delegate, protocol
这个会列出来因为,我感觉问它的数量仅此于内存管理部分,它们用得很频繁,并且这些是设计模式的重要组成部分。
6. event respon-der
Interface Builder First Responder
使用过Xcode的开发者都知道Interface Builder这个开发组件,在Xcode4版本以后该组件已经和xcode整合到一起。它是苹果软件开发中非常重要的部分。ib为开发者减轻了很大一部分界面设计工作。但是其中有一个东西让新接触ib的开发者一头雾水,那就是First Responder, 它是什么东西,为何它会有那么多Actions。这节我会详细介绍如何理解Responder和Cocoa下的事件响应链。
First Responder在IB属性为Placeholders,这意味着它属于一个虚拟实例。就好比TextField里面的string placeholder一样,只是临时显示一下。真正的first responder会被其它对象代替。实际上,任何派生自NSResponder类的对象都可以替代First Responder。而First Responder里面的所有Actions就是NSResponder提供的接口函数,当然你也可以定义自己的响应函数。
MacOS在系统内部会维护一个称为“The Responder Chain”的链表。该列表内容为responder对象实例,它们会对各种系统事件做出响应。最上面的哪个对象就叫做first responder,它是最先接收到系统事件的对象。如果该对象不处理该事件,系统会将这个事件向下传递,直到找到响应事件的对象,我们可以理解为该事件被该这个对象截取了。
The Responder Chain基本结构如下图所示:
The Responder Chain
在理解了上面的概念之后,我希望使用一个例子让大家对responder有更加具体的认识。大家都知道NSTextField这个控件,它是最常见的控件之一。它最基本功能是显示一个字符串,如果启用可选,那么用户可以选中文本,拷贝文本,如果开启编辑选项,还可以运行用户编辑文本,等等基本操作。
下面展示给大家的例子是创建一个我们自己创建的简单textfield叫LXTextField。它不属于NSTextField而是派生自NSView,具有功能显示字符串,全选字符串,响应用户cmd+c的拷贝操作,三个基本功能。注意NSView派生自NSResponder。
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108////&&LXTextField.h//&&lxtextfield////&&Created by xu lian on 12-03-09.//&&Copyright (c) 2012 Beyondcow. All rights reserved.//&#import &AppKit/AppKit.h&&@interface LXTextField : NSView{&&&&NSString *stringValue;&&&&BOOL selectAll;}@property(retain,nonatomic) NSString *stringValue;&@end&&&#!objc////&&LXTextField.m//&&lxtextfield////&&Created by xu lian on 12-03-09.//&&Copyright (c) 2012 Beyondcow. All rights reserved.//&#import &LXTextField.h&&@implementation LXTextField@synthesize stringValue;&- (void)awakeFromNib{&&&&selectAll = NO;}&- (id)initWithFrame:(NSRect)frameRect{&&&&if( self = [super initWithFrame:frameRect] ){&&&&&&&&selectAll = NO;&&&&}&&&&return self;}&- (BOOL)acceptsFirstResponder{&&&&return YES;}&- (BOOL)becomeFirstResponder{&&&&return YES;}&- (BOOL)resignFirstResponder{&&&&selectAll=NO;&&&&[self setNeedsDisplay:YES];&&&&return YES;}&&- (void)setStringValue:(NSString *)string{&&&&stringValue = string;&&&&[self setNeedsDisplay:YES];}&- (void)drawRect:(NSRect)dirtyRect{&&&&if (selectAll) {&&&&&&&&NSRect r = NSZeroRect;&&&&&&&&r.size = [stringValue sizeWithAttributes:nil];&&&&&&&&[[NSColor selectedControlColor] set];&&&&&&&&NSRectFill(r);&&&&}&&&&[stringValue drawAtPoint:NSZeroPoint withAttributes:nil];}&- (IBAction)selectAll:(id)sender;{&&&&selectAll=YES;&&&&[self setNeedsDisplay:YES];}&- (IBAction)copy:(id)sender;{&&&&NSPasteboard *pasteBoard = [NSPasteboard generalPasteboard];&&&&[pasteBoard declareTypes:[NSArray arrayWithObjects:NSStringPboardType, nil] owner:nil];&&&&[pasteBoard setString:stringValue forType:NSStringPboardType];}&- (void)mouseDown:(NSEvent *)theEvent{&&&&if ([theEvent clickCount]&=2) {&&&&&&&&selectAll=YES;&&&&}&&&&[self setNeedsDisplay:YES];}&- (void)keyDown:(NSEvent *)theEvent{}&@end运行实例,可以看到随着LXTextField收到系统发送的becomeFirstResponder消息,LXTextField变成responder chain中的frist responder, 这时候可以理解为IB里的哪个First Responder虚拟实例被该LXTextField取代。这时候mainMenu上哪些菜单项,例如:全选(cmd+a), 拷贝(cmd+a)等事件都会最先发给当前这个LXTextField。一旦你的LXTextField实现了NSResponder的哪些默认函数,那么该对象就会截取系统事件。当然这些事件具体如何实现还是需要你自己写代码实现。例如这里的 – (IBAction)copy:(id) 显然我手动实现了textfield的copy能力。注意上述代码中我实现了一个空函数- (void)keyDown:(NSEvent *)theEvent 这意味着我们希望LXTextField截取键盘事件而不再传递给responder chain后续对象。当然,如果我们希望LXTextField响应特定键盘事件,而其他事件继续传给其他响应对象,我们可以编写如下代码。
(void)keyDown:(NSEvent
*)theEvent
&&&&if(condition){
&&&&&&&&do
something;
&&&&}else{
&&&&&&&&[super
keyDown:theEvent];
7. mem-ory management
内存管理问题,也许是问得最多的问题了吧。
开发速度不言而喻,你少写很多release代码,甚至很少去操心这部分。
执行速度呢?这个还要从runtime说起,还记得我在第2点说得一句话么:“Runtime is everything between your each function call.”
RC有一个古老的内存管理哲学:谁分配谁释放。&通过counting来确定一个资源有几个使用者。道理很简单,但是往往简单的东西人却会犯错。从来没有一个程序员可以充满信心的说,我写得代码从来没有过内存泄露。这样来看,我们就更需要让程序可以自己处理这个管理机制,这就需要把这个机制放到runtime里。
所以RC-&ARC就是把内存管理部分从普通开发者的函数中移到了函数外的runtime中。因为runtime的开发原型简单,逻辑层次更高,所以做这个开发和管理出错的概率更小。实际上编译器开发人员对这部分经过无数次测试,所以可以说用arc几乎不会出错。另外由于编译的额外优化,使得这个部分比程序员自己写得代码要快速很多。而且对于一些通用的开发模式,例如autorelease对象,arc有更优秀的算法保证autoreleasepool里的对象更少。
这个规则可以让NSObject决定是不是要释放内存。当一个对象alloc时候,系统分配其一块内存并且object自动计数retainCount=1 这时候每当[object retain]一次retainCount+1(这里虽然简写也是rc不过是巧合或者当时开发人员故意选的retain这个词吧)每次[object release]时候retainCount-1 当retainCount==0时候object就真正把这快内存还给系统。
当然了,我也做过类似的动作,那种希望一切尽在掌握中的心态。但是你会看到其他人告诉这么做完全没有意义,RC规则并不是这么用的。
首先,这个数字也许并不是你心目中的哪个。因为很难跟踪到底哪些地方引用的该资源。你建立的资源不光只有你的代码才会用到,你调用的各种Framework,Framework调用的Framework,都有可能改变这个资源的retainCount。
其次,这里每个数字意味着有其它对象引用该资源,这样的暴力释放很容易导致程序崩溃。就好比,其它人也许可以翻牌子把门口哪个牌子上的数字改变,但是这会出现问题。还有很多人在里面,把牌子变成0房间锁了结果谁也出不来。又或者,减少牌子上的数字,人进的过多房间变得过于拥挤。
所以去验证rc规则,或者单纯的改变retainCount并不是明智之举。你能做的就是理解规则,使用规则,读文档了解container的引用特性。或者干脆移到 Automatic Reference Counting (ARC) 上面。
我有一个NSMutableArray里面保存了1000个NSString对象,我在release的时候需要循环释放1000个string么?还是只需要release NSMutableArray。
就像上面提到的,如果你了解container的引用特性,这个问题自然就解决了。“NSMutableArray在添加、插入objects时会做retain操作。” 通过这一句话就分析出,用户不否需要帮助NSMutableArray释放1000个string。回忆上面提到的管理哲学,“谁分配谁释放” 编写NSMutableArray的程序员非常熟悉这个规则,NSMutableArray内部retain了,NSMutableArray自然要负责release。但是NSMutableArray才不会管你在外面什么地方引用了这1000个string,它只管理好内部的rc就够了。所以如果你在NSMutableArray外面对1000个string
retain了,你自然需要release。相应的,你作为创建这个NSMutableArray的程序员,你只管release这个NSMutableArray就可以了。
最后说一下不用arc的情况。目前情况来看,有不少第三方的库并未支持arc,所以如果你的旧项目使用了这些库,请检查是否作者发布了新版本,或者你需要自己修正支持arc。
8. class heritage, category and extensions
Objective-C 的 OOP 特性提供 subclass 和 category 这2个非常重要的部分。subclass 应该反复被各种编程书籍介绍过。它是 OOP 继承特性的关键语法,它给类添加了延续并且多样化自己的方法。可以说没有继承就没有 OOP 这玩意。而 category 相对于 subclass 就不那么出名了。其实 category 思想出世于 smalltalk,所以它不能算是一个新生事物。
先说一下这2个特性最主要的区别。简单可以这么理解,subclass 体现了类的上下级关系,而 category 是类间的平级关系。
Subclass and Category
如上图所示,左侧是subclass,可以看到class, subclass1, subclass2是递进关系。同时下面的子类完全继承父类的方法,并且可以覆盖父类的方法。subclass2拥有function1,function2,function3三个函数方法。function1的执行代码来自subclass1, function2的执行代码来自于subclass2。
右侧是category。可以看到,无论如何扩展类的category,最终就只有一个类class。category可以说是类的不同方法的小集合,它把一个类的方法划分成不同的区块。请注意观察,每个category块内的方法名称都没有重复的。这正是category的重要要求。
经过上面简单解释了解了这2点的基本区别,现在深入说一下category。
在Objective-c语言设计之初一个主要的哲学观点就是尽量让一个程序员维护庞大的代码集。(对于庞大的项目‘原则’和‘协议’是非常重要的东西。甚至编写良好的文件名都是非常重要的开发技巧)根据结构化程序设计的经验出发,把一个大块代码划分成一些小块的代码更便于程序员管理。于是objc借用了smalltalk的categories概念。允许程序员把一系列功能相近的方法组织到一个单独的文件内,使得这些代码更容易识别。
更进一步的,和c,c++这种静态语言相比。objc把class categories功能集成到了run-time里面。因此,objc的categories允许程序员为已经存在的类添加新的方法而不需要重新编译旧的类。一旦一个category加入,它可以访问该类所有方法和实例变量,包括私有变量。
category不仅可以为原有class添加方法,而且如果category方法与类内某个方法具有同样的method signature,那么category里的方法将会替换类的原有方法。这是category的替换特性。利用这个特性,category还可以用来修复一些bugs。例如已经发布的Framework出现漏洞,如果不便于重新发布新版本,可以使用category替换特性修复漏洞。另外,由于category有run-time级别的集成度,所以使得cocoa程序安全性有所下降。许多黑客就是利用
category、posting、Method Swizzling 等方法破解软件,或者为软件增加新功能。一个很典型的例子就是,我原来发布的QQ表情管理器(目前已经不再维护)。
值得注意的一点是,由于一个类的categories之间是平级关系。所以如果不同categories拥有相同的方法,这个调用结果是未知的:
Category methods should not override existing methods (class or instance). Two different categories implementing the same method results in undefined behavior.
(因为posting、Method Swizzling这个话题有些深入,本文里我就不介绍了。有兴趣自行Google)
Objc中Categories有其局限的部分,就是你不能为原有的class添加变量,只能添加方法。当然方法里可以添加局部变量。在这个局限基础上就有其它语言做了进一步改进,例如TOM语言就为Categories增加了添加类变量的能力。
自从Objc 2.0以后,语言引入了一个新的特性叫做 Class Extensions, 它可以看做是一类特殊的 category,可以给原有类增加新的属性和方法。
通过介绍我们可以看出,如果
categories 是为类增加外部方法的话,那么 extensions 就是用做类的内部拓展。
Class extensions 的外观很简单,就是一个 Category 后面括号内的名字为空:
12@interface ClassName ()@end接下来,你就可以给你的类里添加属性,方法了:
@interface
_someCustomInstanceVariable;
*extraProperty;
(void)assignUniqueIdentifier;
Class extensions 常用来定义类的私有变量和方法。
总上所属,如果你开发时候遇到无论如何都需要为类添加变量的情况,最好的选择就是subclass。相反如果你只希望增加一些函数簇。Categories是最好的选择。而类内部需要用到的私有变量和方法则最好写在 Class extensions 里。
Categories关注的重心是代码设计,把不同功能的方法分离开。在Objc里因为Categories是runtimes级别的特性,所以这种分离不仅体现在源码结构上,同时体现在运行时过程中。这意味着一个category里的方法在程序运行中如果没有被调用,那么它就不会被加载到内存中。所以合理的使用categories会减少你的程序内存消耗。
所以我个人给大家的建议是,每个Cocoa程序员都应该收集整理自己的一套NS类函数的Categories扩展库。这对你今后程序开发效率和掌控情况都有很大提高。
9. Drawing Issues
大家知道,MacOS 是一个非常注重UI的系统。所以在 MacOS 编程里绘制是一个非常重要的部分。第9部分,我会介绍 MacOS 下绘制编程。
从绘制技术分类上看,Cocoa程序员能接触的几种绘制技术列表如下:
在这里我不打算给大家介绍如何绘制具体的按钮或者表格。只是介绍一下,它们的代码风格,优势和限制。
Cocoa Drawing
Cocoa Drawing应该是学习Cocoa程序开发最先接触的绘制技术。也是目前大多数MacOS程序所使用的绘制技术,其底层使用Quazrtz 2D(Core Graphics)。苹果对应文档为&。Cocoa Drawing并没有统一的绘制函数,所有绘制函数分散在几个主要的NS类的下面。例如, NSImage, NSBezierPath, NSString, NSAttributedString, NSColor, NSShadow,NSGradient …
所以很简单,当你看到如下代码就可以判断,使用的是Cocoa Drawing方法
1234567[anImage drawInRect:rect fromRect:NSZeroRect operation:NSCompositeSourceOver fraction:1.0];&[@&some text& drawAtPoint:NSZeroPoint withAttributes:attrs];&NSBezierPath *p=[NSBezierPath bezierPathWithRect:rect];[[NSColor redColor] set];[p fill];这种代码多出现在NSView的drawRect函数内。Cocoa Drawing 的渲染上下文是 NSGraphicsContext,我不断的看到很多新手把 NSGraphicsContext 和 CoreGraphics 的 CGContextRef 搞混。虽然它们很像并且也确实是有关系的,不过如果你不了解当绘制时候的 render context 很多时候将得到一个空白页面的结果。Core GraphicsCore Graphics 是 Cocoa Drawing layer 的底层技术,在 iOS 开发中非常普遍,因为 iOS 系统中并不存在 Cocoa layer 所以网上可以找到的多是 Core Graphics 绘制代码段子,这给那些不了解 Mac 开发的新手来说造成了很大困扰。Cocoa 是 Mac OS 下的 application framework 而 iOS 下的 application framework 则是 UIKit.framework又叫 Cocoa Touch,它们分享部分代码基础但又不完全一样。例如,Cocoa Touch 下的 UIView 的渲染上下文会使用 UIGraphicsGetCurrentContext() 取得,它得到的是一个 CGContextRef 指针,而在 NSView 里多用 [NSGraphicsContext currentContext] 取得渲染上下文。它得到的是一个 NSGraphicsContext 对象。当然 NSView 里也可以通过 CGContextRef ctx = [[NSGraphicsContext currentContext] graphicsPort]; 来取得一个 Core Graphics 渲染上下文。 可见 Mac OS 下的开发更为灵活一些。因为 iOS 中的 UIKit 开发初期就瞄准了显卡硬件加速,所有 UIView 都是默认 layer-backed 的。iOS 开发者必须使用 Core Graphics 和 Core Animation 这几个相对底层的绘制技术。请看下面等价代码,作用是绘制一个白色矩形。但是分别使用 Core Graphics 和 Cocoa Drawing:
CGContextSetFillColor(cgContextRef,
CGContextSetBlendMode(cgContextRef,
kCGBlendModeNormal);
CGContextFillRect(cgContextRef,
CGRectMake(0,
whiteColor]
NSRectFillUsingOperation(NSMakeRect(0,
NSCompositeSourceOver);
可以看出,这是2种风格完全不同的绘制技术。Cocoa Drawing 是分散式的绘制函数,而 Core Graphics 是传统的类似 OpenGL 的集成式的绘制方式。其实 Cocoa Drawing 下层是 Core Graphics, Core Graphics 的下层是 OpenGL。
在 OSX 下 NSGraphicsContext 和 CGContextRef 大部分时候是可以相互转换的。
NSGraphicsContext 到 CGContextRef:
1234CGContextRef ctx = [[NSGraphicsContext currentContext] graphicsPort];CGContextSaveGState(ctx);//Core Graphic drawing code hereCGContextRestoreGState(ctx); CGContextRef 到 NSGraphicsContext:
NSGraphicsContext
[NSGraphicsContext
graphicsContextWithGraphicsPort:cgContextRef
flipped:NO];
[NSGraphicsContext
saveGraphicsState];
[NSGraphicsContext
setCurrentContext:ctx];
//cocoa drawing code here
[NSGraphicsContext
restoreGraphicsState];
大部分时候使用 Cocoa Drawing 可以绘制出需要的效果,但是某些特殊时候需要 Core Graphic 绘制,例如一些特殊的阴影,clip效果,自定义pattern phase,blending style等等。
Core Animation
如果说 Core Graphics 和 Cocoa Drawing 是通用的 UI 绘制框架的话,那么 CA 显然是界面动画绘制的高级技术。 Core Animation 的对应 Cocoa Animation 部分应该是 NSAnimation 和 NSViewAnimation,但这2个差距比较大。NSAnimation 出现与 OS X 10.4,Core Animation 是 10.5 后出现的。NSViewAnimation 功能和使用相对简单。
简单来说,Core Animation 的作用对象是 CALayer, NSAnimation 的作用对象是 NSView。了解你的程序界面是在处理那种对象很重要。
Core Image
对于这个绘制技术,这篇文章给了我很多启示大家也可以看看。&虽然是一篇 note 但是,记录了很多实际应用中的经历,可以对个各种绘制技术有一个比较全面的解析。
根据此文的介绍。Core Image 适合处理小量大图,而非常不适合处理大量小图。因为 CI 利用 GPU 运算,而数据到 GPU 的round-trip 时间数量级在 millisecond。这就意味着,1000 张小图分别再 GPU 运算,时间至少再 1000*1 ms。此文作者尝试绘制3000张小图片,利用 Cocoa Drawing 原本耗时 750ms,但是改用CI后耗时猛增到3秒。
所以,这就是CI在osx绘制技术里所处的宏观角色:单图做实时处理。
10. design pattern
:这里其实很有意思,为何我用“更高层次思考”,而不是“更底层次”。作为一个编译器和语言开发人员,面对的问题确实更底层没错,但是他们思考的维度更高,更抽象,这样子。一个不算恰当的比方就好像一个三维世界的人处理二维世界的一条线的问题。
:Posting技术在10.5以后deprecated,并且64bit run-time也不再支持
from&http://lianxu.me/-cocoa-objc-newbie-problems/
本文已收录于以下专栏:
相关文章推荐
10个迷惑新手的Cocoa,Objective-c开发难点和问题
首先请谅解我可能使用很多英文,毕竟英文资料将来会是你的主要资料来源。
在你继续深入学习之前,请停下脚步弄清这些问题。如果你...
首先请谅解我可能使用很多英文,毕竟英文资料将来会是你的主要资料来源。
这篇博客将描述一些我见到的众多Cocoa开发新手遇到的问题和障碍。在你继续深入学习MacOS编程之前,请停下脚步弄清这些问题...
September, :52 AM PDT
update on 09 Fri Mar, :24 AM PDT
原文地址:http://lianxu.me/-cocoa-objc-newbie-problems/
25 Comments
language backgrou...
language background
del-e-gate, protocol
event respon-der
mem-ory man...
因为有自己的软件要开发,所以我只能忙里偷闲过来接着写这篇教程。大家莫怪。请按上面的最后更新时间算。
http://lianxu.me/blog//10-cocoa...
10个迷惑新手的Cocoa,Objective-c开发难点和问题
14 September, :52 AM PDT
Last update on 01 Web Fe...
原文地址:/newbie/basic/01.html
10个迷惑新手的Cocoa,Objective-c开发难点和问...
10个迷惑新手的Cocoa,Objective-c开发难点和问题
10个迷惑新手的Cocoa&Objective-c开发问题
NOV 14TH, 2011
languag...
10个迷惑新手的Cocoa,Objective-c开发难点和问题
September, :52 AM PDT
Last update on Fri 2...
他的最新文章
讲师:王禹华
讲师:宋宝华
您举报文章:
举报原因:
原文地址:
原因补充:
(最多只允许输入30个字)

我要回帖

更多关于 xcode opengl es 的文章

 

随机推荐