使用方法的MethodHandle作为键来缓存无参数方法的结果是否安全?

时间:2011-03-04 04:04:15

标签: c# .net caching reflection methodinfo

在WCF客户端应用程序中,有许多无参数的方法,我们要将结果缓存到其中GetAllFoo()GetAllBar()。这些用于填充下拉列表等,结果在客户端的运行期间不会改变。

这些结果当前正由存储在资源文件中的唯一字符串缓存 - 例如,GetAllCountries()缓存在CountryKey资源上。仅当缓存不包含请求的密钥时才会调用该服务。

public T Get<T, V>(string key, Func<V, T> serviceCall, V proxy)
{
    if (!cache.Contains(key))
    {
        cache.Add(key, serviceCall(proxy));
    }   
    return cache.GetData(key) as T;
}

这很好,除了我们需要维护资源文件中的密钥,并且需要确保每个方法都使用正确的缓存密钥,否则会中断。旧的Control + C,Control + V在这里引起一些麻烦,我不想去检查每个调用此方法的地方。

所以问题是:

serviceCall委托上有一个Method属性,用于描述要执行的方法。这是一个MethodInfo实例,后者又包含MethodHandle属性。我是否正确地假设MethodHandle属性唯一且一致地标识了引用的方法?

我将包装器更改为

public T Get<T, V>(Func<V, T> serviceCall, V proxy)
{
    var key = serviceCall.Method.MethodHandle;
    // etc

适当地封装了缓存和关键问题,&amp;消除了对调用者做出正确事情的依赖。

  • 不关心MethodHandle是否在实例之间发生变化 - 缓存仅针对每个实例
  • 不关心MethodHandle是否在客户端之间保持一致 - 缓存是针对每个客户端
  • DO 关心MethodHandle在客户端的实例中是否一致 - 我实际上想要使用缓存,而不是每个请求都会导致新的服务调用并且缓存已满未使用的数据
  • DO 关心MethodHandle在客户端的实例中是不唯一的 - 我必须确保在使用包装器时返回正确的数据(和类型)。

1 个答案:

答案 0 :(得分:2)

MSDN,引用MethodHandle属性,说“此属性用于从非托管代码访问托管类,不应从托管代码调用。”

http://msdn.microsoft.com/en-us/library/system.runtime.interopservices._methodbase.methodhandle.aspx

建议:

  • 无论如何都要使用serviceCall.Method.MethodHandle,因为它应该在同一个AppDomain中是唯一的。
  • 研究serviceCall.Method.GetHashCode(),看看这是否适合作为你的缓存密钥。
  • 将您的缓存机制更改为字典,T&gt;并使用serviceCall作为实际的缓存键。 (这可能不起作用,具体取决于您的代码如何调用包装器方法。)