- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我是 reflection.emit 的新手,一直在尝试生成以下 C# 代码:
public class RepositoryWrapper
{
public void CallRepositoryMethod(IAddressRepository repository, Address address)
{
repository.NODE_I_NodeExtendedDetails_Address3(address.NodeId);
}
}
这是它的 il 表示:
IL_0000: nop
IL_0001: ldarg.1
IL_0002: ldarg.2
IL_0003: callvirt instance int32 ReflectionServices.Node::get_NodeId()
IL_0008: callvirt instance void ReflectionServices.IAddressRepository::NODE_I_NodeExtendedDetails_Address3(int32)
IL_000d: nop
IL_000e: ret
这是我用来创建它的代码:
internal static void Generate(this System.Reflection.Emit.ILGenerator @this, Type target,string method,Type instance)
{
var methodToCall = target.GetMethod(method);
var methodParams = methodToCall.GetParameters();
var instanceProperties = instance.GetProperties(BindingFlags.Public | BindingFlags.Instance);
var orderedProperties = (from mp in methodParams
join p in instanceProperties
on mp.Name.ToLower() equals p.Name.ToLower()
select p).ToArray();
//add properties to the string builder
//load the object reference onto the stack sothat we can access its methods
@this.Emit(OpCodes.Nop);
@this.Emit(OpCodes.Ldarg_1);
@this.Emit(OpCodes.Ldarg_2);
var property = orderedProperties.FirstOrDefault(x => x.Name == "NodeId");
if (property != null)
{
var getMethod = property.GetGetMethod();
@this.Emit(getMethod.IsVirtual ? OpCodes.Callvirt : OpCodes.Call, getMethod);
}
//call method
@this.Emit(methodToCall.IsVirtual ? OpCodes.Callvirt : OpCodes.Call, methodToCall);
@this.Emit(OpCodes.Nop);
//return from function
@this.Emit(OpCodes.Ret);
}
这是我遇到的错误:
System.InvalidProgramException: Common Language Runtime detected an invalid program.
Result StackTrace:
at ReflectionServices.Repository.NODE_I_NodeExtendedDetails3_Address40807399(Repository target, Address )
这是生成的 il:
nop
ldarg.1
ldarg.2
call instance int32 ReflectionServices.Node::get_NodeId()
callvirt instance void
ReflectionServices.Repository::
NODE_I_NodeExtendedDetails3_Address(int32)
nop
返回
谁能看出我卡住了是什么问题?
谢谢
这是我要求的dll和方法:
public sealed class ReflectionEmitWithDebuggingMethodGenerator
{
private AssemblyBuilder Assembly { get; set; }
private ModuleBuilder Module { get; set; }
private AssemblyName Name { get; set; }
public ReflectionEmitWithDebuggingMethodGenerator()
: base()
{
this.Name = new AssemblyName() { Name = Guid.NewGuid().ToString("N") };
this.Assembly = AppDomain.CurrentDomain.DefineDynamicAssembly(
this.Name, AssemblyBuilderAccess.RunAndSave,@"C:\Users\darren\Documents\Visual Studio 2012\Projects\UnityInjection");
this.AddDebuggingAttribute(this.Assembly);
this.Module = this.Assembly.DefineDynamicModule(this.Name.Name + ".dll", true);
}
public Action<TObject, TInstance> Generate<TObject, TInstance>(Type target, string methodN, Type instanceType)
{
var type = this.Module.DefineType(target.Namespace + "." + target.Name);
var methodName = methodN + target.GetHashCode().ToString();
var method = type.DefineMethod(methodName, MethodAttributes.Static | MethodAttributes.Public, typeof(void), new Type[] { target, instanceType });
method.DefineParameter(1, ParameterAttributes.In, "target");
method.DefineParameter(2, ParameterAttributes.In, "instance");
ILGenerator.Generate(method.GetILGenerator(), target,methodN,instanceType);
var createdType = type.CreateType();
var createdMethod = createdType.GetMethod(methodName);
return (Action<TObject, TInstance>)Delegate.CreateDelegate(typeof(Action<TObject, TInstance>), createdMethod);
}
}
最佳答案
比较编译和发出的输出,只有一个区别:
编译:
callvirt instance int32 ReflectionServices.Node::get_NodeId()
发出:
call instance int32 ReflectionServices.Node::get_NodeId()
您调用 int32 get_NodeId()
的类型是 ReflectionServices.Node
,但是您传递给您尝试的方法的对象类型要复制的是 Address
。这使我相信必须以虚拟方式调用定义在 ReflectionServices.Node
上的属性访问器,可能是因为它继承自另一个在 ReflectionServices.Node 之前声明该属性的类(或实现接口(interface))
实现它。
当您发出该行代码时,只需虚拟地调用它:
@this.Emit(OpCodes.Callvirt, getMethod);
编辑:根据进一步提供的代码,这是真正的解决方案。
因此,您在实现接口(interface)的基础知识方面遇到了问题:
var method = type.DefineMethod(methodName, MethodAttributes.Static | Method...
// ^^^^^^
接口(interface)方法不是静态的;他们是实例成员。因此,您首先需要在创建 MethodBuilder
时从属性标志中删除 MethodAttributes.Static
。
其次,当您返回此函数时,您将必须包含一个目标对象,它是调用该方法的实例。为此,您可以使用 Activator.CreateInstance
调用默认生成的构造函数并为您提供一个实例化对象以用作目标。用这些行替换 Generate
方法的最后一行以实现这一点。
var activatedObject = Activator.CreateInstance(type);
return (Action<TObject, TInstance>)Delegate.CreateDelegate(
typeof(Action<TObject, TInstance>), activatedObject, createdMethod);
关于c# - 反射.emit System.InvalidProgramException : Common Language Runtime detected an invalid program,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16950272/
我在内存中有一个 Map 来存储我的对象。我想在内存不足时刷新内存。我现在正在这样做: void add(K key, V value) { if (underPressure()) {
我是 JBPM 的新手。我下载并开始使用 eclipse。然后我安装了 JBPM 插件和 Modeler 2.0 插件。当我创建一个新的 JBPM 项目时,它要求 JBPM 运行时。我在互联网上搜索了
我是 JBPM 的新手。我下载并开始使用 eclipse。然后我安装了 JBPM 插件和 Modeler 2.0 插件。当我创建一个新的 JBPM 项目时,它要求 JBPM 运行时。我在互联网上搜索了
我目前正在将Windows 8.1应用程序移植到Windows 10 UAP应用程序,该应用程序使用互操作在C++和C#之间进行通信,并且 pop 以下错误: $(ProjectName).Inter
我正在使用 Parcel 来捆绑我的项目并 Jest 运行我的测试。 一切正常,但在测试中我有 async/await关键字,我必须导入 regenerator-runtime/runtime像这样:
一旦我尝试使用 Promises,Browserify 就会抛出此错误。 在 中找不到模块“regenerator-runtime/runtime” 我尝试过 Node 版本 10.15.3 和 12
我目前正在试验 quarkus,找不到在运行时更改一些 openapi 信息的方法(这里,我希望授权 url 根据环境而改变)。 应该可以通过使用 OASFilter 并从环境变量中提供信息来实现,但
我正在使用 rust_bert 汇总文本。我需要使用rust_bert::pipelines::summarization::SummarizationModel::new设置模型,该模型可以从互联网
我正在 Visual Studio 中处理其中一个新的“类库(NuGet 包)”项目。一切都很顺利,直到前几天它开始引发有关 System.Runtime.Extensions 程序集的错误: Ass
我正在使用 tokio::runtime::current_thread::Runtime我希望能够在同一个线程中运行 future 并停止 react 堆。页面上的示例未显示如何停止运行时。有什么办
我找不到如何声明 runtime.LockOSThread()和 runtime.UnlockOSThread() .我将其定义为 [runtime.LockOSThread()] 一些代码 [run
我有这样的代码, Routine 1 { runtime.LockOSThread() print something send int to routine 2 runtime.UnlockOSTh
我正在尝试执行此操作:当我单击“搜索”按钮时,命令提示符窗口打开。 我尝试使用ProcessBuilder,没有出现错误,但是不起作用。你能帮我一下吗? package sys.tool; impor
我的应用程序运行大约 4 小时后出现此错误。我不知道为什么会这样。该错误发生在 windows 7 和 windows XP 系统上。抱歉,我没有更多信息。我的应用程序运行大约 4 次后才收到此错误消
我收到错误:“此程序集是由比当前加载的运行时更新的运行时构建的,无法加载。” 我有一个 .NET 2.0 项目调用的 .NET 4.0 dll 项目。有没有办法调和框架的差异? 最佳答案 I have
我有一个 WIX 安装程序,它在安装过程中执行自定义操作。当我运行 WIX 安装程序并遇到它的第一个自定义操作时,安装程序失败,我在 MSI 日志中收到如下错误: Action start 12:
我正在尝试提交 spark 作业 它是这样开始的: import javax.xml.parsers.{SAXParser, SAXParserFactory} import org.apache.s
我正在做一些 WCF 工作,但我不明白为什么会遇到这个错误。 System.Runtime.Serialization.DataContractAttribute命名空间中不存在 System.Run
我尝试按照 this 在 React 应用程序中使用 async/await : 目前,我正在运行webpack --config webpack.dev.config.js --watch --pr
我已经从托管的 Window Server 2008 下载了已发布的(代码隐藏文件不存在,与 bin 文件夹中的 dll 结合)Web 应用程序,并在调试该应用程序时使用 Visual Studio
我是一名优秀的程序员,十分优秀!