abp(net core)+easyui+efcore实现仓储管理系统——ABP WebAPI与EasyUI结合增删改查之五(三十一)

abp(net core)+easyui+efcore实现仓储管理系统目录

abp(net core)+easyui+efcore实现仓储管理系统——创建应用服务(五)

abp(net core)+easyui+efcore实现仓储管理系统——EasyUI前端页面框架 (十八)

 
 
 

         在上一篇文章abp(net core)+easyui+efcore实现仓储管理系统——ABP WebAPI与EasyUI结合增删改查之四(三十) 中我们实现了新增组织部门信息功能,不过还存在一些BUG,如下图。“自动展开和子级”没有显示,“上级组织”下拉框中没有数据显示。今天我们来继续完善组织部门信息新增功能。

十、创建下拉框树

        1. 在Visual Studio 2017的“解决方案资源管理器”中,右键单击在领域层“ABP.TPLMS.Web.Mvc”项目中的Models目录。 选择“添加” > “新建文件夹”。并重命名为“Orgs”。

         2. 在Visual Studio 2017的“解决方案资源管理器”中,鼠标右键单击“Org”文件夹,然后选择“添加” > “新建项…”。 在“添加新项-ABP.TPLMS.Web.Mvc”对话框中,选择“类”,并将名称命名为TreeJsonModel.cs。代码如下。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;

 

namespace ABP.TPLMS.Web.Models.Orgs
{

    /// <summary>
    /// 构建Json数据源的数据格式,属性有id,test,children,这里名字不能够更改否则不能够读取出来

    /// </summary>
    public class TreeJsonViewModel
    {        

            /// <summary>
            /// ID
            /// </summary>
            public int id { get; set; }

            /// <summary>
            /// 分类
            /// </summary>
            public string text { get; set; }
/// <summary>
            /// 子类
            /// </summary>
            public List<TreeJsonViewModel> children { get; set; }

            /// <summary>
            /// 父ID
            /// </summary>
            public int parentId { get; set; }
            public string url { get; set; }
            public string state { get; set; }
    }

}
       3. 在Visual Studio 2017的“解决方案资源管理器”中,右键单击在领域层“ABP.TPLMS.Web.Mvc”项目中的Controller目录中找到OrgsController.cs文件,在此文件中输入创建下拉列表树的JSON代码。
[DontWrapResult]
[HttpGet]
public JsonResult GetJsonTree()
{

    PagedOrgResultRequestDto paged = new PagedOrgResultRequestDto();
    var classlist = _orgAppService.GetAll(paged).GetAwaiter().GetResult().Items;
    List<TreeJsonViewModel> list = LinqJsonTree(classlist,0);

    return Json(list);

}

 /// <summary>
/// 递归
 /// </summary>
/// <param name="list"></param>
/// <returns></returns>

private List<TreeJsonViewModel> LinqJsonTree(IReadOnlyList<OrgDto> orgs,int parentId)
{

   List<TreeJsonViewModel> jsonData = new List<TreeJsonViewModel>();
  List<OrgDto> classlist = orgs.Where(m => m.ParentId == parentId).ToList();

   classlist.ToList().ForEach(item =>
   {

       jsonData.Add(new TreeJsonViewModel
       {

          id = item.Id,
          children = LinqJsonTree(orgs, item.Id),
          parentId = item.ParentId,
          text = item.Name,
          url = string.Empty,
          state = parentId == 0 ? "open" : ""

       });
    });
       return jsonData;

 }

       4. 在Visual Studio 2017中按F5运行应用程序。登录之后,点击“[组织管理]”菜单,我们可以看到货物管理列表页面。然后点击“添加”按钮。如下图。

 

      5.我们发现我们需要的“自动展开和子级”两个Checkbox复选框没有显示,而且点击“上级组织”的下拉箭头,下拉列表中也没有任何数据。我们在浏览器中按F12。会发现如下图。

 

     6.对于上面的情况,是由于样式冲突造成的。我们重写对应的代码。代码如下。

<div id="divAddUpdOrg" class="easyui-dialog" closed="true" data-options="buttons: '#dlg-buttons'">
    <form name="OrgEditForm" role="form" novalidate class="form-validation">
        <table>
            <tr>
                <td><input type="hidden" name="Id" id="IDUpdate" /></td>
            </tr>
            <tr>
                <td>组织名称:</td>
                <td>

                    <input type="text" id="NameUpdate" name="Name" class="form-control input-sm" />
                </td>
            </tr>

            <tr>
                <td> 组织代码:</td>
                <td><input type="text" id="UpdBizCode" name="BizCode" class="form-control input-sm" /></td>

            </tr>
            <tr>

                <td>类型:</td>
                <td>
                    <input type="text" id="UpdType" name="Type" class="form-control input-sm" />
                </td>
            </tr>
            <tr>

                <td> 关区代码:</td>
                <td><input type="text" id="UpdCustomCode" name="CustomCode" class="form-control input-sm" /></td>
            </tr>
            <tr>
                <td>自动展开:</td>
                <td>
                    <div class="form-control input-sm">
                        <input type="checkbox" name="IsAutoExpand" value="true" id="UpdIsAutoExpand" 
class
="filled-in" checked /> <label for="UpdIsAutoExpand"></label> </div> </td> </tr> <tr> <td>子级:</td> <td> <div class="form-control input-sm"> <input type="checkbox" name="IsLeaf" value="true" id="UpdIsLeaf" class="filled-in" checked /> <label for="UpdIsLeaf"></label> </div> </td> </tr> <tr> <td>状态:</td> <td> <input type="text" id="UpdStatus" name="Status" class="form-control input-sm" /> </td> </tr> <tr> <td>上级组织:</td> <td> <input id="AddTree" name="ParentName" class="easyui-combotree" /> </td> </tr> <tr> <td>热键:</td> <td> <input id="UpdHotKey" name="HotKey" class="form-control input-sm" /> </td> </tr> <tr> <td>图标:</td> <td> <input id="UpdIconName" name="IconName" class="form-control input-sm" /> </td> </tr> <tr> <td> <input id="UpdParentId" name="ParentId" type="hidden" /> </td> </tr> <tr> <td>备注:</td> <td> <input type="text" id="RemarkUpdate" name="URemark" class="form-control input-sm" /> </td> </tr> </table> </form> </div>

 

 

 

<think>嗯,用户问的是在.NET Core ABP框架中如何对聚合根子实体进行增删改查同时进行。首先,我需要回忆一下ABP中的领域驱动设计(DDD)概念,特别是聚合根(Aggregate Root)和子实体的关系。聚合根负责维护其子实体的完整性和事务一致性,所以所有对子实体的操作都应该通过聚合根来完成。 接下来,用户提到的是同时进行增删改查,这意味着在一个事务中处理多个操作。需要确保这些操作在同一个工作单元内执行,以保证数据的一致性。ABP的工作单元(Unit of Work)机制会自动管理事务,只要在应用层方法上添加[UnitOfWork]特性即可。 然后,我需要考虑具体的实现步骤。首先,聚合根和子实体的定义是否正确。比如,聚合根类中应该有子实体的集合,通常是使用泛型集合如`Collection<T>`来确保正确的EF Core映射。子实体可能需要一个组合主键,包含聚合根的ID和自身的ID,这需要在DbContext的OnModelCreating方法中配置。 然后是应用服务的实现。用户需要获取聚合根,对子实体进行添加、更新、删除操作,然后通过仓储保存聚合根。这里需要注意,ABP的仓储默认跟踪聚合根的变化,所以当修改聚合根的子实体集合时,EF Core会自动检测这些变更。但是,直接操作集合可能不够明确,可能需要显式调用仓储的Update方法,或者确保变更被正确跟踪。 可能遇到的问题包括:如何正确删除子实体,是否需要从集合中移除还是标记为删除。在EF Core中,如果子实体是聚合根的一部分,删除子实体通常是从集合中移除,然后保存聚合根,EF Core会自动处理删除操作。但需要确认是否正确配置了级联删除或必要的映射。 另外,用户可能关心的是性能问题,比如多次数据库操作是否合并为一个。使用工作单元后,所有操作会在事务提交时一次性提交,所以不会有多次数据库往返的问题。 还需要考虑并发处理,确保在并发修改时数据的一致性。ABP的乐观并发控制可以通过配置实体的Version属性来实现,但用户的问题中没有提到,可能暂时不需要深入,但可以作为一个注意事项提及。 最后,示例代码的结构需要清晰,展示如何在一个应用服务方法中获取聚合根,修改子实体集合,然后保存。同时,要提醒用户注意仓储的使用,确保在更新聚合根时调用Update方法,尽管某些情况下EF Core的变更跟踪可能自动处理,但显式调用会更可靠。 可能还需要提到领域服务的使用,如果操作复杂,涉及多个聚合根或业务逻辑,可能需要引入领域服务来封装逻辑,保持应用层的简洁。不过用户的问题主要是增删改查,所以应用服务直接处理应该足够。 总结步骤:定义聚合根和子实体,配置DbContext,实现应用服务方法,处理子实体的增删改,利用工作单元确保事务。同时,代码示例要准确,比如使用_collection.RemoveAll进行删除,使用Add方法添加新实体,修改现有实体的属性等。</think>在ABP框架中处理聚合根子实体的批量操作时,需严格遵循领域驱动设计原则。以下是具体实现步骤和示例: 一、领域层定义(关键结构) ```csharp // 聚合根 public class Order : AggregateRoot<Guid> { public string OrderNo { get; set; } // 使用Collection保证EF Core正确映射 public ICollection<OrderItem> Items => _items; private readonly List<OrderItem> _items; } // 子实体(需组合主键) public class OrderItem : Entity<Guid> { public Guid OrderId { get; } // 聚合根ID public string ProductName { get; set; } public int Quantity { get; set; } // 必须包含无参构造函数 private OrderItem() { } internal OrderItem(Guid id, Guid orderId, string productName, int quantity) { Id = id; OrderId = orderId; ProductName = productName; Quantity = quantity; } } ``` 二、DbContext配置 ```csharp protected override void OnModelCreating(ModelBuilder modelBuilder) { // 配置组合主键 modelBuilder.Entity<OrderItem>(b => { b.HasKey(x => new { x.OrderId, x.Id }); b.HasOne<Order>().WithMany(x => x.Items).HasForeignKey(x => x.OrderId); }); } ``` 三、应用服务实现(核心逻辑) ```csharp public class OrderAppService : ApplicationService { private readonly IRepository<Order, Guid> _orderRepository; public async Task UpdateOrderItems(UpdateOrderItemsDto input) { // 获取聚合根(包含子项) var order = await _orderRepository .Include(o => o.Items) .FirstOrDefaultAsync(o => o.Id == input.OrderId); // 删除操作 order._items.RemoveAll(item => input.DeletedItems.Contains(item.Id)); // 修改现有项 foreach (var itemDto in input.UpdatedItems) { var item = order._items.First(i => i.Id == itemDto.Id); item.ProductName = itemDto.ProductName; item.Quantity = itemDto.Quantity; } // 新增项 foreach (var itemDto in input.NewItems) { order.AddItem( ObjectMapper.Map<OrderItemDto, OrderItem>(itemDto) ); } // ABP会自动跟踪变更(但推荐显式Update) await _orderRepository.UpdateAsync(order); } } ``` 四、关键技术要点 1. **事务控制**:ABP工作单元默认自动开启事务,确保原子性操作 2. **变更跟踪**:EF Core会自动检测集合变化,但需注意: - 必须通过聚合根访问子集合(保护内部集合) - 推荐使用`Include()`预加载子项 3. **性能优化**:建议添加`[UnitOfWork(isTransactional: true)]`特性 4. **并发控制**:可通过聚合根添加`[ConcurrencyCheck]`版本字段 、最佳实践建议 1. 在聚合根内部添加领域方法控制操作: ```csharp public class Order { internal void AddItem(OrderItem item) { // 领域逻辑校验 if (_items.Count >= 100) throw new BusinessException("ExceedMaxItems"); _items.Add(item); } } ``` 2. 使用AutoMapper配置: ```csharp CreateMap<OrderItemDto, OrderItem>() .ForMember(dest => dest.OrderId, opt => opt.Ignore()); ``` 3. 推荐DTO结构: ```csharp public class UpdateOrderItemsDto { public Guid OrderId { get; set; } public List<OrderItemDto> NewItems { get; set; } public List<OrderItemDto> UpdatedItems { get; set; } public List<Guid> DeletedItems { get; set; } } ``` 这种实现方式保证了: - 事务完整性:所有操作在单个事务中提交 - 领域封装性:业务规则内聚在聚合根中 - 性能优化:单次数据库往返完成所有操作 - 数据一致性:严格遵循聚合根边界约束 实际开发中应根据业务场景适当添加: 1. 乐观并发控制 2. 领域事件发布 3. 操作审计日志 4. 验证逻辑前置检查
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值