好久没写东西,最近在研究服务端框架Firefly和Pomelo,身为菜鸟的我的确花了很大功夫才看懂一些源代码。原来打算玩下Pomelo,不过我不得不说这东西真的是给专业开发者准备的,我搞了半天libpomelo也没顺利链接上服务器,光是链接服务器都那么难搞,更别说通讯了,我还能说什么呢……(真的是网络资料都翻遍了,真不知道其它人是怎么用的),官方示例里并没有简易代码,所以不适合像我这样的超级菜鸟使用,相比之下,Firefly更容易上手,有很多类型的源代码,简易通俗的和系统完整级的都有,认真研究的话真能学到不少东西……

因为官方给出的网络通讯协议示例里只有python的客户端源码,所以对于小白来说,可能不知道如何在cocos2d-x项目中的VC++里实现,这也算是一个添加的教程吧。还和以前一样,把研究出的东西记录下以备后用,希望对初学者也能有所帮助……

Firefly是开源游戏服务器框架,可以直接到九秒社区下载安装,这里不说安装过程了,我使用的是新版的gFirefly,这个也是可以在gitHub上下载到,安装会麻烦些,话说好久没更新了唉……难道最近都在忙CrossAPP项目?

cocos2d-x3.2需要使用VS2012,其具有C++11新特性,在使用线程上已经相当方便了,不再需要依赖于第三方的pthread

通常,在cocos2dx里使用的是http类的短链接通讯,不过我在这里要记录的是使用socket与服务端进行交互,在像linux这样的平台下,一般使用的都是BSD socket,这个当然不是第三方的插件,而是unix / linux系统里自带的,这也使用得跨平台使用也没什么问题,本例只是在windows上测试通过的代码,未在手机真机上测试过,不过应该差不多。

在Firefly的源代码里,一般可以看到都包含一个network的文件夹,里面有网络通讯使用的方法和类,算是一个打了个包,下面只是把里面最核心的代码拿出来修改使用:

socket最核心的三个方法就是:

connect() 用于链接服务器

send() 用于发消息到服务器

recv() 用于接收服务器返回的消息

本身使用上面的东西没什么难的,对于小白来说,真正需要了解的是Firefly的通讯协议,如果你在客户端发送的消息格式与Firefly的消息格式不一样,那Firefly会直接飞出一段英文,意思大概是“接收到一个非法包,没法识别”。所以这里需要了解一下Firefly的通讯协议。

在发送给Firefly服务端的消息中需要包含以下头部信息(这些在官方的教程里是有的):

class Message:public CCObject
{
public:
	
    
    char HEAD0;
    char HEAD1;
    char HEAD2;
    char HEAD3;
    char ProtoVersion;
    
    byte serverVersion[4];
    byte length[4];
    byte commandId[4];
    /**
      * 消息的数据
      */
    char* data;
	
	
	
	
	Message();
    int datalength();
	~Message();
};

上面一直到commandId的声明定义都是消息头,也就是协议头,这个协议头是用来识别消息的基础信息,像协议版本protoversion,整个消息包的长度length,命令号commandId(可以用来执行指定服务端功能函数的识别号)……data就是我们要传送的消息主体信息内容,上面是在客户端里定义的一个基于CCobject的消息对象。下面再看看服务端的,下面的代码取自游戏《烽烟OL》服务端源码,只要在copy到新建的Firefly项目中即可使用:
from gfirefly.server.globalobject import GlobalObject
from gfirefly.netconnect.datapack import DataPackProtoc


def callWhenConnLost(conn):
    dynamicId = conn.transport.sessionno
    GlobalObject().remote['gate'].callRemote("NetConnLost_2",dynamicId)
    print('一个链接已经断开')

def CreatVersionResult(netversion):
    return netversion

def doConnectionMade(conn):
    print('已成功建立一个链接')
    
dataprotocl = DataPackProtoc(78,37,38,48,9,0)
GlobalObject().netfactory.setDataProtocl(dataprotocl)

GlobalObject().netfactory.doConnectionLost = callWhenConnLost
GlobalObject().netfactory.doConnectionMade = doConnectionMade


from gfirefly.server.globalobject import remoteserviceHandle
from gfirefly.server.globalobject import netserviceHandle


@netserviceHandle
def echo_1(_conn,data):
    print(data)
    return data

def echo_2(showtext):
    print(showtext);
    return showtext

其中下面这段就是用来自定义协议头的代码,分别对应于前面客户端上的定义的前6个参数,如果发送过来的包不是包含相同格式及对应信息时,则不会被服务端解析
dataprotocl = DataPackProtoc(78,0)

上面还定义了一个名为echo_1的函数,后面这个_1是Firefly用识别功能函数的ID,绝对不能重复,当我们从客户端发送消息时,如果指定commandId参数为1,则服务端在接收到这个消息时,会执行echo_1这个函数,执行完后的return用来把返回给客户端相应的数据,服务端的代码就算是这样完成了。

再看看消息构造函数,这个也是取自Firefly官方发布的游戏源代码:

Message* networkManager::constructMessage(const char* data,int commandId)
{
    Message* msg = new Message();
    
    msg->HEAD0=78;
    msg->HEAD1=37;
    msg->HEAD2=38;
    msg->HEAD3=48;
    msg->ProtoVersion=9;
    
    int a=0;
    msg->serverVersion[3]=(byte)(0xff&a);;
    msg->serverVersion[2]=(byte)((0xff00&a)>>8);
    msg->serverVersion[1]=(byte)((0xff0000&a)>>16);
    msg->serverVersion[0]=(byte)((0xff000000&a)>>24);
    
    int b=strlen(data)+4;
    
    msg->length[3]=(byte)(0xff&b);;
    msg->length[2]=(byte)((0xff00&b)>>8);
    msg->length[1]=(byte)((0xff0000&b)>>16);
    msg->length[0]=(byte)((0xff000000&b)>>24);
    
    int c=commandId;
    msg->commandId[3]=(byte)(0xff&c);;
    msg->commandId[2]=(byte)((0xff00&c)>>8);
    msg->commandId[1]=(byte)((0xff0000&c)>>16);
    msg->commandId[0]=(byte)((0xff000000&c)>>24);
    
    //    str.append(msg->HEAD0);
    printf("%d",msg->datalength());
    msg->data = new char[msg->datalength()];
    memcpy(msg->data+0,&msg->HEAD0,1);
    memcpy(msg->data+1,&msg->HEAD1,1);
    memcpy(msg->data+2,&msg->HEAD2,1);
    memcpy(msg->data+3,&msg->HEAD3,1);
    memcpy(msg->data+4,&msg->ProtoVersion,1);
    memcpy(msg->data+5,&msg->serverVersion,4);
    memcpy(msg->data+9,&msg->length,4);
    memcpy(msg->data+13,&msg->commandId,4);
    memcpy(msg->data+17,data,strlen(data));
    //memcpy(msg->data+position,bytes+offset,len);
    //msg->data = data;
	return msg;
}
上面的代码对消息从头到尾按次序进行了一次拼接封装,算是打包进data中,让其成为一个完整的数据包,最后返回消息对象。

然后就是链接服务器了,下面是代码:

bool networkManager::Connect() {
	mlock.lock();
	//判断windows平台下初始链接初始化是否成功
	if(Init()==-1){
		return false;
	}
	//判断套接字是否创建成功
	if(Create(AF_INET,SOCK_STREAM,0)==false){
	return false;
	};
	//设置socket为非阻塞模式
	/*int retVal;
	unsigned long ul = 1;
	retVal=ioctlsocket(m_sock,FIONBIO,&ul);
	if(retVal==SOCKET_ERROR){
		cclOG("设置阻塞参数错误");
		closesocket(m_sock);  
		#ifdef WIN32
		WSACleanup();
		#endif
	}*/
	//使用创建的套接字链接服务器
	struct sockaddr_in svraddr;
	svraddr.sin_family = AF_INET;
	svraddr.sin_addr.s_addr = inet_addr(IP_ADDRESS);
	svraddr.sin_port = htons(IP_HOST);

	int ret = connect(m_sock,(struct sockaddr*) &svraddr,sizeof(svraddr));
	if (ret == SOCKET_ERROR) {
		/*closesocket(m_sock);
		#ifdef WIN32
		WSACleanup();
		#endif*/

		cclOG("link Failed");
			//链接成功后开始发送数据到服务器
	//sendThread();
    //recvThread();

		return false;
	}
	//链接成功后开始发送数据到服务器
	sendThread();
	cclOG("link successed");
	mlock.unlock();

	return true;
}

可以看到上面链接代码的尾部已经加入执行了发送数据的函数,发送的实现代码其实很简单,下面是发送了一条"getSendMessage successful!"的信息给服务器,而如果服务器收到这个消息后,也会在log里输出这样一条消息的:
void networkManager::sendThread(){
		Message* msg=constructMessage("getSendMessage successful!",1);
		//发消息
		Send(msg->data,msg->datalength(),0);
}

发送消息后,则可以开始监听接收服务端返回的数据了,下面只给出了基本代码,不包含数据解析,收到服务端返回的消息后可以看到LOG输出的信息:
void networkManager::RecvFunc(){
	char recvBuf[17];
	FD_ZERO(&fdread);
	FD_SET(m_sock,&fdread);
	mlock.lock();
		struct timeval	aTime;
	aTime.tv_sec = 5;
	aTime.tv_usec = 0;
	
	int ret = select(m_sock,&fdread,NULL,&aTime);

if (FD_ISSET(m_sock,&fdread))
{
	cclog("socket State=%d",ret);
	if(ret==1){
		//先拿到时协议头数据,根据里面的信息判断应该调用哪些回调函数进行下一步数据处理
		//while(true){
		int getRevDataLength=recv(m_sock,recvBuf,17,0);
		if(getRevDataLength==17){
			cclOG("recvThread OK,getDataProcess=%d",getRevDataLength);			
		}else{
			cclOG("The connect has terminated! revData is not completed!");
			cclOG("The ERROR CODE:%d",WSAGetLastError()); 
		//closesocket(m_sock);
		}

	}

}else{
	cclOG("select sock error");
}
	mlock.unlock();
}

//执行接收线程
void networkManager::recvThread(){
	//开启一条t2线程,入口函数为RecvFunc()
	std::thread t2(&networkManager::RecvFunc,this);	
	t2.join();
}

最后执行代码后,可以在服务端上看到我们发送的消息,如下图


至此就算完成了一次与服务端的通讯会话。

由于我研究代码功能实现时有随意乱写代码的坏习惯,所以,源代码可能会有些多余和不符合标准的东西,请多包涵!VS2012的客户端项目源代码可到下面地址下载:

http://download.csdn.net/detail/cyistudio/8004925

cocos2d-x3.2与服务端框架Firefly的网络编程初级网络通讯的更多相关文章

  1. 详解html5 postMessage解决跨域通信的问题

    这篇文章主要介绍了详解html5 postMessage解决跨域通信的问题的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  2. 详解使用postMessage解决iframe跨域通信问题

    这篇文章主要介绍了详解使用postMessage解决iframe跨域通信问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  3. HTML5调用手机发短信和打电话功能

    这篇文章主要介绍了HTML5调用手机发短信和打电话功能,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  4. 真正的iOS设备和Watch Simulator可以进行通信以进行测试

    我想为现有的iOS应用创建一个手表应用.但我处于一种情况,我没有苹果手表,我现有的iOS应用程序只能在不在模拟器上的真实设备上运行.是否可以在iPhone设备上运行应用程序并在手表模拟器中测试我的手表应用程序?解决方法至少在目前,不可能配对真正的iPhone和Watch模拟器.我得出这个结论有三个原因:>Watch模拟器在安装过程中自动与iPhone模拟器配对.>根本无法从界面取消配对Watch模拟器.>在模拟器上无法访问蓝牙以与真实设备进行通信.这是一个proof.

  5. xcode6.1 – Xcode 6.1中项目模板中缺少类前缀

    项目模板上曾经有一个类前缀字段,这有助于区分项目类和框架类.Xcode6.1项目模板中不再提供此功能.这背后的意图是什么?

  6. ios – 伞框架

    错误.应用程序,通常位于…错误仍然存在你也可以在这里添加(子)框架的路径.

  7. ios – UIView框架大小的问题

    我正在开发一个iPad项目,目前正在使用Landscape视图.我试着这样做:为什么这总是返回960?虽然在景观中,视图本身的高度尺寸应为768对吗?

  8. 安装自定义cocoa框架的最佳方法

    我有一个自定义框架,遵循Apple的框架编程指南>>中的建议.Installingyourframework我在/Library/Frameworks中安装.我通过使用以下脚本添加RunScript构建阶段来完成此操作:在我的项目中,我然后链接/Library/Frameworks/MyFramework并将其导入我的类中,如下所示:这非常有效,除了我总是在调试器控制台中看到以下消息:Loadin

  9. ios – 如何使用iphone使用蓝牙或wifi与OBD II进行通信

    我想为iOS创建一个简单的应用程序,它通过ODBIIWifi/蓝牙设备从汽车读取数据并在iPhone屏幕上显示.但我不知道从哪里开始.请有人帮我实现以下结果.我有蓝牙和wifi加密狗.第1步:配对蓝牙或Wifi加密狗和iPhone.第2步:通过加密狗阅读详细信息请参阅我能够轻松理解的任何教程或示例代码.我想深入了解这些过程,并希望自己编写代码.所以请帮忙.提前致谢.解决方法正如David所说,在i

  10. ios – 在设备上构建和运行时,仅将嵌入式框架与其他动态框架链接失败

    TL;博士将您的嵌入式框架与其他框架链接,并且不将其他框架与您的应用程序链接,导致Build&在设备上运行.描述:建立:我的设置非常简单(Swift2.3&XcodeXcode8.0;Build版本8S162m):>使用Carthage(0.17.2)我用xcodebuild8.0和TOOLCHAINS=com.apple.dt.toolchain.Swift_2_3carthagebui

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部