- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我发现几乎每个类( Controller 、 View 、HTML 帮助程序、服务等)我都需要当前登录的用户数据。所以我考虑创建一个“环境上下文”而不是直接注入(inject) IUserService 或用户。
我的方法看起来像那样。
public class Bootstrapper
{
public void Boot()
{
var container = new Container();
// the call to IUserService.GetUser is cached per Http request
// by using a dynamic proxy caching mechanism, that also handles cases where we want to
// invalidate a cache within an Http request
UserContext.ConfigureUser = container.GetInstance<IUserService>().GetUser;
}
}
public interface IUserService
{
User GetUser();
}
public class User
{
string Name { get; set; }
}
public class UserContext : AbstractFactoryBase<User>
{
public static Func<User> ConfigureUser = NotConfigured;
public static User ActiveUser { get { return ConfigureUser(); } }
}
public class AbstractFactoryBase<T>
{
protected static T NotConfigured()
{
throw new Exception(String.Format("{0} is not configured", typeof(T).Name));
}
}
示例用法:
public class Controller
{
public ActionResult Index()
{
var activeUser = UserContext.ActiveUser;
return View();
}
}
我的方法是否正确,还是遗漏了什么?您有更好的解决方案吗?
更新:
User 类的更多详细信息:
public class User
{
string Name { get; set; }
bool IsSuperUser { get; set;}
IEnumerable<AzManOperation> Operations { get; set}
}
在 Controllers 中,我们需要检查用户是否是 super 用户,以便仅向 super 用户提供一些额外的功能。
public class BaseController : Controller
{
private readonly IUserService _userService;
BaseControler(IUserService userService)
{
_userService = userService
}
public User ActiveUser
{
get { return _userService.GetUser(); }
}
}
在 Views 中,我们选中 Operations 以仅在用户有权这样做时显示编辑或删除按钮。 View 从不使用 DependencyResolver,而是使用 ViewBag 或 ViewModel。我的想法是实现一个自定义的 ViewBasePage 并提供一个 ActiveUser 属性,以便 View 可以轻松访问。
在 HtmlHelpers 中,我们根据 IsSuperUser 和 Operations 呈现控件(传入 User 对象或使用 DependencyResolver)。
在服务类中我们也需要这些属性。例如,确定购物篮是否有效(检查是否允许用户购买不在标准列表中的商品)。所以 Service 类依赖于 IUserService
并调用 GetUser()
。
在 Action Filters 中强制用户更改密码(仅当它不是 super 用户且 User.ForcePasswordChange 为真时)。这里我们使用 DependencyResolver。
我希望有一种更简单的方法来获取用户对象,而不是使用 DependencyResolver.Current.GetService().GetUser() 或使用诸如 ViewBag.ActiveUser = User
之类的方法。User 对象几乎无处不在,需要检查权限等。
最佳答案
In Views we check Operations to only show an edit or delete button if the user has the right to do so.
View 不应该做这个检查。 Controller 应该将 View 模型返回到包含 bool 属性的 View ,这些属性说明这些按钮是否应该可见。使用 IsSuperUser
返回一个 bool 值已经在 View 中广为人知。 View 不应该知道它应该为 super 用户显示某个按钮:这取决于 Controller 。 View 应该只被告知要显示什么。
如果几乎所有 View 都有此代码,则有一些方法可以从您的 View 中提取重复部分,例如使用部分 View 。如果您发现自己在许多 View 模型上重复这些属性,也许您应该定义一个信封 View 模型(将特定模型包装为 T
的通用 View 模型)。 Controller 可以创建其 View 模型,而您可以创建服务或横切关注点将其包装在信封中。
In Service Classes we need those properties too. For instance to decide if a basket is valid or not
在这种情况下,您谈论的是验证,这是一个横切关注点。您应该改用装饰器来添加此行为。
关于c# - 使用静态工厂 Func<T> 为 ASP.NET 应用程序创建 "Ambient Context"(UserContext),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17011412/
我在等待异步功能完成时苦苦挣扎。特别是,我发现这两种方法在测试继续之前等待异步函数完成,但不知道其中的区别(如果有区别的话):。我的目标是在实际测试开始之前等待bepreEach()块中的两个异步函数
我在等待异步功能完成时苦苦挣扎。特别是,我发现这两种方法在测试继续之前等待异步函数完成,但不知道其中的区别(如果有区别的话):。我的目标是在实际测试开始之前,在beforeEach()块中等待两个Ja
为什么是Func<>从 Expression> 创建通过 .Compile() 比仅使用 Func<> 慢得多直接声明? 我刚从使用 Func 更改为直接声明为从 Expression> 创建的一个在
我正在创建一个 Validator类(class)。我正在尝试实现 Linq SelectMany我的验证器的扩展方法能够使用 Linq 查询组合表达式并验证最终结果,即使基础值发生变化也是如此。 下
function sum(a) { let currentSum = a; function f(b) { currentSum += b; return f; }
我只知道i = i++;是未定义的行为,但是如果一个表达式中调用了两个或多个函数,并且所有功能是一样的。是未定义吗?例如: int func(int a) { std::cout << a <
我如何定义一个对象,以便作用于它的任何函数都作用于它的一个字段?这可能吗? class Mydata(object): def __init__(self, val): sel
这个问题一直很有趣,尽管它不一定很整洁。我有以下代码: import random def d(m): return random.randint(1, m) print(3*d(6)) 这将
能否请您解释一下使用 func.apply(null, arr) 的区别?和 func.apply(this, arr)在下面的代码示例中? var Foo = function() { fu
我想收集/运行任务,然后对它们执行 Task.WhenAll。 var tasks = new List(); foreach (var thing in things) { tasks.Add(
我有以下代码: static Func s_objToString = (x) => x.ToString(); static Func s_stringToString = s_objToStrin
相关主题: Create Expression> dynamically 我在互联网上搜索但所有样本都解释了 Expression来自 T ? 谢谢 编辑 1) T输入我的代码在运行时确定,例如我想用
我正在尝试使用 LinqKit 动态生成 linqtosql 查询.在将表达式发送到 LinqKit 之前,我想检查要为预测添加的字段。所以我想出了一些想法,比如 Expression> GetPr
我遇到了一些麻烦,我写了一个 Func,IDE 不喜欢我在 Func 体内调用 Func ,我不太明白为什么,因为如果我将这个确切的代码放在方法体中,并使用相同的返回类型和参数,那么它就可以工作。 代
我现在正在学习使用 Class 语法来创建 React 组件,请注意我现在必须声明这样的方法: class Foo extends React.Component { ... bar
下面两种说法有区别吗?他们都工作。 if ( ((Func)(()=>true))() ) { .... }; if ( new Func(()=>true)()) { .... }; 最佳答案 不,
这个问题在这里已经有了答案: Difference between func() and (*this).func() in C++ (4 个答案) 关闭 6 年前。 如果我有一个带有虚函数而没有自
主要问题是“是否可以将任何类型的 func 作为参数传递以及如何传递?”。我正在学习 Go 并且想像这样制作我自己的异步包装函数: func AsyncFunc(fn func(), args ...
有没有简单的转换方法 Expression> 到 Expression> T从哪里继承自TBase? 最佳答案 只要 T 派生自 TBase,您就可以使用原始表达式的主体和参数直接创建所需类型的表达式
我有以下方法,其中 T 在 Func 中使用: public void DoSomething(string someString, Func someMethod) { if(some
我是一名优秀的程序员,十分优秀!