- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
假设我创建了一个如下所示的包装器类:
public class Foo : IFoo
{
private readonly IFoo innerFoo;
public Foo(IFoo innerFoo)
{
this.innerFoo = innerFoo;
}
public int? Bar { get; set; }
public int? Baz { get; set; }
}
这里的想法是 innerFoo
可能包装数据访问方法或类似昂贵的东西,我只想要它的 GetBar
和 GetBaz
方法被调用一次。所以我想围绕它创建另一个包装器,它将保存第一次运行时获得的值。
当然,这样做很简单:
int IFoo.GetBar()
{
if ((Bar == null) && (innerFoo != null))
Bar = innerFoo.GetBar();
return Bar ?? 0;
}
int IFoo.GetBaz()
{
if ((Baz == null) && (innerFoo != null))
Baz = innerFoo.GetBaz();
return Baz ?? 0;
}
但如果我使用 10 个不同的属性和 30 个不同的包装器来执行此操作,它会变得相当重复。所以我想,嘿,让我们把它变成通用的:
T LazyLoad<T>(ref T prop, Func<IFoo, T> loader)
{
if ((prop == null) && (innerFoo != null))
prop = loader(innerFoo);
return prop;
}
这几乎让我到达了我想要的位置,但不完全是,因为您不能ref
自动属性(或任何属性)。换句话说,我不能这样写:
int IFoo.GetBar()
{
return LazyLoad(ref Bar, f => f.GetBar()); // <--- Won't compile
}
相反,我必须更改 Bar
以具有显式支持字段并编写显式 getter 和 setter。这很好,除了我最终写了比我一开始写的更多的冗余代码。
然后我考虑了使用表达式树的可能性:
T LazyLoad<T>(Expression<Func<T>> propExpr, Func<IFoo, T> loader)
{
var memberExpression = propExpr.Body as MemberExpression;
if (memberExpression != null)
{
// Use Reflection to inspect/set the property
}
}
这很适合重构 - 如果我这样做,效果会很好:
return LazyLoad(f => f.Bar, f => f.GetBar());
但这实际上并不安全,因为不那么聪明的人(即我自己在 3 天后,当我不可避免地忘记这是如何在内部实现的时)可能会决定改写这个:
return LazyLoad(f => 3, f => f.GetBar());
这要么会崩溃,要么会导致意外/未定义的行为,具体取决于我编写 LazyLoad
方法的防御性程度。所以我也不太喜欢这种方法,因为它会导致运行时错误的可能性,而这些错误在第一次尝试时就可以避免。它还依赖于反射,尽管这段代码公认对性能不敏感,但在这里感觉有点脏。
现在我可以也决定全力以赴使用 DynamicProxy 做方法拦截而不必写任何代码,事实上我已经 在某些应用程序中这样做。但是这段代码驻留在许多其他程序集所依赖的核心库中,在如此低的级别引入这种复杂性似乎可怕是错误的。将基于拦截器的实现与 IFoo
接口(interface)分开,将其放入自己的程序集中并没有多大帮助;事实上,这个类仍然会在所有地方使用,必须被使用,所以这不是可以用一点 DI 魔法轻松解决的问题之一。
我已经想到的最后一个选择是使用如下方法:
T LazyLoad<T>(Func<T> getter, Action<T> setter, Func<IFoo, T> loader) { ... }
这个选项也很“meh”——它避免了反射,但仍然容易出错,并且它并没有真正减少太多重复。这几乎与必须为每个属性编写显式 getter 和 setter 一样糟糕。
也许我只是非常挑剔,但这个应用程序仍处于早期阶段,随着时间的推移它会大幅增长,我真的想保持代码干净利落。
底线:我陷入僵局,正在寻找其他想法。
有没有办法清理顶部的延迟加载代码,这样实现将:
ref
版本;Expression
版本;和换句话说,有没有办法只使用常规的 C# 语言功能和可能的一些小帮助程序类来做到这一点?还是我将不得不接受这里存在的权衡并从列表中删除上述要求之一?
最佳答案
如果你可以使用 .NET 4,你应该只使用 Lazy<T>
.
它提供您所追求的功能,此外,它是完全线程安全的。
如果您不能使用 .NET 4,我仍然建议您查看它,并“窃取”它的设计和 API。它使惰性实例化变得非常容易。
关于c# - 通用的、编译时安全的延迟加载方法的方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2655358/
是否有任何库或框架旨在促进从另一种成熟的编程语言中构建项目? 在 C++、java 等编程语言中指定逻辑、集合和复杂规则非常容易,但在 Makefile 中完成这些事情似乎是一场艰苦的战斗。我还没有深
我有这段代码可以用 clang 编译得很好(即使使用 -Weverything),但是 gcc 会发出错误。 #include #include #include using namespace
我有以下 block 头文件 BKE_mesh.h: /* Connectivity data */ typedef struct IndexNode { struct IndexNode *
我在我的一个项目中遇到了一个奇怪的问题。我的代码库依赖于一个外部库,其中包含一个名为 Dataset 的类. Dataset类私有(private)继承自 std::vector (其中 Sample
当使用 gcc、g++ 或 make 在终端中编译一个小型 C 或 C++ 项目时,我收到以下错误: /tmp/ccG1caGi.o: In function `main': main.c:(.tex
我正在尝试从 CVS 为 Windows 上的 Emacs 23.1.50 编译 CEDET,但在“第 6 步:打开 EDE...”时出现错误:“defvar:作为变量的符号值是无效的:cedet-m
我正在(重新)学习编程,我从 C 开始。我的 IDE(如果我可以这么说)是 Windows7 上的 cygwin(32 位)和 Visual-Studio 2010。我总是编译我用 gcc (cygw
我喜欢在模板类中使用本地类来执行类似“static if”的构造。但是我遇到了 gcc 4.8 不想编译我的代码的问题。但是 4.7 可以。 这个例子: #include #include #in
我有一个项目,必须仅使用 java 1.4 进行编译。但我计划使用mockito 编写一些单元测试。我想要一种在 pom 中指定的方法,以便 src/main/java 使用 jdk 1.4 编译,但
我想了解 PHP 编译过程是如何工作的。 假设我有一个名为funcs.php 的文件并且这个文件有三个函数,如果我include 或require 它,所有的在文件加载期间编译三个函数?或者源代码会被
编译工具链 我们写程序的时候用的都是集成开发环境 (IDE: Integrated Development Environment),集成开发环境可以极大地方便我们程序员编写程序,但是配置起来
当我编写一些 Scala 代码时,在尝试编译代码时收到一条奇怪的错误消息。我将代码分解为一个更简单的代码(从语义的角度来看这完全没有意义,但仍然显示了错误)。 scala> :paste // Ent
我正在编译一个 SCSS 文件,它似乎删除了我的评论。我可以使用什么命令来保留所有评论? >SASS input.scss output.css 我在 SCSS 中看到两种类型的注释。 // Comm
这是我的代码: #include typedef struct { const char *description; float value; int age; } swag
当您编译 grails war 时,我知道 .groovy 代码被编译为字节码类文件,但我不明白容器(例如 tomcat)如何在请求 GSP 时知道如何编译它们。容器了解 GSP 吗?安装在服务器上的
我正在努力将多个文件编译成一个通用程序。我收到一个错误: undefined reference to 'pi' 这是我的代码和 Makefile 的框架。我做错了什么?谢谢! 文件:calcPi.c
我尝试使用 LD_PRELOAD 来 Hook sprintf function ,所以我将打印到缓冲区的结果: #define _GNU_SOURCE #include #include int
我正在寻找最简单的方法来自动将 CoffeeScript 重新编译为 JS。 阅读documentation但仍然很难得到我想要的东西。 我需要它来监视文件夹 src/ 中的任何 *.coffee 文
我想使用定制waveformjs 。我发现this on SO但是,我不知道如何编译/安装波形来开始。我从 GitHub 克隆它并进行了更改,但是我不知道如何将其转换为 .js 文件。 最佳答案 为了
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我是一名优秀的程序员,十分优秀!