Akka边学边写(2)-- Echo Server
EchoServer
上篇文章里,我们用Akka写了一个简单的HelloWorld样例,对Akka(以及Actor模式)有了初步的认识。本文将用Akka写一个EchoServer,看看在Actor的世界里,怎样使用TCP协议。
Github项目
照例。EchoServer的代码被放在了Github上。EchoServer比HelloWorld略微复杂一点,一共同拥有三个类。例如以下图所看到的:
Main
这次先从主类入手:
main()方法的第一行创建了一个Actor系统,名字为mySystem。接下来的四行代码。请看以下的详解。
TCP Manager
Akka将整个TCP层抽象为一个Actor,这个Actor就是TCP Manager。在main()方法的第二行代码里,我们给Actor系统加入了一个TCP Manager,并和它的经纪人取得了联系:
ActorRef tcpManager = Tcp.get(mySystem).getManager();
如今我们的Actor系统看起来是这样:
Props
Props应该是Properties的缩写,它的作用是告诉Actor系统怎样创建一个Actor。Props提供了四个工厂方法来创建Props实例,例如以下所看到的:
public static Props create(Class<? > type, Object[] os)
public static <T extends Actor> Props create(Class<T> type, Creator<T> crtr)
public static <T extends Actor> Props create(Creator<T> crtr)
public static Props create(Class<? > type, Seq<Object> seq)
main()方法的第三行使用了上面的第一个工厂方法。这个工厂方法有两个參数:一个Class。和一个数组(实际上是vararg)。
第一个參数表示Actor的class,第二个參数是传递给Actor构造函数的參数。这样,Actor系统就知道怎样依据Props创建(用反射调用构造函数)一个Actor实例。main()方法的第四行创建了一个Accepter,起名为accepter。Accepter须要用到tcpManager,这也是我们将tcpManager当做第二个參数传递给Props.create()方法的缘故。如今Actor系统变成了以下这样(我用虚线箭头来表示Actor之间的依赖关系):
main()方法的第五行给accepter发了一个消息:整数12345。告诉它绑定port12345,開始监听TCP连接,准备echo服务。
Accepter
Accepter负责监听port,然后把收到的tcp连接交给Handler去处理。Accepter略微有点复杂,以下是所有代码:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvenhob28=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="">
构造函数是为了建立对tcpManager的依赖,以下具体介绍onReceive()方法:
- 假设收到的消息是Integer类型(port)。我们给tcpManager发送绑定消息,通知它绑定某个port。
- 假设tcpManager成功绑定port。它会回应已绑定消息(Bound)。
- 否则,回应绑定失败消息(CommandFailed)。accepter通过调用getContext().stop()方法结束自己。
- 成功绑定port后。假设有连接到达,则会收到Connected消息。
accepter通过调用getContext().actorOf()创建一个Handler,然后把它注冊给tcpManager。
之后tcpManager就会把与这个连接相关的消息发给handler,换句话说,这个连接被handler接管。
Actor之间的父子关系
在Actor系统内部。Actor们之间并非仅仅有简单的依赖(或引用)关系,而是能够形成父子关系:Actor能够创建子Actor。然后把子任务分配给它们去处理。
一个Actor的Children是通过它的context来管理的,上面代码中。accepter通过getContext()来获得它自己的context,然后通过调用context的actorOf()方法创建子Actor。如果如今有两个client连接到了我们的echoserver,那么整个Actor系统将是以下这样(Actor系统内的实线箭头表示父子关系):
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvenhob28=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="">
Handler
最后一个类是Handler,它比較简单,代码例如以下所看到的:
Handler仅仅处理两种消息:
- Received消息告诉handler有消息到达,由于是echoserver。所以并不关心到达的消息里面是什么内容。getSender()方法能够得到消息的发送者。也就是tcpManager。然后给tcpManager发一条Write消息。告诉它把数据原封不动的写回给client。
- ConnectionClosed告诉handler连接已经断开,handler通过调用context的stop()方法结束自己短暂的生命。
測试EchoServer
启动EchoServer。然后能够通过telnet命令进行測试,这里就不具体说明了。
结论
要想用Akka写一个echoserver。还是挺难的,须要了解Akka的非常多方面。可是相对于Socket,或者NIO来说,Akka版的echoserver显然更简单。
Akka边学边写(2)-- Echo Server的更多相关文章
- Akka边学边写(4)-- MiniRPG
前面几篇文章用Akka写了HelloWorld和EchoServer,为了更进一步学习Akka,本文将会实现一个非常小的RPG游戏server:MiniRPG. 游戏逻辑 由于是迷你RPG,所以逻辑非 ...
- Akka边学边写(1)-- Hello, World!
Akka Akka是什么呢?直接引用Akka站点上面的描写叙述吧: Akka is a toolkit and runtime for building highly concurrent, dist ...
- Akka边学边写(3)-- ByteString介绍
Akka的IO层设计能够參考这篇文档,本文简介一下ByteString的设计. Immutable消息 Actor之间是通过消息沟通的.但为了避免同步问题,消息必须是Immutable. 因此.Akk ...
- netty写Echo Server & Client完整步骤教程(图文)
1.创建Maven工程 1.1 父节点的pom.xml代码(root pom文件) 1 <?xml version="1.0" encoding="UTF-8&qu ...
- [Python]再学 socket 之非阻塞 Server
再学 socket 之非阻塞 Server 本文是基于 python2.7 实现,运行于 Mac 系统下 本篇文章是上一篇初探 socket 的续集, 上一篇文章介绍了:如何建立起一个基本的 sock ...
- [z]libevent入门教程:Echo Server based on libevent 不指定
[z]https://www.felix021.com/blog/read.php?2068 花了两天的时间在libevent上,想总结下,就以写简单tutorial的方式吧,貌似没有一篇简单的说明, ...
- libevent 入门教程:Echo Server based on libevent(转)
下面假定已经学习过基本的socket编程(socket, bind, listen, accept, connect, recv, send, close),并且对异步/callback有基本的认识. ...
- 网络编程-UDP echo server
1. UDP简介 UDP 和TCP 的区别包括 1. 面向字节流和面向报文 2. TCP必须要建立连接后才能进行数据交换,但是UDP则并没有连接的建立和释放过程.面向字节流说明,tcp报文段(segm ...
- 当标签上写了runat="server" 后,<%%>就会无效
当标签上写了runat="server" 后,<%%>就会无效 //这是错误的写法 <input type="hidden" runat=&q ...
随机推荐
- [Regex Expression] Match mutli-line number, number range
/^-?\d{,}\.\d+$/gm
- iOS绘图教程
本文是<Programming iOS5>中Drawing一章的翻译,考虑到主题完整性,翻译版本中加入了一些书中未涉及到的内容.希望本文能够对你有所帮助.(本文由海水的味道翻译整理,转载请 ...
- Java中String直接赋字符串和new String的区别
解析Java中的String对象的数据类型 1. String是一个对象. 因为对象的默认值是null,所以String的默认值也是null:但它又是一种特殊的对象,有其它对象没有的一些特性. 2. ...
- Spring项目的建立-移植流程(非入门教程)
Creat by Zhou yong in 2016/4/15/19:00 jar包 java 1.7 tomcat的两个jar包 2个文件上传的jar包 json支持的jar包 hibernate- ...
- 面试前的准备---C#知识点回顾----04
播下的种子,慢慢开始发芽收获了,陆陆续续offer就来了,该轮到我挑的时候了 今天面试的一家公司,技术问的相对宽广和细致,程度令人发指 1.谈谈ViewState 这个问题,回答的好,工资翻一级 基本 ...
- J2EE 中 The function valueOf must be used with a prefix when a default namespace is not specified 错误
jsp页面中,JSTL El表达式字符串比较常用方法 fn:contains 判断字符串是否包含另外一个字符串 <c:if test="${fn:contains(name, sear ...
- python re 正则
*:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: 0 !important; } /* ...
- SqlBulkCopy类进行大数据(一万条以上)插入测试
好多天没写博客了,刚刚毕业一个多月! 关于上一篇博客中提到的,在进行批量数据插入数据库的时候可以通过给存储过程传递一个类型为Table的参数进行相关操作,在这个过程中本人没有进行效率的测试.后来查找发 ...
- Pull生成&解析
开篇注意,由于解析有可能有大文件非常耗时,建议另开一个线程解析也可以不开具体视情况而定 Pull生成 1.通过xml获得序列化的实例 XmlSerializer nxs = Xml.newSe ...
- windows2008无线网卡和.net3.5安装
今天在联想T420S笔记本上安装windows2008标准版,安装完成后部分驱动软件不能安装,要求.net framework3.5,下载.net3.5安装时提示应该用角色管理器安装. 根据提示打开服 ...