NetCore使用DotNetCore.CAP框架

CAP 是一个EventBus,同时也是一个在微服务或者SOA系统中解决分布式事务问题的一个框架。它有助于创建可扩展,可靠并且易于更改的微服务系统。

这个项目中使用到SqlServer(数据库方面大同小异)、MongoDb、Kafka、RabbitMq以及Consul

1:引入相关Nuget就行

DotNetCore.CAP
DotNetCore.CAP.Kafka
DotNetCore.CAP.RabbitMQ
DotNetCore.CAP.SqlServer
DotNetCore.CAP.MongoDB
DotNetCore.CAP.Dashboard

2:在program中使用(我的项目是基于net6的)

builder.Services.AddDbContext<AppDbContext>(opt =>
{
    opt.UseSqlServer(builder.Configuration.GetSection("CAP:SqlServer").Value);
});
//Options, If you are using MongoDB
builder.Services.AddSingleton<IMongoClient>(new MongoClient(builder.Configuration.GetSection("CAP:MongoDB").Value));

builder.Services.AddCap(x =>
{
    // If you are using EF, you need to add the configuration:
    //x.UseEntityFramework<AppDbContext>(); //Options, Notice: You don't need to config x.UseSqlServer(""") again! CAP can autodiscovery.

    // If you are using ADO.NET, choose to add configuration you needed:
    x.UseSqlServer(builder.Configuration.GetSection("CAP:SqlServer").Value);
    //x.UseMySql("Your ConnectionStrings");
    //x.UsePostgreSql("Your ConnectionStrings");

    // If you are using MongoDB, you need to add the configuration:
    //x.UseMongoDB(opt => {
    //    opt.DatabaseConnection = builder.Configuration.GetSection("CAP:MongoDB").Value;
    //});  //注意,仅支持MongoDB 4.0+集群

    // CAP support RabbitMQ,Kafka,AzureService as the MQ, choose to add configuration you needed:
    x.UseRabbitMQ(opt =>
    {
        opt.HostName = "192.168.1.7";
        opt.Port = 5672;
        opt.UserName = "admin";
        opt.Password = "admin";
        opt.VirtualHost = "/";
    });
    x.UseKafka(opt =>
    {
        opt.Servers = "192.168.1.12:9092";
    });
    //x.UseRabbitMQ(builder.Configuration.GetSection("CAP:RabbitMQ").Value);
    x.UseDashboard();
    DiscoveryOptions discoveryOptions = new DiscoveryOptions();
    discoveryOptions.CurrentNodePort = 5173;
    builder.Configuration.Bind(discoveryOptions);
    // Register to Consul
    x.UseDiscovery(d =>
    {
        d.DiscoveryServerHostName = "localhost";
        d.DiscoveryServerPort = 8500;
        d.CurrentNodeHostName = "localhost";
        d.CurrentNodePort = 5222;
        d.NodeId = "1";
        d.NodeName = "fanlin";
        d.Scheme = "http";
        d.MatchPath = "/api/HealthCheck";
    });
    x.FailedRetryInterval = 10;//失败重试的间隔时间
    x.FailedRetryCount = 10;//失败重试的次数
    x.FailedThresholdCallback = info =>
    {
        Console.WriteLine("Publish Message Error::" + info.Message);
    };
    //x.UseKafka("ConnectionString");
    //x.UseAzureServiceBus("ConnectionString");
    //x.UseAmazonSQS();
});

3:新建PublishController,主要用于发布,代码很简单。代码如下

/// <summary>
    /// CAP的消费是自动消费的
    /// </summary>
    [Route("api/[controller]")]
    [ApiController]
    public class PublishController : ControllerBase
    {
        private static string _publishName = "FanlinCAPDemo.Servces.Test";
        private readonly ICapPublisher _capBus;
        private readonly IConfiguration _configuration;
        private readonly AppDbContext _appDbContext;

        public PublishController(ICapPublisher capPublisher, IConfiguration configuration, AppDbContext appDbContext)
        {
            _capBus = capPublisher;
            _configuration = configuration;
            _appDbContext = appDbContext;
        }
        [HttpGet]
        [Route("no/transaction")]//根目录
        public async Task<IActionResult> WithoutTransaction()
        {
            Console.WriteLine("普通----无事务");
            var user = _appDbContext.Users.Find("1");
            await _capBus.PublishAsync(_publishName, user);//应该把数据写到publish表
            return Ok();
        }
        [HttpGet]
        [Route("adonet/transaction")]
        public IActionResult AdonetWithTransaction()
        {
            Console.WriteLine("普通事务---连接字符串----事务");
            var user = _appDbContext.Users.Find("1");
            using (var connection = new SqlConnection(_configuration.GetSection("CAP:SqlServer").Value))
            {
                using (var transaction = connection.BeginTransaction(_capBus, true))
                {
                    //your business logic code

                    _capBus.Publish(_publishName, user);
                }
            }

            return Ok();
        }
        [HttpGet]
        [Route("ef/transaction")]
        public IActionResult EntityFrameworkWithTransaction()
        {
            Console.WriteLine("上下文---事务");
            var user = _appDbContext.Users.Find("1");
            //带header
            IDictionary<string, string?> dicHeader = new Dictionary<string, string?>();
            dicHeader.Add("Husband", "Fanlin");
            dicHeader.Add("Wife", "Baoting");
            dicHeader.Add("SumAge", "34");
            using (var trans = _appDbContext.Database.BeginTransaction(_capBus, autoCommit: true))
            {
                //your business logic code

                _capBus.Publish(_publishName, user, dicHeader);
            }

            return Ok();
        }
    }

4:新建ConsumerController 用于消费,代码很简单,需要注意,订阅需要标注特性CapSubscribe,其属性name就是发布者的_publishName,也可以顶一个多个group,在rabbitmq里面会自动新建相应的队列!代码如下

[Route("api/[controller]")]
    [ApiController]
    public class ConsumerController : ControllerBase
    {
        private readonly AppDbContext _context;
        private readonly IMongoClient _client;
        private readonly ICapPublisher _capBus;
        public ConsumerController(AppDbContext appDbContext, IMongoClient mongoClient, ICapPublisher iCapPublisher)
        {
            _context = appDbContext;
            _capBus = iCapPublisher;
            _client = mongoClient;
            var collection = _client.GetDatabase("MyCap").GetCollection<Users>("MyCap.User");
            collection.InsertOne(new Users()
            {
                UserID = "3",
                UserName = "范小包",
                UserPy = "fxb",
                UserPwd = "1234",
                department = "1",
                userRemark = "jkl",
                UserWb = "abc",
                headImageFile = "none"
            });
        }

        [NonAction]
        [CapSubscribe("FanlinCAPDemo.Servces.Test")]
        public void CheckReceivedMessage(Users users, [FromCap] CapHeader header)
        {
            Console.WriteLine($"{DateTime.Now} NoGroup Info:{users},Header:{header.Count}");
        }
        [NonAction]
        [CapSubscribe("FanlinCAPDemo.Servces.Test", Group = "Group1")]
        public void CheckReceivedMessageGroup(Users users)
        {
            Console.WriteLine($"{DateTime.Now} NoGroup Info:{users}");
            //这里不支持事务 使用事务会报错 
            //using (var session = _client.StartTransaction(_capBus, autoCommit: false))
            //{
                var collection = _client.GetDatabase("test").GetCollection<BsonDocument>("test.collection");
                collection.InsertOne( new BsonDocument { { "hello", "world" } });

                _capBus.Publish("sample.rabbitmq.mongodb", DateTime.Now);

                //session.CommitTransaction();
            //}
        }
    }

5:调用接口,查看kafka和rabbitmq以及相关数据库的变化

a:RabbitMQ:

这里自动新建了两个队列,如果有定义group已group属性为准,如果没有,名称是自定义的

 b:Kafka

CAP自动创建了3个topic

 c:SqlServer

数据库会自动生成名为cap的数据库,里面有两张表,其中publish是发布的数据,receive是接收到的数据

 

 d:再来看Mongodb,这里是MongoDB的数据库,如果要使用事务,事务里面的数据库必须先存在

 e:如果你需要使用consul,只需要启动consul.exe即可,CAP框架会自动将服务注册,访问你的服务http://ip:port/cap,然后就可以通过界面看到相关数据了

 

 

 

以上就是CAP框架的简单使用,这是大神杨晓东的作品,有关文档可参考

https://github.com/dotnetcore/CAP

介绍 - CAP

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
.NET Core IoT框架是一个为物联网(IoT)应用程序开发设计的开源框架。它提供了一个跨平台的、轻量级的运行时环境,可用于构建可靠、高效的IoT解决方案。 首先,.NET Core IoT框架支持多种硬件平台,包括嵌入式设备和单板计算机,例如树莓派和Arduino等。这使得开发人员可以在不同的环境中轻松部署和运行他们的应用程序,而无需担心硬件兼容性问题。 其次,框架提供了丰富的API和工具,使开发人员能够方便地访问和控制物联网设备的各种功能和传感器。例如,开发人员可以利用框架提供的API读取温度、湿度和光照等传感器数据,或者控制设备的LED灯、电机和继电器等输出设备。 此外,.NET Core IoT框架提供了强大的网络连接功能,使开发人员能够轻松地将物联网设备与云平台和其他设备进行通信。开发人员可以使用框架的网络库来实现远程传感器数据传输、设备控制和远程监控等功能。 还值得一提的是,.NET Core IoT框架具有高度的可扩展性和灵活性。开发人员可以根据自己的需求选择所需的模块和组件,以构建自定义的IoT解决方案。框架还支持使用不同的编程语言,例如C#和F#,以满足不同开发人员的喜好和技能。 总而言之,.NET Core IoT框架是一个功能强大且易于使用的开源框架,可帮助开发人员快速构建高效可靠的物联网应用程序。它的跨平台支持、丰富的API和工具,以及强大的网络连接功能使其成为物联网开发的理想选择。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值