MSIL实用指南-加载bool、sbyte、byte、char、short等值
这一篇讲解怎么加载bool值、sbyte值、byte值、char值、short值。
加载bool值
在.NET程序实际运行中,是没有true和false值的,实际上是以1和0表示它们,加载它们的指令是Ldc_I4_1和Ldc_I4_0。
下面是实际例子:
/* 加载true,即加载1 */
ilGenerator.Emit(OpCodes.Ldc_I4_1);
/* 加载false,即加载0 */
ilGenerator.Emit(OpCodes.Ldc_I4_0);
加载sbyte值
sbyte是8位有符号整数类型,它的加载指令是Ldc_I4_S,格式是
ilGenerator.Emit(OpCodes.Ldc_I4_S, <sbyte值>);
例如:
ilGenerator.Emit(OpCodes.Ldc_I4_S,(sbyte)100);
加载byte值
byte是8位无符号整数类型,加载它实际就是加载整数,加载它用加载前面实现的加载int值的函数
LoadInt(ILGenerator il, int value)就可以。
例如:
LoadInt(ilGenerator, (int)(100));
加载char值
char实际上是16位无符号整数类型,加载它也就是加载整数,加载它用加载前面实现的加载int值的函数
LoadInt(ILGenerator il, int value)就可以。
例如:
LoadInt(ilGenerator, (int)('A'));
加载short值
short是16位有符号整数类型,加载它和加载整数一样,与加载char差不多,加载它用加载前面实现的加载int值的函数
LoadInt(ILGenerator il, int value)就可以。
例如:
LoadInt(ilGenerator, (short)(-100));
完整的程序:
using System;
using System.Reflection;
using System.Reflection.Emit; namespace LX1_ILDemo
{
/// <summary>
/// load char、sbyte、char值
/// </summary>
class Demo03_LoadBSBC
{
static string binaryName = "Demo03_LoadBSBC.exe";
static string namespaceName = "LX1_ILDemo";
static string typeName = "LoadBSBC"; static AssemblyBuilder assemblyBuilder;
static ModuleBuilder moduleBuilder;
static TypeBuilder typeBuilder;
static MethodBuilder mainMethod;
static ILGenerator ilGenerator; static void Emit_Ldc()
{
/* 加载bool值 */
MethodInfo writeBoolLineMethod = typeof(Console).GetMethod("WriteLine", new Type[] { typeof(bool) }); /* 加载true,即加载1 */
ilGenerator.Emit(OpCodes.Ldc_I4_1);
ilGenerator.Emit(OpCodes.Call, writeBoolLineMethod); /* 加载false,即加载0 */
ilGenerator.Emit(OpCodes.Ldc_I4_0);
ilGenerator.Emit(OpCodes.Call, writeBoolLineMethod); /* 加载sbyte值,用Ldc_I4_S */
MethodInfo writeIntLineMethod = typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) });
ilGenerator.Emit(OpCodes.Ldc_I4_S,(sbyte));
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); ilGenerator.Emit(OpCodes.Ldc_I4_S, (sbyte)(-));
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); ilGenerator.Emit(OpCodes.Ldc_I4_S, sbyte.MaxValue);
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); ilGenerator.Emit(OpCodes.Ldc_I4_S, sbyte.MinValue);
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); /* 加载byte值,和加载整数一样 */
LoadInt(ilGenerator, (int)());
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); LoadInt(ilGenerator, (int)(byte.MaxValue));
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); LoadInt(ilGenerator, (int)(byte.MinValue));
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); /* 加载char值,即加载16位无符号整数 */
MethodInfo writeCharLineMethod = typeof(Console).GetMethod("WriteLine", new Type[] { typeof(char) });
LoadInt(ilGenerator, (int)('A'));
ilGenerator.Emit(OpCodes.Call, writeCharLineMethod); LoadInt(ilGenerator, (int)('C'));
ilGenerator.Emit(OpCodes.Call, writeCharLineMethod); LoadInt(ilGenerator, (int)(char.MaxValue));
ilGenerator.Emit(OpCodes.Call, writeCharLineMethod); LoadInt(ilGenerator, (int)(char.MinValue));
ilGenerator.Emit(OpCodes.Call, writeCharLineMethod); /* 加载short值,和加载整数一样 */
LoadInt(ilGenerator, (short)(-));
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); LoadInt(ilGenerator, (short)());
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); LoadInt(ilGenerator, (int)(short.MaxValue));
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod); LoadInt(ilGenerator, (int)(short.MinValue));
ilGenerator.Emit(OpCodes.Call, writeIntLineMethod);
} public static void LoadInt(ILGenerator il, int value)
{
switch (value)
{
case -:
il.Emit(OpCodes.Ldc_I4_M1);
return;
case :
il.Emit(OpCodes.Ldc_I4_0);
return;
case :
il.Emit(OpCodes.Ldc_I4_1);
return;
case :
il.Emit(OpCodes.Ldc_I4_2);
return;
case :
il.Emit(OpCodes.Ldc_I4_3);
return;
case :
il.Emit(OpCodes.Ldc_I4_4);
return;
case :
il.Emit(OpCodes.Ldc_I4_5);
return;
case :
il.Emit(OpCodes.Ldc_I4_6);
return;
case :
il.Emit(OpCodes.Ldc_I4_7);
return;
case :
il.Emit(OpCodes.Ldc_I4_8);
return;
} if (value > - && value < )
{
il.Emit(OpCodes.Ldc_I4_S, (SByte)value);
}
else
{
il.Emit(OpCodes.Ldc_I4, value);
}
} public static void Generate()
{
InitAssembly(); /* 生成 public class LoadLFDSN */
typeBuilder = moduleBuilder.DefineType( namespaceName+"."+ typeName, TypeAttributes.Public); /* 生成 public static void Main() */
GenerateMain(); Emit_Ldc(); EmitReadKey();
ilGenerator.Emit(OpCodes.Ret); /* 设置assembly入口方法 */
assemblyBuilder.SetEntryPoint(mainMethod, PEFileKinds.ConsoleApplication); SaveAssembly();
Console.WriteLine("生成成功");
} static void EmitReadKey()
{
/* 生成 Console.ReadKey(); */
MethodInfo readKeyMethod = typeof(Console).GetMethod("ReadKey", new Type[] { });
ilGenerator.Emit(OpCodes.Call, readKeyMethod);
ilGenerator.Emit(OpCodes.Pop);
} static void GenerateMain()
{
mainMethod = typeBuilder.DefineMethod("Main", MethodAttributes.Public
| MethodAttributes.Static, typeof(void), new Type[] { });
ilGenerator = mainMethod.GetILGenerator();
} static void InitAssembly()
{
AssemblyName assemblyName = new AssemblyName(namespaceName);
assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
moduleBuilder = assemblyBuilder.DefineDynamicModule(assemblyName.Name, binaryName);
} static void SaveAssembly()
{
Type t = typeBuilder.CreateType(); //完成Type,这是必须的
assemblyBuilder.Save(binaryName);
}
}
}
MSIL实用指南-加载bool、sbyte、byte、char、short等值的更多相关文章
- MSIL实用指南-加载int值
这一篇讲的是怎样加载整数值到运算栈上.这一类的指令都是以Ldc_I4开头. Ldc_I4类OpCodes的Ldc_I4字段的功能是把一个int值压入运算栈上.它的使用方法是ilGenerator.Em ...
- MSIL实用指南-加载null、string、long、float、double等值
本篇讲述怎么加载null.string值.long值.float值.double值. 加载null不需要参数值,只要 Emit ldnull 其它几个命令要 Emit <指令> <值 ...
- MSIL实用指南-加载和保存参数
本篇讲解怎么加载和保存参数,以及参数起始序号的确定. 参数的加载加载参数的指令是Ldarg.Ldarg_S.Ldarg_0.Ldarg_1.Ldarg_2.Ldarg_3.Ldarg_0是加载第0个参 ...
- MSIL实用指南-Action的生成和调用
MSIL实用指南-Action的生成和调用 System.Action用于封装一个没有参数没有返回值的方法.这里生成需要Ldftn指令. 下面讲解怎生成如下的程序. class ActionTest ...
- MSIL实用指南-生成索引器
MSIL实用指南-生成索引器 索引器是一种特殊的属性,它有参数的,也有get和set方法,属性名称一般是"Item",并且方法名称一般名称是"get_Item" ...
- MSIL实用指南-字段的加载和保存
字段有静态字段和非静态字段之分,它们的加载保存指令也是不一样的,并且非静态字段要生成this. 静态字段的加载加载静态字段的指令是Ldsfld.ilGenerator.Emit(OpCodes.Lds ...
- MSIL实用指南-局部变量的声明、保存和加载
这一篇讲解方法内的局部变量是怎么声明.怎样保存.怎样加载的. 声明局部变量声明用ILGenerator的DeclareLocal方法,参数是局部变量的数据类型,得到一个局部变量对应的创建类LocalB ...
- MSIL实用指南-一维数组的操作
本篇讲解怎么生成和操作一维数组.各种数组类型创建的步骤是一样的,但是加载和保存步骤有所不同. 一.创建数组所有类型的一维数组创建都是一样的,分三步.1.加载数组长度2.生成指令 Newarr < ...
- MSIL实用指南-比较运算
数值的比较就是大于.小于.等于.大于等于.小于等于.不等于,它们的运算结果都是布尔值.大于.小于.等于有直接对应的指令,分别是Cgt.Clt.Ceq.大于等于.小于等于.不等于没有直接对应的指令,它的 ...
随机推荐
- PowerDesigner Constraint name uniqueness 问题处理(转载)
使用PowerDesigner生成数据库脚本时报 Constraint name uniqueness 错误: 双击每行错误,发现外键引用的名字有重复的: 惯性去网上找解决办法,找到的主要是两个方法: ...
- 菜鸟之旅——.NET垃圾回收机制
.NET的垃圾回收机制是一个非常强大的功能,尽管我们很少主动使用,但它一直在默默的在后台运行,我们仍需要意识到它的存在,了解它,做出更高效的.NET应用程序:下面我分享一下我对于垃圾回收机制(GC)的 ...
- spring core 与 context理解
Spring core是核心层,拥有这BeanFactory这个强大的工厂,是所有bean的管理器: 而spring context是上下文运行环境,基于spring core之上的一个架构, 它之上 ...
- C#语言Devdevexpress控件chart在C/S框架中的使用
声明.数据库连接机制框架已经写好.框架模式是MVC模式.就以我的from测试.我的做法是在查询页面创建一个按钮,然后在这个按钮上绑定一个点击事件.点击查询按钮把查询条件传到需要显示图例的页面.查询页面 ...
- tcp/ip 卷一 读书笔记(5)arp和rarp 同网段和不同网段之间的通信过程
arp和rarp 同网段和不同网段之间的通信过程 IPv6中已经没有arp rarp协议,所以这里都是IPv4. 链路层使用以太网地址来确定目的地址,应用则常使用ip地址通信 arp协议是指从ip地址 ...
- HashMap/HashSet,hashCode,哈希表
hash code.equals和“==”三者的关系 1) 对象相等则hashCode一定相等: 2) hashCode相等对象未必相等. == 是比较地址是否相等,JAVA中声明变量都是引用嘛,不同 ...
- jmeter 脚本规范
总结了一下公司正在用 jmeter 脚本规范. 使用 jmeter 进行接口级测试, 随着接口增多以及业务逻辑越来越复杂, 导致 jmeter 脚本的维护会更加困难.针对实际使用中发现的问题进行一些规 ...
- linux权限归属及特殊权限设置
访问权限:读取:允许查看内容 -read写入:允许修改内容 -write可执行:允许运行和切换 -excute(以上三点rwx共同决定最终权限)归属关系:所有者:拥有此文件/目录的用户 -user所属 ...
- alsa声卡分析alsa-utils调用过程
如何分析tinyplay 播放音频和tinymix的过程?需要相应的工具来支持追查: 一.利用strace工具分析tinyplay和tinymix: strace -o tinyplay.log ti ...
- ARM开发软件ADS教程
ARM开发软件ADS教程 ADS(ARM Developer Suite)是ARM公司推出ARM集成开发环境,操作简单方便,获得广大开发人员的青睐.下面使用ADS v1.2做一个实例教程,帮助大家学会 ...