什麼是Multipeer Connectivity?
在iOS7中,引入了一個全新的架構——Multipeer Connectivity(多點連接配接)。利用Multipeer Connectivity架構,即使在沒有連接配接到WiFi(WLAN)或移動網絡(xG)的情況下,距離較近的Apple裝置(iMac/iPad/iPhone)之間可基于藍牙和WiFi(P2P WiFi)技術進行發現和連接配接實作近場通信。
Multipeer Connectivity擴充的功能與利用AirDrop傳輸檔案非常類似,可以将其看作AirDrop不能直接使用的補償,代價是需要自己實作。
本Demo主要用到4個類:
MCBrowserViewController
:MCBrowserViewController繼承自UIViewController,提供了基本的UI應用架構。
MCAdvertiserAssistant
、MCAdvertiserAssistant為針對Advertiser封裝的管理助手,主要處理廣播資訊。
MCSession
:類似TCP連結中的socket。建立MCSession時,需指定自身MCPeerID,類似bind。
MCPeerID
:類似sockaddr,用于辨別連接配接的兩端endpoint,通常是昵稱或裝置名稱。
1、簡單地建立一個界面,主要有連接配接和發送2個UIButton。
@interface ViewController ()<MCBrowserViewControllerDelegate, MCSessionDelegate>
{
NSInteger noOfDataSend;
NSInteger noOfData;
NSMutableArray *marrReceiveData;
NSMutableArray *marrFileData;
}
@property (nonatomic, strong) MCPeerID *myPeerID;
/** <#Description#> */
@property (nonatomic, strong) MCSession *mySession;
/** m */
@property (nonatomic, strong) MCBrowserViewController *browserVC;
/** <#Description#> */
@property (nonatomic, strong) MCAdvertiserAssistant *advertiser;
//圖檔
@property (weak, nonatomic) IBOutlet UIImageView *imageView;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
marrFileData = [NSMutableArray array];
marrReceiveData = [NSMutableArray array];
}
- (IBAction)buttonShareClick:(UIButton *)sender {
if (!self.mySession) {
[self setupMultipeer];
}
[self showBrowserVC];
}
- (IBAction)buttonSendClick:(UIButton *)sender {
[self sendData];
}
2、Multipeer Connectivity架構初始化這4個類。
- (void)setupMultipeer {
//類似sockaddr,用于辨別連接配接的兩端endpoint,通常是昵稱或裝置名稱
self.myPeerID = [[MCPeerID alloc] initWithDisplayName:[UIDevice currentDevice].name];
//類似TCP連結中的socket。建立MCSession時,需指定自身MCPeerID,類似bind。
self.mySession = [[MCSession alloc] initWithPeer:self.myPeerID];
self.mySession.delegate = self;
//提供了基本的UI應用架構
self.browserVC = [[MCBrowserViewController alloc] initWithServiceType:@"chat" session:self.mySession];
self.browserVC.delegate = self;
//封裝的管理助手,主要處理廣播資訊。
self.advertiser = [[MCAdvertiserAssistant alloc] initWithServiceType:@"chat" discoveryInfo:nil session:self.mySession];
[self.advertiser start];
}
- (void)showBrowserVC {
[self presentViewController:self.browserVC animated:YES completion:nil];
}
- (void)dismissBrowserVC {
[self.browserVC dismissViewControllerAnimated:YES completion:^{
[self invokeAlertMethod:@"連接配接成功" Body:@"Both device connected successfully." Delegate:nil];
}];
}
- (void)stopWifiSharing:(BOOL)isClear {
if (isClear && self.mySession != nil) {
[self.mySession disconnect];
[self.mySession setDelegate:nil];
self.mySession = nil;
self.browserVC = nil;
}
}
3、MCBrowserViewController 代理方法
//點選完成
- (void)browserViewControllerDidFinish:(MCBrowserViewController *)browserViewController {
[self dismissBrowserVC];
[marrReceiveData removeAllObjects];
}
//點選取消
- (void)browserViewControllerWasCancelled:(MCBrowserViewController *)browserViewController {
[self dismissBrowserVC];
}
4、MCSession代理方法 主要處理發送方傳遞的檔案或者資訊
// Received data from remote peer
- (void)session:(MCSession *)session didReceiveData:(NSData *)data fromPeer:(MCPeerID *)peerID {
NSLog(@"data reseived long: %lu",(unsigned long)data.length);
if (data.length > 0) {
if (data.length < 2) {
noOfDataSend++;
NSLog(@"noOfdataSend:%zd", noOfDataSend);
NSLog(@"array count:%zd", marrFileData.count);
if (noOfDataSend < [marrFileData count]) {
[self.mySession sendData:[marrFileData objectAtIndex:noOfDataSend] toPeers:[self.mySession connectedPeers] withMode:MCSessionSendDataReliable error:nil];
}else {
[self.mySession sendData:[@"File Transfer Done" dataUsingEncoding:NSUTF8StringEncoding] toPeers:[self.mySession connectedPeers] withMode:MCSessionSendDataReliable error:nil];
}
}else {
if ([[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding] isEqualToString:@"File Transfer Done"]) {
[self appendFileData];
}else {
[self.mySession sendData:[@"1" dataUsingEncoding:NSUTF8StringEncoding] toPeers:[self.mySession connectedPeers] withMode:MCSessionSendDataReliable error:nil];
[marrReceiveData addObject:data];
}
}
}
}
// Received a byte stream from remote peer
- (void)session:(MCSession *)session didReceiveStream:(NSInputStream *)stream withName:(NSString *)streamName fromPeer:(MCPeerID *)peerID {
NSLog(@"did receive stream streamName:%@", streamName);
}
// Start receiving a resource from remote peer
- (void)session:(MCSession *)session didStartReceivingResourceWithName:(NSString *)resourceName fromPeer:(MCPeerID *)peerID withProgress:(NSProgress *)progress {
NSLog(@"start receiving peerID :%@", peerID);
}
// Finished receiving a resource from remote peer and saved the content in a temporary location - the app is responsible for moving the file to a permanent location within its sandbox
- (void) session:(MCSession *)session didFinishReceivingResourceWithName:(NSString *)resourceName fromPeer:(MCPeerID *)peerID atURL:(NSURL *)localURL withError:(NSError *)error {
NSLog(@"finish receiving resource resourceName:%@", resourceName);
}
- (void)session:(MCSession *)session peer:(MCPeerID *)peerID didChangeState:(MCSessionState)state {
NSLog(@"change state:%zd", state);
}
5、發送圖檔(此Demo隻是簡單地做了個收發圖檔的Demo,此架構可實作的功能當然不止這麼簡單。)
- (void)sendData {
[marrFileData removeAllObjects];
NSData *sendData = UIImagePNGRepresentation([UIImage imageNamed: @"test"]);
self.imageView.image = [UIImage imageWithData:sendData];
NSUInteger length = [sendData length];
NSUInteger chunkSize = 100 * 1024;
NSUInteger offset = 0;
do {
NSUInteger thisChunkSize = length - offset > chunkSize ? chunkSize : length - offset;
NSData *chunk = [NSData dataWithBytesNoCopy:(char *)[sendData bytes] + offset length:thisChunkSize freeWhenDone:NO];
NSLog(@"chunk length:%lu",(unsigned long)chunk.length);
[marrFileData addObject:[NSData dataWithData:chunk]];
offset += thisChunkSize;
} while (offset < length);
noOfData = [marrFileData count];
noOfDataSend = 0;
if ([marrFileData count] > 0) {
[self.mySession sendData:[marrFileData objectAtIndex:noOfDataSend] toPeers:[self.mySession connectedPeers] withMode:MCSessionSendDataReliable error:nil];
}
}
- (void)appendFileData {
NSMutableData *fileData = [NSMutableData data];
for (int i = 0; i < [marrReceiveData count]; i++) {
[fileData appendData:[marrReceiveData objectAtIndex:i]];
}
NSString *path = [NSString stringWithFormat:@"%@/Image.png", [NSHomeDirectory() stringByAppendingPathComponent:@"Documents"]];
[fileData writeToFile:path atomically:YES];
NSLog(@"Documents:%@", path);
UIImageWriteToSavedPhotosAlbum([UIImage imageWithData:fileData], self, @selector(image:didFinishSavingWithError:contextInfo:), nil);
}
- (void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo
{
if (!error) {
[self invokeAlertMethod:@"發送成功" Body:@"圖檔已儲存到手機相冊" Delegate:nil];
}
}
- (void)invokeAlertMethod:(NSString *)strTitle Body:(NSString *)strBody Delegate:(id)delegate
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:strTitle
message:strBody
delegate:delegate
cancelButtonTitle:@"OK"
otherButtonTitles:nil];
[alert show];
alert = nil;
}