首先,在这里使用了大量的Emit和反射,因为不喜欢直接写IL汇编,所以看起来有点累,但是IL的知识是必须的,否则很难看明白这些代码是在干什么。
其次,要理解为什么会这样,需要了解一些程序底层的知识。
最后,准确的说,这里的对象并不是一个真实的对象,仅仅是把某段内存认为是某个对象,一般的高级语言不允许这样的代码,所以只能用最基础的IL来做。
1
static void Foo()
2
{
3
AssemblyName name = new AssemblyName("a");
4
AssemblyBuilder a = AppDomain.CurrentDomain.DefineDynamicAssembly(name, AssemblyBuilderAccess.Run);
5
ModuleBuilder m = a.DefineDynamicModule("test", false);
6
TypeBuilder tb = Bar(m);
7
Type t = tb.CreateType();
8
MethodInfo mi = t.GetMethod("test", BindingFlags.Public | BindingFlags.Static);
9
mi.Invoke(null, null);
10
}
11data:image/s3,"s3://crabby-images/6cac0/6cac0dd46b2c91794627282c0d73fed6efa5ea05" alt=""
12
static TypeBuilder Bar(ModuleBuilder m)
13
{
14
// 创建一个Foo类
15
TypeBuilder result = m.DefineType("Foo");
16
// 定义一个bar字段
17
FieldBuilder field = result.DefineField("bar", typeof(int), FieldAttributes.Assembly);
18
// Foo的实例构造器
19
ConstructorBuilder ctor = result.DefineConstructor(MethodAttributes.Public, CallingConventions.Standard, Type.EmptyTypes);
20
ILGenerator gen = ctor.GetILGenerator();
21
gen.Emit(OpCodes.Ldarg_0);
22
gen.Emit(OpCodes.Call, typeof(object).GetConstructor(Type.EmptyTypes));
23
gen.Emit(OpCodes.Ldarg_0);
24
gen.Emit(OpCodes.Ldc_I4_0);
25
gen.Emit(OpCodes.Stfld, field);
26
gen.Emit(OpCodes.Ret);
27data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
28
// 定义test方法
29
MethodBuilder method = result.DefineMethod("test",
30
MethodAttributes.Public | MethodAttributes.Static,
31
CallingConventions.Standard, typeof(void), Type.EmptyTypes);
32data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
33
gen = method.GetILGenerator();
34
// 定义本地变量1、2、3
35
LocalBuilder local1 = gen.DeclareLocal(typeof(int));
36
LocalBuilder local2 = gen.DeclareLocal(typeof(int));
37
LocalBuilder local3 = gen.DeclareLocal(typeof(int));
38data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
39
// 用local2的地址作为对象地址,调用实例构造器
40
gen.Emit(OpCodes.Ldloca, local2);
41
gen.Emit(OpCodes.Call, ctor);
42
// 用local2的地址作为对象地址,把数字123放到字段里面
43
gen.Emit(OpCodes.Ldloca, local2);
44
gen.Emit(OpCodes.Ldc_I4, 123);
45
gen.Emit(OpCodes.Stfld, field);
46data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
47
// 来看看local1里面是什么
48
gen.Emit(OpCodes.Ldstr, "local1 address=");
49
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
50
gen.Emit(OpCodes.Ldloca, local1);
51
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
52
gen.Emit(OpCodes.Ldstr, "local1=");
53
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
54
gen.Emit(OpCodes.Ldloc, local1);
55
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
56
// 来看看local2里面是什么
57
gen.Emit(OpCodes.Ldstr, "local2 address=");
58
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
59
gen.Emit(OpCodes.Ldloca, local2);
60
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
61
gen.Emit(OpCodes.Ldstr, "local2=");
62
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
63
gen.Emit(OpCodes.Ldloc, local2);
64
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
65
// 来看看local3里面是什么
66
gen.Emit(OpCodes.Ldstr, "local3 address=");
67
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
68
gen.Emit(OpCodes.Ldloca, local3);
69
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
70
gen.Emit(OpCodes.Ldstr, "local3=");
71
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
72
gen.Emit(OpCodes.Ldloc, local3);
73
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
74data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
75
// 用local2的地址作为对象地址,看看字段里面是什么
76
gen.Emit(OpCodes.Ldstr, "Field address=");
77
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
78
gen.Emit(OpCodes.Ldloca, local2);
79
gen.Emit(OpCodes.Ldflda, field);
80
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
81
gen.Emit(OpCodes.Ldstr, "Field =");
82
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("Write", new Type[] { typeof(string) }));
83
gen.Emit(OpCodes.Ldloca, local2);
84
gen.Emit(OpCodes.Ldfld, field);
85
gen.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(int) }));
86data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
87
gen.Emit(OpCodes.Ret);
88data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
89
return result;
90
}
91data:image/s3,"s3://crabby-images/6cac0/6cac0dd46b2c91794627282c0d73fed6efa5ea05" alt=""
data:image/s3,"s3://crabby-images/6cac0/6cac0dd46b2c91794627282c0d73fed6efa5ea05" alt=""
2
data:image/s3,"s3://crabby-images/fadc8/fadc87475c0a604562c53d1ce0aa3cd58aa8e33a" alt=""
3
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
4
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
5
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
6
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
7
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
8
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
9
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
10
data:image/s3,"s3://crabby-images/236c7/236c7110a3d28630d6db8780cfd27f09834f6fe9" alt=""
11
data:image/s3,"s3://crabby-images/6cac0/6cac0dd46b2c91794627282c0d73fed6efa5ea05" alt=""
12
data:image/s3,"s3://crabby-images/6cac0/6cac0dd46b2c91794627282c0d73fed6efa5ea05" alt=""
13
data:image/s3,"s3://crabby-images/fadc8/fadc87475c0a604562c53d1ce0aa3cd58aa8e33a" alt=""
14
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
15
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
16
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
17
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
18
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
19
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
20
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
21
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
22
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
23
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
24
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
25
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
26
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
27
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
28
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
29
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
30
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
31
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
32
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
33
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
34
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
35
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
36
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
37
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
38
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
39
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
40
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
41
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
42
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
43
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
44
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
45
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
46
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
47
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
48
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
49
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
50
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
51
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
52
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
53
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
54
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
55
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
56
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
57
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
58
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
59
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
60
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
61
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
62
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
63
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
64
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
65
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
66
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
67
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
68
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
69
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
70
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
71
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
72
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
73
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
74
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
75
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
76
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
77
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
78
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
79
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
80
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
81
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
82
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
83
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
84
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
85
data:image/s3,"s3://crabby-images/660db/660dbadd6c9ef03df5698a1947200796a88b70bb" alt=""
86
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
87
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
88
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
89
data:image/s3,"s3://crabby-images/f808d/f808dab32a53c4f032c00727b665ec78271457b6" alt=""
90
data:image/s3,"s3://crabby-images/236c7/236c7110a3d28630d6db8780cfd27f09834f6fe9" alt=""
91
data:image/s3,"s3://crabby-images/6cac0/6cac0dd46b2c91794627282c0d73fed6efa5ea05" alt=""
乍一看,代码有些长,下面仔细分析一下:
1、Foo方法,作用是为Emit做准备,并且测试,调用"test"方法
2、Bar方法,Emit一个叫"Foo"的引用类型的类,带有一个"Bar"的字段,并且有一个实例构造器和一个"test"方法
3、在实例构造器中,对"Bar"字段赋初始值0
4、在"test"方法中,用local2的地址作为对象的地址,Call实例构造器(注意不能用newobj),这样的效果等于初始化对象的初始值(也就是说,Call实例构造器后,"Bar"字段的值就是构造器中赋的初始值)
5、再次用local2的地址作为对象的地址,给"Bar"字段赋数字123(注意,只有这里出现了123这个数字)
6、从控制台分别输出local1、local2、local3,以及用local2的地址作为对象的地址的"Bar"字段的内存地址和他们的值
代码的作用就是这些,看一下运行的结果:
local1 address=1240840
local1=123
local2 address=1240836
local2=0
local3 address=1240832
local3=0
Field address=1240840
Field =123
local1=123
local2 address=1240836
local2=0
local3 address=1240832
local3=0
Field address=1240840
Field =123
运行结果中,可以意外的发现,local1也是123,而且local1的地址和"Bar"字段的地址一致。
当然这个值并不固定,而且local1、local2、local3的地址并不一定是递减的,也有可能是递增的,或者其它情况(虽然还没遇到,但是也不排除有分散的可能性)。
在递减的情况下,local1与以local2的地址作为对象地址的"Bar"字段的地址是一致的,也就是说,以local2的地址作为对象地址的情况下,"Bar"字段并不是在堆上,而是在栈上。对local1的任何改动可以影响"Bar"字段,反过来,对"Bar"字段的任何改动也可以影响local1,因为本质上它们就是同一个地址。
同样,以local2的地址作为对象地址的这个虚假引用类型的"对象",也是存在于栈上的,是从local2地址开始的,一片隶属于栈的内存。
最后,需要注意的一点是,这个方法有种钻运行时漏洞的感觉,在x86 .net 2.0的环境上测试通过,不保证其它运行时也能正常运行。