- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
首先,很抱歉,如果之前有人问过这个问题。我进行了相当全面的搜索,没有发现任何类似的东西,但我可能遗漏了一些东西。
现在回答问题:我试图通过反射调用构造函数,但没有成功。基本上,我有一个要克隆的对象,因此我查找其类型的复制构造函数然后调用它。这是我拥有的:
public Object clone(Object toClone) {
MethodBase copyConstructor = type.GetConstructor(
new Type[] { toClone.GetType() });
return method.Invoke(toClone, new object[] { toClone }); //<-- doesn't work
}
我这样调用上面的方法:
List<int> list = new List<int>(new int[] { 0, 1, 2 });
List<int> clone = (List<int>) clone(list);
现在,请注意我使用的调用方法是 MethodBase
的调用。 ConstructorInfo
提供了一个 invoke 方法,如果像这样调用它确实有效:
return ((ConstructorInfo) method).Invoke(new object[] { toClone });
但是,我想使用 MethodBase
的方法,因为实际上我不是每次都查找复制构造函数,而是将其存储在字典中,而字典包含方法和构造函数,所以它是 Dictionary<MethodBase>
, 不是 Dictionary<ConstructorInfo>
.我当然可以转换为 ConstructorInfo
正如我上面所做的,但我宁愿避免转换并使用 MethodBase
方法直接。我只是想不出正确的参数。
有什么帮助吗?非常感谢。
编辑
本杰明,
非常感谢您的建议。我实际上完全按照你在第二次编辑中的建议做了,除了(这是一个很大的“除了”)我的字典在哪里
class ClonerMethod {
public MethodBase method;
public bool isConstructor;
...
public Object invoke(Object toClone) {
return isConstructor ?
((ConstructorInfo) method).Invoke(new object[] { toClone }) : //<-- I wanted to avoid this cast
method.Invoke(toClone, null);
}
}
然后我调用ClonerMethod
的 invoke
根据我在字典中找到的内容。我没有添加处理所有这些的代码,因为我正在寻找的答案只是如何在 ConstructorInfo
上调用 Invoke。使用 MethodBase
的 Invoke
方法,所以我不想添加不必要的信息和太多代码供你们阅读。不过,我喜欢你使用 Func<,>
好多了,所以我转向那个。同时制作 Clone
方法 generic 是一个很好的补充,但在我的例子中,调用者不知道对象的类型,所以我将保留它为非泛型。
我不知道 Func<,>
,如果我知道我忘记的 lambda 运算符(我以前真的不需要这样的东西),所以我实际上从你的回答中学到了很多东西。我总是喜欢学习新事物,这在将来会派上用场,非常感谢! :)
最佳答案
如果您知道该对象具有这样的构造函数,您是否考虑过使用 Activator.CreateInstance
的重载?相反?
更新:所以您已经对 MethodInfo/MethodBase 进行了级联搜索并存储了它们 -> 您不想/不能使用 Activator
.
在那种情况下,我看不出有什么方法可以在没有类型转换的情况下做你想做的事。但是 - 也许您可以更改架构以存储 Dictionary<Type, Func<object, object>>
并添加那些 Func<>
实例代替。使调用代码更好(我假设)并允许您执行一次此转换:
// Constructor
dictionary.Add(type,
source => ((ConstructorInfo) method).Invoke(new object[] {source})
);
// Clone
dictionary.Add(type,
source => method.Invoke(source, new object[]{})
);
事实上,由于您只关心构造函数和普通方法在抓取它们的地方的区别,所以根本不需要强制转换,对吗?
// Constructor 2
dictionary.Add(type,
source => yourConstructorInfo.Invoke(new object[] {source})
);
除非我遗漏了什么(当然很有可能)这可以通过在围栏的定义侧执行一次来解决问题,调用者不需要介意这是否是构造函数?
最后一次,然后我将停止编辑垃圾邮件。我很无聊,想出了以下代码。这就是您要实现的目标吗?
public class Cloner {
private readonly IDictionary<Type, Func<object, object>> _cloneMap =
new Dictionary<Type, Func<object, object>>();
public T Clone<T>(T source) {
Type sourceType = source.GetType();
Func<object, object> cloneFunc;
if (_cloneMap.TryGetValue(sourceType, out cloneFunc)) {
return (T)cloneFunc(source);
}
if (TryGetCopyConstructorCloneFunc(sourceType, out cloneFunc)) {
_cloneMap.Add(sourceType, cloneFunc);
return (T)cloneFunc(source);
}
if (TryGetICloneableCloneFunc(sourceType, out cloneFunc)) {
_cloneMap.Add(sourceType, cloneFunc);
return (T)cloneFunc(source);
}
return default(T);
}
private bool TryGetCopyConstructorCloneFunc(Type type,
out Func<object, object> cloneFunc) {
var constructor = type.GetConstructor(new[] { type });
if (constructor == null) {
cloneFunc = source => null;
return false;
}
cloneFunc = source => constructor.Invoke(new[] { source });
return true;
}
private bool TryGetICloneableCloneFunc(Type type,
out Func<object, object> cloneFunc) {
bool isICloneable = typeof(ICloneable).IsAssignableFrom(type);
var cloneMethod = type.GetMethod("Clone", new Type[] { });
if (!isICloneable || (cloneMethod == null)) {
cloneFunc = source => null;
return false;
}
cloneFunc = source => cloneMethod.Invoke(source, new object[] {});
return true;
}
}
关于c# - 在构造函数上调用 MethodBase 的 Invoke(反射),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2543711/
谁能解释一下原因: (define a (lambda() (cons a #f))) (car (a)) ==> procedure ((car (a))) ==> (procedure . #f)
这是 PyBrain 网站的摘录。我了解大部分正在发生的事情,但是一行让我完全难住了。我以前从未在 python 代码中看到过这样的东西。这是整个循环,对于上下文: for c in [0,
我是gradle / groovy的新手。我想创建将做一些事情的自定义任务。我的第一个问题是任务完成时该如何做?我可以覆盖doFirst / doLast闭包吗?也许我可以重写某些在开始和结束时都会执
我刚刚开始评估 MS 企业库。他们使用以下指令来获取实例: var customerDb = EnterpriseLibraryContainer.Current.GetInstance("C
这是我的 if else Ansible 逻辑.. - name: Check certs exist stat: path=/etc/letsencrypt/live/{{ rootDomain
我正在使用construct 2.8 对一些失传已久的 Pascal 程序创建的一些文件的 header 进行逆向工程。 header 由许多不同的记录组成,其中一些是可选的,我不确定顺序是否固定。
我在将 getchar() 的输入放入 char *arr[] 数组时遇到问题。我这样做的原因是因为输入数据(将是一个带有命令行参数的文件)将存储在一个 char 指针数组中以传递给 execvp 函
通常我们不能约束类型参数 T派生自密封类型(例如 struct 类型)。这将毫无意义,因为只有一种类型适合,因此不需要泛型。所以约束如下: where T : string 或: where T :
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve th
#include using namespace std; class A { private: int m_i; friend int main(int argc, char cons
这个问题在这里已经有了答案: Are there legitimate uses for JavaScript's "with" statement? (33 个答案) 关闭 9 年前。 我有这个代
在this answer我看到了下一个 Bash 结构。 yes "$(< file.txt)" 什么意思 "$(< file.txt)" ? 我明白了 命令替换 - $(command)用命令的结
if (a == 1) //do something else if (a == 2) //do something else if (a == 3) //do somethi
关于构造的快速简单的问题。 我有以下用于将项目添加到 ListView 的代码。 ListViewItem item = new ListViewItem(); item.Text = file; i
我想使用 std::vector 来控制给定的内存。首先,我很确定这不是好的做法,但好奇心占了上风,无论如何我都想知道如何做到这一点。 我遇到的问题是这样的方法: vector getRow(unsi
下面显示了一段简单的javascript: var mystring = ("random","ignored","text","h") + ("ello world") 这个字符串会生成 hello
在 Java 中,创建对象的标准方法是使用 MyClass name = new MyClass(); 我也经常看到构造 new MyClass() { /*stuff goes in here*/
我正在编写 C++ ndarray 类。我需要动态大小和编译时大小已知的数组(分别分配自由存储和分配堆栈)。我想支持从嵌套的 std::initializer_list 进行初始化。 动态大小的没问题
我正在将一个项目从 Visual Studio 2005 转换为 Visual Studio 2008,并提出了上述结构。 using Castle.Core.Resource; using Cast
我想知道我在这里的想法是否正确,我主要针对接口(interface)进行编程,所以我想知道下面的类是否应该通过 DI 注入(inject),或者我应该自己实例化一个类... 注意:这些服务保存在我的核
我是一名优秀的程序员,十分优秀!