Actor是一种高并发处理模型,每个Actor都有着自己的状态有序消息处理机制,所以在业务处理的情况并不需要制定锁的机制,从而达到更高效的处理能性。XRPC是一个基于远程接口调用的RPC组件,它可以简单地实现高性能的远程接口调用;XRPC在创建远程接口时是支持针对接口创建对应的Actor实例。当创建接口Actor后,所有Client针对这一例实例Actor的所有方法调用都是有序处理。以下介绍如何在XRPC创建并使用Actor

什么场景适用于Actor

既然每个Actor都有着自己的状态和顺序处理机制,那可以针对这优点进行相关业务的展开;在棋牌游戏中的桌子可以是一个Actor,车票里的每一辆车可以是一个Actor,秒杀里的每一种商品是一个Actor。在这些Actor所有的操作都是有序进行,不存在锁,也不需要事务(通过EventSourcing来保障)和不会产生死锁;数据变更全内存操作,通过这样可以大提高业务的处理性能。

引用XRPC

Install-Package BeetleX.XRPC

定义Actor的RPC服务

XRPC支持的Actor服务功能是建立在EventNext之上,它的好处是直接接口行为的Actor创建,而不是传统的消息加接收方法,这样在应用设计和调用上也是非常方便灵活。接下来定义一个简单的Actor服务

  • 接口定义

       public interface IAmountService
    {
    Task<int> Income(int amount);
    Task<int> Payout(int amount);
    Task<int> Get();
    }

    以上是一个简单的帐记变更行为接口

  • 实现接口

       [Service(typeof(IAmountService))]
    public class AmountService : ActorState, IAmountService
    { private int mAmount; public override Task ActorInit(string id)
    {
    return base.ActorInit(id);
    } public Task<int> Get()
    {
    return mAmount.ToTask();
    } public Task<int> Income(int amount)
    {
    mAmount += amount;
    return mAmount.ToTask();
    } public Task<int> Payout(int amount)
    {
    mAmount -= amount;
    return mAmount.ToTask();
    }
    }
  • 启动对应的RPC服务

          private static XRPCServer mXRPCServer;
    
          static void Main(string[] args)
    {
    mXRPCServer = new XRPCServer();
    mXRPCServer.ServerOptions.LogLevel = LogType.Error;
    mXRPCServer.Register(typeof(Program).Assembly);
    mXRPCServer.Open();
    Console.Read();
    }

    以上代码是在默认端口9090上绑定RPC服务,可以通过运行日志查看服务启动情况

创建远程Actor调用

  • 创建RPC Client

     client = new XRPCClient("192.168.2.18", );
    client.Connect();

    以上代码是创建一个RPC客户端,通过它的Create可以创建接口代理

  • 创建接口Actor实例
    IAmountService henry = client.Create<IAmountService>("henry");
    IAmountService ken = client.Create<IAmountService>("ken");

    以上是针对IAmountService创建了两个Actor对象,这两个对象的操作都是相互隔离互不干扰;每个Actor对象中的方法在并发下都是有序执行,并不会产生线程安全问题,所以在不同方法中操作对像的数据成员都不需要锁来保证数据安全性。

测试

为了更好地验证Actor的隔离和并发安全性,简单地并发测试一下

             for (int i = ; i < concurrent; i++)
{
var task = Task.Run(async () =>
{
for (int k = ; k < requests; k++)
{
await henry.Income();
System.Threading.Interlocked.Increment(ref mCount);
}
});
tasks.Add(task);
task = Task.Run(async () =>
{
for (int k = ; k < requests; k++)
{
await henry.Payout();
System.Threading.Interlocked.Increment(ref mCount);
}
});
tasks.Add(task);
task = Task.Run(async () =>
{
for (int k = ; k < requests; k++)
{
await ken.Income();
System.Threading.Interlocked.Increment(ref mCount);
}
});
tasks.Add(task);
task = Task.Run(async () =>
{
for (int k = ; k < requests; k++)
{
await ken.Payout();
System.Threading.Interlocked.Increment(ref mCount);
}
});
tasks.Add(task);
}
await Task.WhenAll(tasks.ToArray());
double useTime = EventCenter.Watch.ElapsedMilliseconds - start;
Console.WriteLine($"Completed count:{mCount}|use time:{useTime}|rps:{(mCount / useTime * 1000d):###.00} |henry:{await henry.Get()},ken:{await ken.Get()}");

两个程序同时在本机跑了一下,在50并发的情况大概是11万RPS

服务中的Actor隔离性

服务是通过名称来实例化接口的不同Actor,同一服务即使多个Client同时对一名称的Actor进行创建服务也可以保证它的唯一性。

完整示例代码

https://github.com/IKende/XRPC/tree/master/Samples/Actors

.Net core使用XRPC创建远程接口的Actor对象的更多相关文章

  1. RMI(远程接口调用)

    1. RMI的原理: RMI系统结构,在客户端和服务器端都有几层结构. 方法调用从客户对象经占位程序(Stub).远程引用层(Remote Reference Layer)和传输层(Transport ...

  2. C#winForm调用WebService的远程接口

    Web Service 的创建简单编码.发布和部署 上一篇详细概述了WebService的创建,编码,发布和部署,那么作为客户端的程序如何访问远程端的WebService 接下来看一下具体步骤:   ...

  3. Android系统进程间通信(IPC)机制Binder中的Client获得Server远程接口过程源代码分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6633311 在上一篇文章中,我 们分析了And ...

  4. Android开发,Eclipse创建aidl接口时,出错

    Android开发中,当我们需要调用远程Service时,我们一般通过远程接口(RMI)来实现的,而Android的RMI需要AIDL(Android Interface Definition Lan ...

  5. .net core 在扩展中使用接口实例之IServiceProvider

    在.net core 2.0中,我们使用的对象实例大多数都是通过构造函数依赖注入进来的,但那是在一般的类中使用. 如果需要在静态/扩展类中使用某些服务类的对象实例,可以使用如下方式: 1.新建一个Se ...

  6. 带SSL证书的httpclient 远程接口工具类

    package com.iups.wx.util; import java.io.IOException; import java.io.UnsupportedEncodingException; i ...

  7. Atitit.远程接口 监控与木马   常用的api 标准化v2 q216

    Atitit.远程接口 监控与木马   常用的api 标准化v2 q216 1. 木马与远程接口 监控的常用的api2 1.1. 文件复制2 1.2. 屏幕定时截图2 1.3. 邮件发送2 1.4.  ...

  8. HttpClient远程接口调用-实名认证

    1.HttpClient远程接口调用 1)用户注册 注册按钮button提交表单时,要return false form表单 <!-- action="http://localhost ...

  9. HttpClient 远程接口调用方式

    远程接口调用方式HttpClient 问题:现在我们已经开发好了接口了,那该如何调用这个接口呢? 答:使用Httpclient客户端.   Httpclient简介 什么是httpclient Htt ...

随机推荐

  1. ES6学习之Reflect

    Reflect对象与Proxy对象一样,也是 ES6 为了操作对象而提供的新 API Reflect设计目的: 将Object对象的一些明显属于语言内部的方法(比如Object.definePrope ...

  2. 【java并发编程艺术学习】(五)第二章 java并发机制的底层实现原理 学习记录(三) 原子操作的实现原理学习

    章节介绍 主要包括 术语定义.处理器如何实现原子操作.Java如何实现原子操作: 原子(atomic)本意是 不能再进一步分割的最小粒子,“原子操作” 意为 不可被中断的一个或一系列操作. 术语定义 ...

  3. %.*s, printf

    %.*s_百度搜索 c语言%.*s是什么_百度知道 *用来指定宽度,对应一个整数 .(点)与后面的数合起来 是指定必须输出这个宽度,如果所输出的字符串长度大于这个数,则按此宽度输出,如果小于,则输出实 ...

  4. 判断页面是在pc端打开还是在移动端打开

    在项目开发中会遇到在不同的设备中打开页面是不同的,比如: 我在手机中打开一个网站和pc打开一个网站,页面是不同的 具体实施如下 //判断打开网站的终端 var ua = window.navigato ...

  5. Levenberg-Marquardt优化算法以及基于LM的BP-ANN

    一.LM最优化算法     最优化是寻找使得目标函数有最大或最小值的的参数向量.根据求导数的方法,可分为2大类.(1)若f具有解析函数形式,知道x后求导数速度快.(2)使用数值差分来求导数.根据使用模 ...

  6. C++经典题目:约瑟夫环问题

    问题描述: 有n个人围成一圈,顺序排号.从第一个人开始报数(1~3报数),凡报到3的人退出圈子,问最后留下的人原来排在第几号. 分析: 首先由用户输入人数n,然后对这n个人进行编号[因为如果不编号的话 ...

  7. BSGS(大小步)算法

    BSGS算法主要用于求解形如ax≡b(mod p)的式子中x的值. 在这里我们不妨设 x=k1*n-k2 这时我们就可以将式子转化为 ak1*n≡b*ak2(mod p) 这里的n我们设为√p,所以我 ...

  8. hdu1065

    #include <stdio.h> int main() { int t; double x, y; scanf("%d", &t); for; i < ...

  9. 关于java中的编码问题

    ok,今天搞了一天都在探索java字符的编码问题.十分头疼.最后终于得出几点: 1.网上有很多博客说判断一个String的编码的方法是通过如下代码;但其实这个代码完全是错的,用一种编码decode后, ...

  10. HTML5程序开发范例宝典 完整版 (韩旭等著) 中文pdf扫描版

    HTML5程序开发范例宝典紧密围绕编程者在编程中遇到的实际问题和开发中应该掌握的技术,全面介绍了利用HTML进行程序开发的各方面技术和技巧.全书共16章,内容包括HTML网页布局.HTML基本元素.H ...