- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在我的应用程序设计中,我试图实现 Pagination作为Cross-Cutting Concern与 Decorator应用于 an implementation 的图案的 CQRS图案。
我还有一个 multi-layered architecture我认为分页不是业务逻辑的一部分(因此是一个横切关注点)。这是已经做出的决定,不应在本主题中讨论。
在我的设计中,意图是表示层可以使用具有特定封闭通用类型的分页查询
IQueryHandler<GetAllItemsQuery, PaginatedQuery<Item>>
具有以下签名:
public class GetAllItemsQuery : PaginatedQuery<Item>
public class PaginatedQuery<TModel> :
IQuery<PaginatedResult<TModel>>, IQuery<IEnumerable<TModel>>
public class PaginatedResult<TModel>
这个想法是消费者应该收到一个 PaginatedResult
对于特定模型,它包含分页项和一些元数据(例如,在未应用分页的情况下执行的查询的项目总数),以便 UI 可以呈现它的分页。
我设计的主要理念是查询处理程序应该只应用它的业务逻辑(例如获取所有项目)。它只描述了它将如何执行此操作,不一定非要执行查询。
在我的例子中,查询处理程序上的装饰器实际上对查询应用分页并执行它(例如,通过在 .ToArray()
查询上调用 Linq to Entities)。
我想要的是我的查询处理程序应该像这样实现:
public class GetAllItemsQueryHandler : IQueryHandler<GetAllItemsQuery, IEnumerable<Item>>
所以处理程序的返回类型是IEnumerable<Item>
.这样处理程序就被强制为 Single Responsible .我面临的问题可能是我使用 Simple Injector 的方式.因为我正在注册我的 IQueryHandler<,>
喜欢
container.Register(typeof(IQueryHandler<,>), assemblies);
这不会验证我的设计,因为明显的无效配置:我正在注入(inject) IQueryHandler<GetAllItemsQuery, PaginatedResult<Item>>
进入我的消费者,但实际上并没有实现它。相反,处理程序实现了 IQueryHandler<GetAllItemsQuery, IEnumerable<Item>>
.
因此,作为一种解决方案,我尝试实现一个 Interceptor并有条件地注册(注意 C# 7.0 local functions 的用法):
Type PaginationInterceptorFactory(TypeFactoryContext typeContext)
{
// IQueryHandler<TQuery, TResult> where TResult is PaginatedResult<TModel>
var queryType = typeContext.ServiceType.GetGenericArguments()[0]; // TQuery
var modelType = typeContext.ServiceType.GetGenericArguments()[1].GetGenericArguments()[0]; // TModel in PaginatedResult<TModel> as TResult
return typeof(PaginatedQueryHandlerInterceptor<,>).MakeGenericType(queryType, modelType);
}
bool PaginationInterceptorPredicate(PredicateContext predicateContext) =>
predicateContext.ServiceType.GetGenericArguments()[0].IsPaginatedQuery(); // if TQuery is of type PaginatedQuery<>
container.RegisterConditional(typeof(IQueryHandler<,>), PaginationInterceptorFactory, Lifestyle.Singleton, PaginationInterceptorPredicate);
但这给了我一个验证异常:
System.InvalidOperationException occurred
Message=The configuration is invalid. Creating the instance for type [TYPE] failed. This operation is only valid on generic types.
Source=SimpleInjector
StackTrace:
at SimpleInjector.InstanceProducer.VerifyExpressionBuilding()
at SimpleInjector.Container.VerifyThatAllExpressionsCanBeBuilt(InstanceProducer[] producersToVerify)
at SimpleInjector.Container.VerifyThatAllExpressionsCanBeBuilt()
at SimpleInjector.Container.VerifyInternal(Boolean suppressLifestyleMismatchVerification)
at SimpleInjector.Container.Verify()
Inner Exception 1:
ActivationException: This operation is only valid on generic types.
Inner Exception 2:
InvalidOperationException: This operation is only valid on generic types.
关于操作 是什么以及为什么它无效的异常并不是很清楚。也许我做错了什么?
拦截器的实现如下:
public class PaginatedQueryHandlerInterceptor<TQuery, TModel> : IQueryHandler<TQuery, PaginatedResult<TModel>>
where TQuery : PaginatedQuery<TModel>
{
private readonly IQueryHandler<TQuery, IEnumerable<TModel>> _queryHandler;
public PaginatedQueryHandlerInterceptor(IQueryHandler<TQuery, IEnumerable<TModel>> queryHandler)
{
_queryHandler = queryHandler;
}
public PaginatedResult<TModel> Handle(TQuery query)
{
return (dynamic) _queryHandler.Handle(query);
}
}
和装饰器:
public class PaginationQueryHandlerDecorator<TQuery, TResult> : IQueryHandler<TQuery, TResult>
where TQuery : class, IQuery<TResult>
{
private readonly IQueryHandler<TQuery, TResult> _decoratee;
public PaginationQueryHandlerDecorator(
IQueryHandler<TQuery, TResult> decoratee)
{
_decoratee = decoratee;
}
public TResult Handle(TQuery query)
{
query.ThrowIfNull(nameof(query));
var result = _decoratee.Handle(query);
if (query.IsPaginationQuery(out var paginatedQuery))
{
return Paginate(result, paginatedQuery.Pagination);
}
return result;
}
private static TResult Paginate(TResult result, Pagination pagination)
{
return Paginate(result as dynamic, pagination.Page, pagination.ItemsPerPage);
}
private static PaginatedResult<TModel> Paginate<TModel>(IEnumerable<TModel> result, int page, int itemsPerPage)
{
var items = result as TModel[] ?? result.ToArray();
var paginated = items.Skip(page * itemsPerPage).Take(itemsPerPage).ToArray();
return new PaginatedResult<TModel>
{
Items = paginated,
Count = items.Length
};
}
}
最佳答案
This is a decision already made and should not be discussed in this topic.
好吧....如果你坚持的话:)
但至少要防止这些查询返回 IEnumerable<T>
, 但返回 IQueryable<T>
反而。 IEnumerable<T>
的使用将导致从数据库返回所有数据,即使您翻页也是如此。
也就是说,我不确定您的代码有什么问题,但我想建议一种稍微不同的方法:
public class PagedQueryHandler<TQuery, TItem>
: IQueryHandler<PagedQuery<TQuery, TItem>, Paged<TItem>>
where TQuery : IQuery<IQueryable<TItem>>
{
private readonly IQueryHandler<TQuery, IQueryable<TItem>> handler;
public PagedQueryHandler(IQueryHandler<TQuery, IQueryable<TItem>> handler)
{
this.handler = handler;
}
public Paged<TItem> Handle(PagedQuery<TQuery, TItem> query)
{
var paging = query.PageInfo ?? new PageInfo();
IQueryable<TItem> items = this.handler.Handle(query.Query);
return new Paged<TItem>
{
Items = items.Skip(paging.PageIndex * paging.PageSize)
.Take(paging.PageSize).ToArray(),
Paging = paging,
};
}
}
这个通用 IQueryHandler
实现可以将分页查询映射到非分页查询。这里Paged<T>
和 PageInfo
和 PagedQuery<TQuery, TItem>
定义如下:
public class Paged<T>
{
public PageInfo Paging { get; set; }
public T[] Items { get; set; }
}
public class PageInfo
{
public int PageIndex { get; set; }
public int PageSize { get; set; } = 20;
}
public class PagedQuery<TQuery, TItem> : IQuery<Paged<TItem>>
where TQuery : IQuery<IQueryable<TItem>>
{
public TQuery Query { get; set; }
public PageInfo PageInfo { get; set; }
}
PageInfo
和 Paged<T>
源自此 Github 存储库:https://github.com/dotnetjunkie/solidservices/tree/master/src/Contract/
PagedQueryHandler<TQuery, TItem>
可以注册如下:
container.Register(typeof(IQueryHandler<,>), typeof(PagedQueryHandler<,>));
通过此类及其注册,您可以简单地将可分页查询处理程序注入(inject)消费者,例如:
public class ItemsController
{
IQueryHandler<PagedQuery<GetAllItemsQuery, Item>, Paged<Item>> handler;
public ItemsController(
IQueryHandler<PagedQuery<GetAllItemsQuery, Item>, Paged<Item>> handler)
{
this.handler = handler;
}
public ActionResult Index(PagedQuery<GetAllItemsQuery, Item> query)
{
return View(this.handler.Handle(query));
}
}
关于c# - 分页作为带有简单注入(inject)器的 cqrs 中的横切关注点,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46777396/
众所周知,jQuery 的 $.post 函数非常棒,但我遇到的问题是查看页面源代码的人可以查看数据的去向,从而移向该页面进行窥探,或者,上帝禁止找到保存所有内容的文件夹。所以我的问题是,谁知道如何隐
我在下面有这个程序,它执行简单的关注/取消关注功能。一切都很好,除了当我刷新页面时,只有行中的第一个用户保留正确的关注/取消关注按钮。示例 我可以关注 user1 user2 和 user3,但是当我
我想要创建的是一个关注者/关注系统,您不是简单地关注用户,而是关注他们共享的内容部分。几乎就好像您关注的是 Twitter 的“列表”或群组而不是人员。不过,有了这个,您就可以关注/取消关注用户共享的
这个问题已经有答案了: facebook social plug-in not showing up when added dynamically (2 个回答) 已关闭 7 年前。 使用 HTML
我正在构建一个编辑器,它使用 TImage 来显示图片,并具有鼠标事件来能够在图像上绘制、移动框和调整框大小。这一切都很完美。现在我正在尝试实现使用键盘上的箭头移动选定框的功能,但是A)TImage没
我有两个问题,请记住我是一个java新手1.我有一个使用 JFrame 创建 GUI 的类。JFrame 有 2 个面板,我使用 JSplitPane 添加了 问题是我可以设法将焦点设置在所需的 JP
我目前正在使用iOS应用程序进行开发,该应用程序会从流式API捕获一些推文。因此,我使用用户名和密码进行身份验证。除此之外,我想给用户提供在Twitter上关注某些人的机会。我创建了一个UIButto
有没有办法钩入Play evolutions framework这样当它成功从 n.sql 迁移时至 n+1.sql至 n+2.sql ...,它在 Play 应用程序中调用了一些成功后 Hook (
我的 gorm 中有文本模式为多行的文本框。我必须通过 jQuery 将 css 应用到该文本框。为此,我使用了以下脚本。 $(document).ready(function() {
我在强制关注动态生成的 JQuery 对话框内容中的文本字段时遇到问题。我已经在 google 上搜索过这一点,似乎如果 Jquery 对话框设置为模式,JQuery 将“窃取”文档级别的焦点。老实说
下午,我正在使用 PHP、HTML5 和 Bootstrap。我构建了一个分为 5 个选项卡的表单,该表单中有几个必填字段。所有必需的输入字段都是“文本”,并且还标有 required="requir
我创建了一个带有 GridView 的 WPF 页面。在 GridView 中,每行有 5 个可用的 TextBox。当我在第一行的第一个 TextBox 上输入数据,然后按 Tab 键时,焦点移动到
请给我Java中密码验证的正则表达式代码,它应该由一个大写字符、一个整数、一个后面的符号(@、#、$、%、^、&、+、=)和小字符组成。 我一直在尝试使用不同的独立正则表达式和一个组合的正则表达式。
我想在我的 mean-stack 网页上添加一个 Twitter 的关注按钮。我使用以下代码: https://jsbin.com/herikik/3/edit?html,output 在 Ma
在下添加如下代码后到我在 Tumblr 上的主题 .tail { position:fixed; bottom:0px; right:0px; margin-bottom:434px; margin-
我必须从 Angular 应用程序启动一系列窗口。我希望能够让用户单击主页上的按钮以使该窗口重新成为焦点。通常我会在 javascript 中使用类似以下内容来执行此操作: //Launch the
因此,我想显示一些用 AND 或 OR 连接的规则,并且我想为 AND 或 OR 添加颜色,如红色、绿色等。 Fruit = Apple AND Market = SuperMarket1 那么我应该
我正在开发 Windows 商店应用程序,我正在使用 ListView 控件动态添加数据。这些项目被添加到列表的末尾。 Scrollbar 在添加更多数据时出现。我想用底部的滚动条以编程方式突出显示最
(问题仅在 Ubuntu 中出现。在 Windows 中工作正常。我不知道在其他 Linux 环境中) 我已经使用 ComponentListener 的方法在对话框中调用 JTextField 中的
如何将焦点放在时间选择器元素上?我正在开发电视应用程序,因此需要远程操作。所以我需要关注每个元素。TimePicker 有 3 个元素 - 小时列、分钟列和 AM/PM 列。 那么我如何才能专注于这
我是一名优秀的程序员,十分优秀!