赞
踩
abp学习日记 初记
abp学习日记一(安装)
abp学习日志二(DDD)
abp学习日志三(实体&聚合根)
abp学习日志四(仓储)
abp学习日志五(领域服务)
abp学习日志六(模块化开发)
abp学习日志七(动态API)
abp学习日志八(多租户)
abp学习日志九(总结)
abp支持从Service映射到API,在Service 中有一个CurdAppService的泛型类,泛型类用到了几个参数
代码如下
(举例代码)
/*SystemApp:集合跟(实体) *SystemAppDto:查询结果dto *Guid:Key *PagedAndSortedResultRequestDto:abp给的默认分页排序查询RequestDto *CreateSystemAppDto:创建对象的Dto *UpdateSystemAppDto:更新对象的Dto */ public class SystemAppService : CrudAppService<SystemApp, SystemAppDto, Guid, PagedAndSortedResultRequestDto, CreateSystemAppDto, UpdateSystemAppDto>, ISystemAppService { private readonly IRepository<SystemApp, Guid> productRepository; private readonly ICurrentTenant currentTenant; public SystemAppService(IRepository<SystemApp, Guid> productRepository, ICurrentTenant currentTenant) : base(productRepository) { this.productRepository = productRepository; this.currentTenant = currentTenant; } }
看到上面的代码有那么多Dto,dto和聚合根(实体)是如何实现属性对应的呢?Abp使用的是AutoMap。
使用方法就是在项目中(Application项目)有一个XXXXAutoMapperProfile.cs的文件,直接在这里面写上
CreateMap<SystemApp,SystemAppDto>()
如果需要更多那就都写上。
官方文档也是这样说的。
如果你的项目使用的是Module模板,并且是2.2.1以上的版本,请注意,不能使用默认的,除非你的Dto和实体的属性丝毫不差,否则会出现异常。
MemberList
枚举类型有三个值,有兴趣可以都试试如果项目存在很多个Dto和实体,那么写Create会写到怀疑人生。可以试试反射,反射的前提是做好namespace,否则反射的方法写起来比较麻烦,我的namespace规则,Dto都在XXXx.Dto命名空间。我的聚合根都在XXX.Model命名空间。
public VShopApplicationAutoMapperProfile() { /* You can configure your AutoMapper mapping configuration here. * Alternatively, you can split your mapping configurations * into multiple profile classes for a better organization. */ var types = Assembly.Load("xxxr.xxx.Application.Contracts").GetTypes(); var domainTypes = Assembly.Load("xxx.xxx.Domain").GetTypes(); foreach (var type in types) { if (type.Namespace == "xxx.xxx.Dto") { if (type.Name.StartsWith("Create") || type.Name.StartsWith("Update")) { var domainTypeName = type.Name.Replace("Create", "").Replace("Update", "").Replace("Dto", ""); if (domainTypes.Any(t => t.Name == domainTypeName)) CreateMap(type, domainTypes.First(t => t.Name == domainTypeName), MemberList.None); } else { var domainTypeName = type.Name.Replace("Dto", ""); if (domainTypes.Any(t => t.Name == domainTypeName)) CreateMap(domainTypes.First(t => t.Name == domainTypeName), type, MemberList.None); } } } }
为什么会定位到这个项目呢,因为错误是发生在Service里面的。错误内容如下
2020-03-29 11:25:38.867 +08:00 [ERR] { "code": null, "message": "对不起,在处理你的请求期间,产生了一个服务器内部错误!", "details": null, "validationErrors": null } 2020-03-29 11:25:38.868 +08:00 [ERR] An exception was thrown while activating Castle.Proxies.SystemRoleServiceProxy. Autofac.Core.DependencyResolutionException: An exception was thrown while activating Castle.Proxies.SystemRoleServiceProxy. ---> Autofac.Core.DependencyResolutionException: None of the constructors found with 'Autofac.Core.Activators.Reflection.DefaultConstructorFinder' on type 'Castle.Proxies.SystemRoleServiceProxy' can be invoked with the available services and parameters: Cannot resolve parameter 'Volo.Abp.Domain.Repositories.IRepository`2[Sugar.VShop.Model.SystemRole,System.Guid] productRepository' of constructor 'Void .ctor(Castle.DynamicProxy.IInterceptor[], Volo.Abp.Domain.Repositories.IRepository`2[Sugar.VShop.Model.SystemRole,System.Guid], Volo.Abp.MultiTenancy.ICurrentTenant)'. at Autofac.Core.Activators.Reflection.ReflectionActivator.GetValidConstructorBindings(IComponentContext context, IEnumerable`1 parameters) at Autofac.Core.Activators.Reflection.ReflectionActivator.ActivateInstance(IComponentContext context, IEnumerable`1 parameters) at Autofac.Core.Resolving.InstanceLookup.Activate(IEnumerable`1 parameters, Object& decoratorTarget) --- End of inner exception stack trace --- at Autofac.Core.Resolving.InstanceLookup.Activate(IEnumerable`1 parameters, Object& decoratorTarget) at Autofac.Core.Resolving.InstanceLookup.Execute() at Autofac.Core.Resolving.ResolveOperation.GetOrCreateInstance(ISharingLifetimeScope currentOperationScope, IComponentRegistration registration, IEnumerable`1 parameters) at Autofac.Core.Resolving.ResolveOperation.ResolveComponent(IComponentRegistration registration, IEnumerable`1 parameters) at Autofac.Core.Resolving.ResolveOperation.Execute(IComponentRegistration registration, IEnumerable`1 parameters) at Autofac.Core.Lifetime.LifetimeScope.ResolveComponent(IComponentRegistration registration, IEnumerable`1 parameters) at Autofac.ResolutionExtensions.TryResolveService(IComponentContext context, Service service, IEnumerable`1 parameters, Object& instance) at Autofac.ResolutionExtensions.ResolveService(IComponentContext context, Service service, IEnumerable`1 parameters) at Autofac.ResolutionExtensions.Resolve(IComponentContext context, Type serviceType, IEnumerable`1 parameters) at Autofac.ResolutionExtensions.Resolve(IComponentContext context, Type serviceType) at Autofac.Extensions.DependencyInjection.AutofacServiceProvider.GetRequiredService(Type serviceType) at Microsoft.Extensions.DependencyInjection.ServiceProviderServiceExtensions.GetRequiredService(IServiceProvider provider, Type serviceType) at Microsoft.AspNetCore.Mvc.Controllers.ServiceBasedControllerActivator.Create(ControllerContext actionContext) at Microsoft.AspNetCore.Mvc.Controllers.ControllerFactoryProvider.<>c__DisplayClass5_0.<CreateControllerFactory>g__CreateController|0(ControllerContext controllerContext) at Microsoft.AspNetCore.Mvc.Infrastructure.ControllerActionInvoker.Next(State& next, Scope& scope, Object& state, Boolean& isCompleted) at Microsoft.AspNetCore.Mvc.Infrastructure.ControllerActionInvoker.InvokeInnerFilterAsync() --- End of stack trace from previous location where exception was thrown --- at Microsoft.AspNetCore.Mvc.Infrastructure.ResourceInvoker.<InvokeNextExceptionFilterAsync>g__Awaited|25_0(ResourceInvoker invoker, Task lastTask, State next, Scope scope, Object state, Boolean isCompleted)
大致的意思就是无法反射出对应的构造函数。
网上找到了一个文章很有启发意义 https://www.codetd.com/article/7994226
这个文章的意思是说没有做DbSet 导致实体无法映射,也就是说聚合根对应的仓储
无法反射,这个倒是挺符合道理的。
但是同样的代码放到 有UI的项目下可用,放到–no-ui的项目下不可用。
填坑。
需要在EntityFramework项目中添加如下代码
namespace Sugar.VShop.EntityFrameworkCore { [DependsOn( typeof(VShopDomainModule), typeof(AbpEntityFrameworkCoreModule) )] public class VShopEntityFrameworkCoreModule : AbpModule { public override void ConfigureServices(ServiceConfigurationContext context) { context.Services.AddAbpDbContext<VShopDbContext>(options => { //这里是需要添加的代码 options.AddDefaultRepositories(); /* Add custom repositories here. Example: * options.AddRepository<Question, EfCoreQuestionRepository>(); */ }); } public override void OnApplicationInitialization(ApplicationInitializationContext context) { base.OnApplicationInitialization(context); } } }
因为abp的模板文件就是添加到这里的,但是这个不合理,所以我把这段代码添加到了启动项目下。
不要让Service中包含App或者Application等关键字,比如有个服务叫SystemAppService,映射到API后会没有APP关键字,直接变成了XXX/System/{1}
https://www.codetd.com/article/7994226
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。