Sunday, November 18, 2012

Vấn đề về delegate??

Click here...
Delegate dịch nghĩa ra thì là người được ủy nhiệm.
Ví dụ thế này, sếp bạn cử bạn đi công tác nước ngoài. Nhưng quyền quyết định mọi việc ko phải ở bạn mà là ở sếp. Ví dụ như bạn muốn ký hợp đồng, phá vỡ hợp đồng hay ko thì phải là do sếp quyết định. Vấn đề nảy sinh ra là khi bạn ra nước ngoài, sếp bạn sẽ ko biết có chuyện gì nảy sinh ra hay ko để biết lúc nào mà ra quyết định. Còn nếu sếp cứ phải kè kè bên cạnh bạn thì việc cử bạn đi là vô ích. Để giải quyết vấn đề, sếp ủy nhiệm 1 vật đi theo bạn, và khi có việc gì, bạn chỉ cần thông báo qua vật này thì sếp sẽ biết. Ở đây vật này có thể là 1 số điện thoại liên lạc. số điện thoại liên lạc này chính là vật được ủy nhiệm, delegate.
Viết kiểu code thì thế này
Trong class ông chủ:

Nhanvien* a=[self luachonNhanVien];
a.solienlac_delegate=self;
[a dicongtac];

//---->Example-----
protocolTest = [[ClassWithProtocol alloc] init];
    [protocolTest setDelegate:self];
    //or--> protocolTest.delegate=self;
    [protocolTest startSomeProcess];

//-------------------

Trong class nhanvien:
-(void) dicongtac{
//Lam gi thi lam,
//lam xong thi thong bao ve cho ong chu de ong chu quyet dinh
[self.solienlac_delegate baocaosep:ketqua]
}



//------Example------
-(void)startSomeProcess
{
    NSLog(@"---->startSomeProcess!");
   
    [self.delegate processSuccessful:@"HELLO OBJC"]; //Bao cao lai cho sep!
 }
//----------------------

ông chủ sẽ phải comfort cái protocol của delegate, nghĩa là đảm bảo khi được gọi thì sẽ xử lý, nghĩa là cần có function này:
-(void) baocaosepint) ketqua{
if(ketqua==1)
[self goinhanvienve];
if(ketqua==2)
[self ralenhmoichonhanvien];
}
//-----Tai class ong chu, ong chu se quyet dinh lam di voi cai bao cao do cua nhan vien-----
- (void) processSuccessful:(NSString *) success
{
    NSLog(@"Process completed-->%@", success);
}
//---------------------------

Cốt lõi của vấn đề ở đây là CallBack function, hàm gọi ngược, chứ ko phải là trao đổi dữ liệu gì như 1 số bạn nói. Ở đây nhân viên có thể gọi ngược lại, yêu cầu sếp ra quyết định vì nhân viên đã có vật ủy nhiệm của sếp.
Trong C, C++, để thực hiện việc gọi ngược, người ta dùng function pointer.
Trong C#, người ta cũng gọi nó là delegate nhưng delegate này vai trò tương đương với function pointer của C, C++.
Trong java, người ta ko gọi là delegate, nhưng thực chất cũng là truyền 1 object ủy nhiệm tương tự như delegate của objective C mà thôi. Khác biệt với C,C++,C# ở chỗ delegate của Objective C sẽ có thể chứa nhiều functions

The Basics of Protocols and Delegates

Source: http://mobiledevelopertips.com/objective-c/the-basics-of-protocols-and-delegates.html

Apple offers a good overview of working with protocols in their Objective-C Programming Reference. However, sometimes a simple working example can go a long ways…
Introduction
Protocols can be helpful in a number of scenarios, a common usage is to define methods that are to be implemented by other classes. A familiar example is when using a tableview, your class implements the cellForRowAtIndexPath method which asks for cell content to insert into a table – the cellForRowAtIndexPath method is defined within the UITableViewDataSource protocol.
Let’s walk through a very simple example of defining and adopting a protocol.

Protocol Definition
Here is an example of a protocol which includes one method, notice the instance variable delegate is of type id, as it will be unknown at compile time the type of class that will adopt this protocol.
#import <Foundation/Foundation.h>
 
@protocol ProcessDataDelegate <NSObject>
@required
- (void) processSuccessful: (BOOL)success;
@end
 
@interface ClassWithProtocol : NSObject 
{
 id <ProcessDataDelegate> delegate;
}
 
@property (retain) id delegate;
 
-(void)startSomeProcess;
 
@end
Protocol Implementation
Inside the implementation section for the interface defined above we need to do two things at a minimum – first synthesize the delegate instance variable and second, call the method defined in the protocol as needed (more on that in a moment).
Let’s look at a bare bones implementation of the ClassWithProtocol.m:
#import "ClassWithProtocol.h"
 
@implementation ClassWithProtocol
 
@synthesize delegate;
 
- (void)processComplete
{
  [[self delegate] processSuccessful:YES];
}
 
-(void)startSomeProcess
{
  [NSTimer scheduledTimerWithTimeInterval:5.0 target:self 
    selector:@selector(processComplete) userInfo:nil repeats:YES];
}
 
@end
Understand this is a rather contrived example – the intention is to show how/where one might use a protocol. For the sake of discussion assume you have a class that is processing (or downloading) some type of data. Further, assume this class is called from another class to begin the processing. Chances are, at some point the caller will want to be notified that the class processing the data is done, this is where the protocol comes in.
In the calling class, the method defined in the protocol, processSuccessful, will be implemented and will be called from the object doing the processing, once it is complete.
For this example, inside the class where the protocol is defined, I have one method, startSomeProcess, which simply starts a timer and calls processComplete after 5 seconds. Inside processComplete the calling object will be notified through its delegate that the process is done.
Adopting the Protocol
To keep the example short, I am using the applicaton delegate as the class that adopts the protocol. Here is how the app delegate looks:
#import <UIKit/UIKit.h>
#import "ClassWithProtocol.h"
 
@interface TestAppDelegate : NSObject <UIApplicationDelegate, ProcessDataDelegate>
{
  UIWindow *window;
  ClassWithProtocol *protocolTest;
}
 
@property (nonatomic, retain) UIWindow *window;
 
@end
A few things to note – ProcessDataDelegate is defined as part of the interface, which signifies that this class will adhere to the protocol. Looking back to the code for defining the protocol, notice that I added @required to the definition, which means that any class that adopts the protocol must implement the processSuccessful method (you will receive a compile warning if you don’t).
Here is the implementation of the app delegate and the required method for the protocol:
#import "TestAppDelegate.h"
#import "ClassWithProtocol.h"
 
@implementation TestAppDelegate
 
@synthesize window;
 
 UITableViewDelegate
 
- (void)processSuccessful:(BOOL)success;
{
  NSLog(@"Process completed");
}
 
- (void)applicationDidFinishLaunching:(UIApplication *)application 
{   
  // Create and initialize the window
  window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
 
  protocolTest = [[ClassWithProtocol alloc] init];
  [protocolTest setDelegate:self];
  [protocolTest startSomeProcess];
 
  [window makeKeyAndVisible];
}
 
- (void)dealloc 
{
  [window release];
  [super dealloc];
}
 
@end
How it all works
Things goes as follows: the app delegate will create a new instance of the ClassWithProtocol object. It sets itself as the delegate and then calls the startSomeProcess method. At some point in the future, when the protocolTest object has completed its work – after the 5 second timer has fired – it will call the processSuccessful method in the app delegate to let it know it is done processing.
Download Xcode Project
You can find a complete (albeit trivial) working example here: Protocol Xcode Project