我正在寻找一种在访问propertys getter和setter时自动检查访问权限的好方法。
这仅仅是出于好奇和实验目的,所以表现并不重要。
我找到的一种方法如下(例子仅适用于吸气剂):
public class DynamicPropertyClass : DynamicObject
{
/// <summary>
/// I want this to work for a default auto-property
/// </summary>
public string TestString { get; set; }
public override bool TryGetMember(GetMemberBinder binder, out object result)
{
if (StaticSecurityContext.CheckSecurityCondition(binder.GetType(), binder.Name))
{
return base.TryGetMember(binder, out result);
}
else
{
throw new SecurityException();
}
}
}
如果未给出安全条件,则使用动态对象中止。 这种方法的问题是,实际存在于类中的属性不会被TryGetMember方法处理。 所以这种方法迫使我只处理代码中实际不存在的属性。
正如您所看到的,动态功能与我的方法并不真正相关。 我希望类似的东西能够用我的代码编写代码,不需要支持动态功能。
那么,是否有另一种方法,可以在c#中,不在每个属性上应用属性或向getter和setter添加自定义代码?
答案 0 :(得分:1)
您可以使用代理模式。 它存在很多由AOP框架提供的实现,但如果性能不是条件,你可以简单地尝试远程API使用的Transparent / RealProxy。
https://msdn.microsoft.com/fr-fr/library/system.runtime.remoting.proxies.realproxy(v=vs.110).aspx
答案 1 :(得分:1)
这是一个老问题,但我认为贡献很有意思,因为没有提到DynamicProxy。 您也可以使用DynamicProxy nuget package from Castle.Core。
您可以拦截对类的所有虚拟属性的Get和Set方法的调用。
I have written a Gist explaining the concept,跟踪 何时访问和设置属性。
以下是拦截器的样子。
public class GetSetInterceptor : Interceptor
{
protected override void ExecuteBefore(IInvocation invocation)
{
}
protected override void ExecuteAfter(IInvocation invocation)
{
if(invocation.Method.Name.StartsWith("get_") || invocation.Method.Name.StartsWith("set_"))
{
var target = invocation.InvocationTarget as TrackedObject;
if(target == null)
{
return;
}
var methodInvoked = invocation.Method.Name.Split("_");
switch (methodInvoked[0])
{
case "get":
target.AddEvent(EventType.Get, methodInvoked[1], invocation.ReturnValue);
break;
case "set":
target.AddEvent(EventType.Set, methodInvoked[1], invocation.Arguments[0]);
break;
}
}
}
}
代理的创建是这样完成的:
ProxyGenerator generator = new ProxyGenerator();
var tracked = generator.CreateClassProxy<TrackedClass>(new GetSetInterceptor());
TrackedClass必须具有虚拟属性:
public class TrackedClass : TrackedObject
{
public virtual string SomeContent { get; set; }
public virtual int SomeInt { get; set; }
}
测试在这里(使用xUnit):
public class GetterSetterInterceptorTests
{
[Fact]
public void SuccessFullyRegisterGetAndSetEvents()
{
ProxyGenerator generator = new ProxyGenerator();
var tracked = generator.CreateClassProxy<TrackedClass>(new GetSetInterceptor());
tracked.SomeContent = "some content";
Assert.Single(tracked.GetEvents());
var eventAfterSomeContentAssigned = tracked.GetEvents().Last();
Assert.Equal(EventType.Set, eventAfterSomeContentAssigned.EventType);
Assert.Equal("some content", eventAfterSomeContentAssigned.Value);
Assert.Equal("SomeContent", eventAfterSomeContentAssigned.PropertyInfo.Name);
tracked.SomeInt = 1;
Assert.Equal(2, tracked.GetEvents().Count);
var eventAfterSomeIntAssigned = tracked.GetEvents().Last();
Assert.Equal(EventType.Set, eventAfterSomeContentAssigned.EventType);
Assert.Equal(1, eventAfterSomeIntAssigned.Value);
Assert.Equal("SomeInt", eventAfterSomeIntAssigned.PropertyInfo.Name);
var x = tracked.SomeInt;
Assert.Equal(3, tracked.GetEvents().Count);
var eventAfterSomeIntAccessed = tracked.GetEvents().Last();
Assert.Equal(EventType.Get, eventAfterSomeIntAccessed.EventType);
Assert.Equal(1, eventAfterSomeIntAccessed.Value);
Assert.Equal("SomeInt", eventAfterSomeIntAccessed.PropertyInfo.Name);
}
}