在IOS中為了實現回調一般有如下幾個方法:
delegate 通知中心 block KVO(較特殊的回調,姑且也算一種) 以上四種中在我自己的項目中比較常用的就是delegate和block了。 在現實中回調的需求也分兩種 一對一的回調。 一對多的回調。 對於一對一的回調,在IOS中使用delegate、block都能實現。而一對多的回調基本就是通知中心了。 假如現在有一個需求,我們以圖片下載為例。這裡先忽略哪些SDWebimage等已經封裝好的第三方類庫。對於圖片下載一般的過程如下: 先判斷該圖片url是否已經下載完畢。如果已經下載完畢那麼直接回調顯示圖片。如果沒有下載那麼進入下載過程. 使用合適的圖片下載器下載圖片。 圖片下載完畢後回調顯示圖片。並且把該圖片存到緩存中。 這裡的難點是回調。如果一個頁面中有多個地方需要顯示同一張圖片,那麼勢必會發生這樣一種情況,就是同時有多個請求下載同意url的圖片,並且下載完成後需要同時在多個地方顯示圖片。要是實現這樣的需求,用現有的方案貌似很難解決。有的同學會想到通知中心,但是通知中心其實是一個廣播服務,只要注冊了接受該通知那麼所有的注冊者都能收到通知,但事實上我只需要在我需要下載的那個url的圖片下載完後給出通知,而不需要所有的下載完畢事件都通知。這時候我們就需要多播委托了。 什麼是多播委托?我直接拿其他博客上的一個定義來解釋。簡單地說,多播委托是指允許創建方法的調用列表或者鏈表的能力。當多播委托被調用時,列表中的方法均自動執行 在IOS中我就以我們平常用的最多的delagate為例,普通的delegate只能是一對一的回調,無法做到一對多的回調。而多播委托正式對delegate的一種擴展和延伸,多了一個注冊和取消注冊的過程,任何需要回調的對象都必須先注冊。 如何在IOS中實現多播委托?老外早就已經寫好了,而且相當的好用。我最初接觸IOS多播委托是我在研究XMPPframework的時候,而多播委托可以說是XMPPframework架構的核心之一。具體的類名就是GCDMulticastDelegate,從名字就可以看出,這是一個支持多線程的多播委托。那為什麼要支持多線程呢?我的理解是多個回調有可能不是在同一個線程的,比如我注冊回調的時候是在後台線程,但是你回調的時候卻在UI線程,那就有可能出問題了。因此必須保證你注冊的時候在哪個線程上注冊的,那麼回調的時候必須還是在那個線程上回調的。 下面我講解寫多播委托在IOS中的用法。 我先舉個例子,比如有一個UserInfo(有一個userName的屬性)的類,頁面上有三個lable和一個按鈕,當點擊按鈕的時候給userInfo的userName屬性賦值,這時候三個lable同時顯示userInfo的userName屬性的值。 針對以上過程,我們需要對每個lable向userInfo實例注冊,也就是向多播委托注冊。當對userInfo的userName賦值的時候調用多播委托的方法,這裡也就是調用setText方法。這樣就能實現上面的需求了。 用代碼表示就是: 復制代碼 //繼承自多播委托基類的userInfo類 @interface UserInfo : MulticastDelegateBaseObject @property (nonatomic,strong)NSString *userName; @end @implementation UserInfo -(void)setUserName:(NSString *)userName{ _userName=userName; [multicastDelegate setText:userName];//調用多播委托 } @end - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typically from a nib. //初始化一個userinfo的實例 userInfo=[[UserInfo alloc] init]; //添加一個lable UILabel *lable =[[UILabel alloc] initWithFrame:CGRectMake(0, 20, 100, 30)]; lable.backgroundColor=[UIColor blueColor]; lable.textColor=[UIColor blackColor]; [userInfo addDelegate:lable delegateQueue:dispatch_get_main_queue()];//向多播委托注冊 [self.view addSubview:lable]; lable =[[UILabel alloc] initWithFrame:CGRectMake(0, 60, 100, 30)]; lable.backgroundColor=[UIColor blueColor]; lable.textColor=[UIColor blackColor]; [userInfo addDelegate:lable delegateQueue:dispatch_get_main_queue()]; [self.view addSubview:lable]; lable =[[UILabel alloc] initWithFrame:CGRectMake(0, 100, 100, 30)]; lable.backgroundColor=[UIColor blueColor]; lable.textColor=[UIColor blackColor]; [userInfo addDelegate:lable delegateQueue:dispatch_get_main_queue()]; [self.view addSubview:lable]; //添加一個按鈕 UIButton *btn=[[UIButton alloc] initWithFrame:CGRectMake(200, 20, 100, 50)]; [btn setBackgroundColor:[UIColor blueColor]]; [btn setTitle:@"button1" forState:UIControlStateNormal]; [btn addTarget:self action:@selector(btnCLicked:) forControlEvents:UIControlEventTouchUpInside]; [self.view addSubview:btn]; } -(void)btnCLicked:(UIButton *)btn{ userInfo.userName=@"123456";//給userInfo賦值 }