著作权声明:本文由http://www.cnblogs.com/icker 原创,欢迎转载分享。转载时请保留该声明和作者博客链接,谢谢!

  最近使用Unity3d制作的IOS游戏需要加入联网对战功能功能,因此使用ObjC语言利用IOS SDK的GameKit.framework的Peer-to-peer Connectivity实现了网络连接,在此分享。

啥话都不说,先上代码。点我下载工程文件

类NetWorkP2P,继承自NSObject。提供GKSessionDelegate和GKPeerPickerControllerDelegate的实现。并且利用单例模式实现一个类方法+( NetWorkP2P *) sharedNetWorkP2P;此方法返回一个NetWorkP2P的实例。

//
// NetWorkP2P.h
// P2PTapWar
//
// Created by on 11-9-14.
// Copyright 2011年 __MyCompanyName__. All rights reserved.
// #import<Foundation/Foundation.h>
#import<GameKit/GameKit.h> #define AMIPHD_P2P_SESSION_ID @"amiphd-p2p"
#define START_GAME_KEY @"startgame"
#define TIME_KEY @"time"
#define END_GAME_KEY @"endgame"
#define TAP_COUNT_KEY @"taps"
#define TIMES_KEY @"times" @interface NetWorkP2P : NSObject<GKSessionDelegate,GKPeerPickerControllerDelegate>{ UInt32 playerScore;
UInt32 opponentScore; UInt32 playerTimes;
UInt32 opponentTimes; NSString *opponentID;
BOOL actingAsHost;
GKSession *gkSession;
}
+( NetWorkP2P *) sharedNetWorkP2P;
-(void)showPeerPickerController; -(void)addTheScore: (UInt32)score;
-(void)addTimes; -(UInt32)getOpponentScore;
-(UInt32)getPlayerScore;
-(UInt32)getPlayerTimes;
-(UInt32)getOpponentTimes; -(NSString *)getOpponentID;
@end

类NetWorkP2P的实现文件,fileName:NetWorkP2P.m

//
// NetWorkP2P.m
// P2PTapWar
//
// Created by on 11-9-14.
// Copyright 2011年 __MyCompanyName__. All rights reserved.
// #import"NetWorkP2P.h" @implementation NetWorkP2P
+(NetWorkP2P *) sharedNetWorkP2P{
static NetWorkP2P *sharedNetWorkObject;
if(!sharedNetWorkObject)
sharedNetWorkObject=[[NetWorkP2P alloc] init];
return sharedNetWorkObject;
} - (id)init
{
self = [super init];
if (self) {
// Initialization code here.
} return self;
}
-(UInt32)getPlayerScore{
return playerScore;
}
-(UInt32)getOpponentScore{
return opponentScore;
}
-(UInt32)getOpponentTimes{
return opponentTimes;
}
-(UInt32)getPlayerTimes{
return playerTimes;
}
-(NSString *)getOpponentID
{
return opponentID;
}
-(void) showPeerPickerController
{
if(!opponentID)
{
actingAsHost=YES;
GKPeerPickerController *peerPickerContrller=[[GKPeerPickerController alloc] init];
peerPickerContrller.delegate=self;
peerPickerContrller.connectionTypesMask=GKPeerPickerConnectionTypeNearby;
[peerPickerContrller show];
}
}
-(void)addTheScore:(UInt32)score
{
playerScore+=score;
NSMutableData *message=[[NSMutableData alloc]init];
NSKeyedArchiver *archiver=[[NSKeyedArchiver alloc]initForWritingWithMutableData:message];
[archiver encodeInt:playerScore forKey:TAP_COUNT_KEY];
[archiver finishEncoding];
GKSendDataMode sendMode=GKSendDataUnreliable;
[gkSession sendDataToAllPeers:message withDataMode:sendMode error:NULL];
[archiver release];
[message release]; }
-(void)addTimes
{
playerTimes++;
NSMutableData *message=[[NSMutableData alloc]init];
NSKeyedArchiver *archiver=[[NSKeyedArchiver alloc] initForWritingWithMutableData:message];
[archiver encodeInt:playerTimes forKey:TIMES_KEY];
[archiver finishEncoding];
GKSendDataMode sendMode=GKSendDataUnreliable;
[gkSession sendDataToAllPeers:message withDataMode:sendMode error:NULL];
[archiver release];
[message release];
}
#pragma mark game logic -(void) initGame{
playerScore=0;
opponentScore=0;
}
-(void) hostGame{
[self initGame];
NSMutableData *message=[[NSMutableData alloc] init];
NSKeyedArchiver *archiver=[[NSKeyedArchiver alloc] initForWritingWithMutableData:message];
[archiver encodeBool:YES forKey:START_GAME_KEY];
[archiver finishEncoding]; NSError *sendErr=nil;
[gkSession sendDataToAllPeers:message withDataMode:GKSendDataReliable error:&sendErr];
if (sendErr) {
NSLog(@"send greeting failed : %@",sendErr);
}
[message release];
[archiver release];
}
-(void) joinGame{
[self initGame];
}
-(void) showEndGameAlert{
}
-(void) endGame{
opponentID=nil;
[gkSession disconnectFromAllPeers];
[self showEndGameAlert];
} #pragma mark GKPeerPickerControllerDelegate methods -(GKSession *) peerPickerController:(GKPeerPickerController *)picker sessionForConnectionType:(GKPeerPickerConnectionType)type
{
if(!gkSession)
{
gkSession=[[GKSession alloc] initWithSessionID:AMIPHD_P2P_SESSION_ID displayName:nil sessionMode:GKSessionModePeer];
gkSession.delegate=self;
}
return gkSession;
}
-(void) peerPickerController:(GKPeerPickerController *) picker didConnectPeer:(NSString *)peerID toSession:(GKSession *)session
{
NSLog ( @"connected to peer %@", peerID);
[session retain]; // TODO: who releases this?
[picker dismiss];
[picker release];
}
- (void)peerPickerControllerDidCancel:(GKPeerPickerController *)picker {
NSLog ( @"peer picker cancelled");
[picker release];
} #pragma mark GKSessionDelegate methods //START:code.P2PTapWarViewController.peerdidchangestate
- (void)session:(GKSession *)session peer:(NSString *)peerID
didChangeState:(GKPeerConnectionState)state {
switch (state)
{
case GKPeerStateConnected:
[session setDataReceiveHandler: self withContext: nil];
opponentID = peerID;
actingAsHost ? [self hostGame] : [self joinGame];
break;
}
}
//END:code.P2PTapWarViewController.peerdidchangestate //START:code.P2PTapWarViewController.didreceiveconnectionrequestfrompeer
- (void)session:(GKSession *)session
didReceiveConnectionRequestFromPeer:(NSString *)peerID {
actingAsHost = NO;
}
//END:code.P2PTapWarViewController.didreceiveconnectionrequestfrompeer - (void)session:(GKSession *)session connectionWithPeerFailed:(NSString *)peerID withError:(NSError *)error {
NSLog (@"session:connectionWithPeerFailed:withError:");
} - (void)session:(GKSession *)session didFailWithError:(NSError *)error {
NSLog (@"session:didFailWithError:");
}
#pragma mark receive data from session
-(void) receiveData: (NSData *) data fromPeer : (NSString *) peerID inSession: (GKSession *) session context:(void*) context{
NSKeyedUnarchiver *unarchiver=[[NSKeyedUnarchiver alloc] initForReadingWithData:data];
if([unarchiver containsValueForKey:TAP_COUNT_KEY])
{
opponentScore=[unarchiver decodeIntForKey:TAP_COUNT_KEY];
}
if([unarchiver containsValueForKey:TIMES_KEY])
{
opponentTimes=[unarchiver decodeIntForKey:TIMES_KEY];
}
if([unarchiver containsValueForKey:END_GAME_KEY])
{
[self endGame];
}
if([unarchiver containsValueForKey:START_GAME_KEY])
{
[self joinGame];
}
[unarchiver release];
}
@end

fileName:NetWorkP2PBinding.m在此文件中实现C语言接口,以方便在UnityScript中调用。

//NetWorkP2PBinding.mm
#import"NetWorkP2P.h"

extern"C"{
void _showPeerPicker()
{
[[NetWorkP2P sharedNetWorkP2P] showPeerPickerController];
NSLog(@"call the mothed _showPeerPicker");
} constchar* _getName()
{
return [@"fyn" UTF8String];
}
int _getOpponentScore()
{
return [[NetWorkP2P sharedNetWorkP2P]getOpponentScore];
}
int _getPlayerScore()
{
return [[NetWorkP2P sharedNetWorkP2P]getPlayerScore];
} void _addTheScore( UInt32 score)
{
[[NetWorkP2P sharedNetWorkP2P] addTheScore:score];
}
void _addTheTimes()
{
[[NetWorkP2P sharedNetWorkP2P] addTimes];
}
int _getOpponentTimes()
{
return [[NetWorkP2P sharedNetWorkP2P] getOpponentTimes];
}
int _getPlayerTimes()
{
return [[NetWorkP2P sharedNetWorkP2P] getPlayerTimes];
}
constchar* _getOpponentID()
{
return [[[NetWorkP2P sharedNetWorkP2P ]getOpponentID] UTF8String];
}
}

FileName:P2PBinding.cs 这是在Unity3d中C#脚本,在这里面调用NetWorkP2PBinding.mm实现的C语言方法。

//P2PBinding.cs

using UnityEngine;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Runtime.InteropServices; publicclass P2PBinding{
[DllImport("__Internal")]
privatestaticexternvoid _showPeerPicker(); publicstaticvoid showPeerPicker()
{
if(Application.platform==RuntimePlatform.IPhonePlayer)
_showPeerPicker();
} [DllImport("__Internal")]
privatestaticexternint _getOpponentScore(); [DllImport("__Internal")]
privatestaticexternint _getOpponentTimes(); [DllImport("__Internal")]
privatestaticexternint _getPlayerScore(); [DllImport("__Internal")]
privatestaticexternint _getPlayerTimes(); [DllImport("__Internal")]
privatestaticexternvoid _addTheTimes(); [DllImport("__Internal")]
privatestaticexternvoid _addTheScore(int score); publicstaticint getOpponentScore()
{
if(Application.platform==RuntimePlatform.IPhonePlayer)
{
return _getOpponentScore();
}
return-1;
}
publicstaticint getOpponentTimes()
{
if(Application.platform==RuntimePlatform.IPhonePlayer)
{
return _getOpponentTimes();
}
return-1;
}
publicstaticint getPlayerScore()
{
if(Application.platform==RuntimePlatform.IPhonePlayer)
{
return _getPlayerScore();
}
return-1;
}
publicstaticint getPlayerTimes()
{
if(Application.platform==RuntimePlatform.IPhonePlayer)
{
return _getPlayerTimes();
}
return-1;
}
publicstaticvoid addTheScore(int score)
{
if(Application.platform==RuntimePlatform.IPhonePlayer)
{
_addTheScore(score);
}
}
publicstaticvoid addTheTimes()
{
_addTheTimes();
}
}

FileName:GameDate.cs 此文件提供游戏的数据,并保持和NetWorkP2P文件中数据同步,将此文件附加到一个GameObject上。

using UnityEngine;
using System.Collections; publicclass GameDate : MonoBehaviour { publicstring opponentID; publicint playerTimes;
publicint opponentTimes;
publicint playerScore;
publicint opponentScore; // Use this for initialization
void Start () {
P2PBinding.showPeerPicker();
} // Update is called once per frame
void Update () {
opponentTimes=P2PBinding.getOpponentTimes();
opponentScore=P2PBinding.getOpponentScore();
}
}

Unity3d使用蓝牙(bluetooth)开发IOS点对点网络游戏的更多相关文章

  1. iOS蓝牙BLE开发

    蓝牙是一个标准的无线通讯协议,具有设备成本低.传输距离近和功耗低等特点,被广泛的应用在多种场合.蓝牙一般分为传统蓝牙和BLE两种模式:传统蓝牙可以传输音频等较大数据量,距离近.功耗相对大:而BLE则用 ...

  2. iOS 上的蓝牙框架 - Core Bluetooth for iOS

    原文: Core Bluetooth for iOS 6 Core Bluetooth 是在iOS5首次引入的,它允许iOS设备可以使用健康,运动,安全,自动化,娱乐,附近等外设数据.在iOS 6 中 ...

  3. Unity3d开发IOS游戏 基础

    Unity3d开发IOS游戏 基础 @阿龙 -  649998群 1.先说明两个问题,我在WIN7下面的U3D里面,用了雅黑字体,但是导出为ios后,字体就看不见了,这是为什么呢?这是需要在MAC下找 ...

  4. Android开发之蓝牙(Bluetooth)操作(一)--扫描已经配对的蓝牙设备

    版权声明:本文为博主原创文章,未经博主允许不得转载. 一. 什么是蓝牙(Bluetooth)? 1.1  BuleTooth是目前使用最广泛的无线通信协议 1.2  主要针对短距离设备通讯(10m) ...

  5. 蓝牙(Bluetooth) IEEE 802.15.1 协议学习

    catalogue . 蓝牙概念 . 配对和连接 . 机密安全性 . 蓝牙协议分类 . 蓝牙协议栈 1. 蓝牙概念 蓝牙(Bluetooth)是一种无线技术标准,可实现固定设备.移动设备和楼宇个人域网 ...

  6. (转)火溶CEO王伟峰:Unity3D手机网游开发

    今天看到这篇文章,感觉很不错,尤其是那句“Unity3D的坑我觉得最严重的坑就是没有懂3D的程序员,把Unity当成Office用”. 转自http://blog.csdn.net/wwwang891 ...

  7. 深入了解Android蓝牙Bluetooth——《基础篇》

    什么是蓝牙?   也可以说是蓝牙技术.所谓蓝牙(Bluetooth)技术,实际上是一种短距离无线电技术,是由爱立信公司公司发明的.利用"蓝牙"技术,能够有效地简化掌上电脑.笔记本电 ...

  8. iOS开发-iOS 10 由于权限问题导致崩溃的那些坑

     iOS开发-iOS 10 由于权限问题导致崩溃的那些坑 6月份的WWDC大会结束有一段时间了,相信很多开发者也是在努力工作的闲时用着Xcode8 Beta版学习着新的特性吧. 使用Xcode8写自己 ...

  9. BLK-MD-BC04-B蓝牙模块开发说明

    BLK-MD-BC04-B蓝牙模块开发说明 日期:2011-9-24 浏览次数:4178     BLK-MD-BC04-B蓝牙通信模块, BLK-MD-BC04-B蓝牙通信模块 为本公司自主开发的智 ...

随机推荐

  1. javascript 事件传播与事件冒泡,W3C事件模型

    说实话笔者在才工作的时候就听说了什么"事件冒泡",弄了很久才弄个大概,当时理解意思是子级dom元素和父级dom元素都绑定了相同类型的事件,这时如果子级事件触发了父级也会触发,然后这 ...

  2. nonatomic, retain,weak,strong用法详解

    strong weak strong与weak是由ARC新引入的对象变量属性 ARC引入了新的对象的新生命周期限定,即零弱引用.如果零弱引用指向的对象被deallocated的话,零弱引用的对象会被自 ...

  3. Python学习日记

    江林楠学习了一下午后给大家呈现的20分钟速成Python—— 一些基本的语法:1.python无变量声明 直接a = []即可.2.python为对齐语言,用制表符表示语句块的嵌套.3.python语 ...

  4. java.lang.NoClassDefFoundError: antlr/ANTLRException

    在用Hibernate进行查询时,出现这样的错误:Exception in thread "main" java.lang.NoClassDefFoundError: antlr/ ...

  5. UVA5870 乱搞 Smooth Visualization

    #include<stdio.h> #include<string.h> #define maxn 1201 ][],s[maxn]; int col; int getmax( ...

  6. POJ 2153 stl

    #include<iostream> #include<map> #include<string> using namespace std; int main() ...

  7. Java设计模式-享元模式(Flyweight)

    享元模式的主要目的是实现对象的共享,即共享池,当系统中对象多的时候可以减少内存的开销,通常与工厂模式一起使用. FlyWeightFactory负责创建和管理享元单元,当一个客户端请求时,工厂需要检查 ...

  8. Java基础-字面值

    在Java源代码中,字面值用于表示固定的值(fixed value).数值型的字面值是最常见的,字符串字面值可以算是一种,当然也可以把特殊的null当做字面值.字面值大体上可以分为整型字面值.浮点字面 ...

  9. 【ZOJ 3502】Contest

    题 题意 n个问题,解决的顺序影响正确的概率,无论之前解决的问题是否答对,当前问题 j 答对概率为max{a[i][j]} (i为解决过的问题).求答对题目的最大期望和对应的答题顺序.T组测试,T ( ...

  10. codeforces 375D:Tree and Queries

    Description You have a rooted tree consisting of n vertices. Each vertex of the tree has some color. ...