`
cwqcwk1
  • 浏览: 86853 次
文章分类
社区版块
存档分类
最新评论

erlang 实现与 c++ 通讯

 
阅读更多

网络通讯常用的有2种:文本通讯和二进制通讯。文章分别在两种方式上实现erlang与c++通讯。

一、erlang 与 c++ 行通讯

行通讯是一种文本通讯,格式如“cmdargs\n”,比较简单,erlang接收数据也比较简单,可以设置参数 {packet, line} ,缺点是表示方式单一,很难表示结构数据。

1、erlang 代码(server.erl):

-module(server).
-export([start/0]).
-define( PORT, 5678).

%% 启动服务并接受客户端的连接
start() ->
  {ok, LSock} = gen_tcp:listen(?PORT, [binary, {packet, line},{active, false}]),
  io:format("socket listen: ~p on ~p ~n",[LSock, ?PORT]),
  accept(LSock).

accept(LSock) ->
  {ok, ASock} = gen_tcp:accept(LSock),
  spawn(fun() -> server_loop(ASock) end),
  accept(LSock).

server_loop(ASock) ->
  case gen_tcp:recv(ASock, 0) of
    {ok, Data} ->
      io:format("recv line data: ~p~n", [Data]),
      gen_tcp:send(ASock, Data),
      server_loop(ASock);
    {error, _} ->
      {ok, recv_error}
    end.

2、c++ 代码(test.cpp):

#include <winsock2.h>
#include <iostream>
#pragma comment(lib, "ws2_32.lib")

int main(int argc, char* argv[])
{
	WSADATA wsaData;
	WSAStartup( MAKEWORD( 2, 2 ), &wsaData ); 
	SOCKET sc = WSASocket(AF_INET, SOCK_STREAM, 0, NULL, 0, NULL);
	SOCKADDR_IN addr;
	int len=sizeof(addr);
	addr.sin_family=AF_INET;
	addr.sin_addr.s_addr = inet_addr("127.0.0.1");
	addr.sin_port = htons(5678);
	connect(sc, (struct sockaddr *)&addr, len);
	char buff[1024];
	memset(buff, 0, 1024);

	char data[] = "aaaaaa\n";

	//发送数据
	printf("send data: %s \n", data);
	send(sc, data, sizeof(data), 0);

	//接收数据
	recv(sc, buff, 1024, 0);
	printf("recv data: %s \n", buff);

	closesocket(sc);
	WSACleanup();
	getchar();
	return 0;
}
二、erlang 与 c++ 包体通讯

包体通讯是二进制通讯,目前主要有结构体,json,asn.1,protobuf等等,最简单还可以这样定义:2个字节cmd + 2个字节字符长度 + 字符内容。

下面以结构体包体通讯示例:

1、erlang代码(server.erl):

-module(server).
-export([start/0]).

-define( UINT, 32/unsigned-little-integer).
-define( INT, 32/signed-little-integer).
-define( USHORT, 16/unsigned-little-integer).
-define( SHORT, 16/signed-little-integer).
-define( UBYTE, 8/unsigned-little-integer).
-define( BYTE, 8/signed-little-integer).

-define( PORT, 5678).

%% 启动服务并接受客户端的连接
start() ->
  {ok, LSock} = gen_tcp:listen(?PORT, [binary, {packet, 0},{active, false}]),
  io:format("socket listen: ~p on ~p ~n",[LSock, ?PORT]),
  accept(LSock).

accept(LSock) ->
  {ok, ASock} = gen_tcp:accept(LSock),
  spawn(fun() -> server_loop(ASock) end),
  accept(LSock).

server_loop(ASock) ->
  case gen_tcp:recv(ASock, 0) of
    {ok, <<Len:?USHORT,Cmd:?USHORT,Contain:4/binary-unit:8,_Rest/binary>> = A} ->
      %% _Rest 匹配C++字符串后面的结束符/0
      io:format("recv data: ~p ~p ~p~n", [Len, Cmd, Contain]),
      gen_tcp:send(ASock, A),
      server_loop(ASock);
    {ok, Data} ->
      io:format("recv unformated data: ~p~n", [Data]),
      server_loop(ASock);
    {error, _} ->
      {ok, recv_error}
    end.
2、c++ 代码(test.cpp):

#include <winsock2.h>
#include <iostream>
#pragma comment(lib, "ws2_32.lib")

struct data{
	unsigned short int len;
	unsigned short int cmd;
	char content[5] ;
};

int main(int argc, char* argv[])
{
	WSADATA wsaData;
	WSAStartup( MAKEWORD( 2, 2 ), &wsaData ); 
	SOCKET sc = WSASocket(AF_INET, SOCK_STREAM, 0, NULL, 0, NULL);
	SOCKADDR_IN addr;
	int len=sizeof(addr);
	addr.sin_family=AF_INET;
	addr.sin_addr.s_addr = inet_addr("127.0.0.1");
	addr.sin_port = htons(5678);
	connect(sc, (struct sockaddr *)&addr, len);
	char buff[1024];
	memset(buff, 0, 1024);

	struct data pdata = {4, 1001, "test"};

	//发送数据
	printf("send data: %d %d %s \n", pdata.len, pdata.cmd, pdata.content);
	send(sc, (char *)&pdata, sizeof(pdata), 0);

	//接收数据
	recv(sc, buff, 1024, 0);
	struct data * pdata_get = (struct data *) buff;
	printf("recv data: %d %s \n", pdata_get->cmd, pdata_get->content);

	closesocket(sc);
	WSACleanup();
	getchar();
	return 0;
}

分享到:
评论

相关推荐

    Erlang高级应用和原理

    Erlang与传统的操作系统如Unix相比,具有独特的设计哲学。在Unix中,程序通常由C++编写,包括函数、类、模块和可执行文件。而在Erlang中,它的结构更像是一系列的函数、模块和BEAM文件,这些文件由Erlang虚拟机(VM...

    fast_xml:基于Fast Expat的Erlang XML解析库

    NIFs是Erlang的一项特性,允许编写C/C++代码来扩展Erlang的功能。当ErlangVM执行到NIF时,会暂停Erlang代码并跳转到C代码执行。NIFs提高了性能,特别是在需要低级别I/O或高性能计算的场景中。`fast_xml`使用NIFs来...

    C++服务程序员岗位职责.docx

    3. **开发经验**:岗位要求至少2年的游戏服务端开发经验和至少1款已上线项目,以及3年以上C/C++编程经验,同时精通erlang开发语言。 4. **网络编程**:必须精通TCP/IP协议,擅长网络编程,具备通讯协议设计经验,这...

    thrift-0.13.0.zip

    windows), 而Thrift最特别之处在于它是跨语言的:你可以用几乎所有流行语言(C++, Java, Python, PHP, Ruby, Erlang, Perl, Haskell, C#, Cocoa, JavaScript等等)来实现通讯过程,好处就是你不用为编程语言发愁;...

    云风:大世界网络游戏服务器的构架deepcold.ppt

    10. **开发经验与教训**:避免过度依赖特定技术(如C++),防止设计模式滥用,强调数据文本化,简化单个任务,不预先设计未明确的需求。 通过以上这些设计,云风的大世界网络游戏服务器构架能够有效地支撑大规模...

    thrift,facebook实例

    使用不同开发语言开发的系统可以通过该框架实现彼此间的通讯,开发者只需编辑一份thrift脚本,即可自动获得其它开发语言的代码(比如 c++ java python ruby c# haskell ocmal erlang cocoa php squeak). thrift侧重点...

    XMPP学习入门笔记

    - Jreen:一个使用C++和Qt框架实现的XMPP客户端库。 20. **ADA** - AXMPP:一个使用ADA语言实现的XMPP客户端库。 #### 三、XMPP的社区参与 由于XMPP市场的动态性和不断变化的特点,任何列出的XMPP服务器、客户端...

    开源项目研究与应用小结.pdf

    - **优点**:遵循AMQP协议,由Erlang实现,Spring社区活跃。 - **缺点**:需要掌握AMQP协议,Erlang语言可能不熟悉。 ##### 4. ZeroMQ - **优点**:轻量级、高效,提供多种常见通讯模型的抽象,几乎支持所有主流...

    thrift入门

    Thrift 可以支持多种编程语言,如 C++、C#、Cocoa、Erlang、Haskell、Java、Ocaml、Perl、PHP、Python、Ruby、Smalltalk 等。 Thrift 的主要特点是支持多种编程语言、跨平台、跨语言的数据交换和 RPC 服务,可以...

    8种Nosql数据库系统对比

    随着信息技术的飞速发展与互联网应用的日益增多,传统的SQL数据库在处理大规模、高并发数据时逐渐暴露出不足之处。为了更好地满足现代应用场景的需求,NoSQL(Not Only SQL)数据库应运而生。NoSQL数据库以其灵活的...

    ActiveMQ教程

    注意代码中创建的`ActiveMQConnectionFactory`对象是ActiveMQ提供的连接工厂实现,用于与ActiveMQ服务器建立连接。 总结来说,ActiveMQ是使用广泛、功能强大的消息中间件。通过上述对ActiveMQ的介绍,我们可以了解...

Global site tag (gtag.js) - Google Analytics