因为在 iPhone 上运行的 MonoTouch 程序是编译成静态代码的, 所以需要在运行时动态生成代码的功能是不可用的。
以下是与桌面版的 Mono 相比, MonoTouch 的限制:
1. 受限的泛型支持
与传统的 Mono/.Net 不同, 在 iPhone 上运行的代码不是由 JIT 编译器动态编译的, 而是是预编译成静态代码的。 Mono 的 Full AOT 技术在泛型方面有一些限制, 它们是:1.1 泛型虚函数
不支持泛型虚函数, 因为不能静态确定在不同情况下什么方法会被调用。 (这也是 C++ 不支持虚模板函数(virtual template method)的原因)class HasGenericVirtualMethod { public virtual void PrintValues
1.2 泛型类型中的平台调用
不支持在泛型类型中使用平台调用 (P/Invoke):class GenericType
1.3 不支持在 Nullable 类型上使用 Property.SetInfo
不支持在 Nullable1.4 字典的关键字为值类型
将值类型作为 Dictionary对于引用类型来说,这没有什么问题(因为反射+创建新类型的步骤会被跳过), 但是对于值类型来说, 一旦在移动设备商运行, 你的应用将可能会崩溃。 (原文是:This works for reference types (as the reflection+create a new type step is skipped), but for value types it crashes and burns rather quickly _disibledevent=>IEqualityComparer
2. 没有动态代码生成
由于 iPhone 的内核禁止应用程序动态生成代码, 在 iPhone 上 Mono 不支持任何形式的动态代码生成, 包括:- System.Reflection.Emit 不可用;
- 不支持 System.Runtime.Remoting;
- 不支持创建动态类型(例如 Type.GetType(“MyType`1)), 不过可以创建已知类型(例如: Type.GetType(“System.String”) 可以正常使用);
- 反向回调函数 (Reverse callback) 必须在编译时注册。
2.1 System.Reflection.Emit
缺少 System.Reflection.Emit 意味着任何依赖运行时的生成代码的代码不能工作, 它们包括:- 动态语言运行时 (Dynamic Language Runtime);
- 构建于动态语言运行时的任何语言;
- 远程调用中的透明代理或者其它任何需要在运行时动态生成代码的东西。
但是关于反射的API, 包括 Type.GetType(“SomeClass”)、 列举类型方法、属性, 获取特性(Attributes)和值是可以使用的。
2.2 反向回调
在标准的 Mono 中, 可以将 C# 的委托实例传递给非托管代码, 用于替换函数指针。 运行时通常会将函数指针转换, 允许非托管代码回调托管代码。这个转换的 Mono 中是通过 JIT 编译器实现的。 当使用 iPhone 需要的 AOT 编译器时, 有两个限制:
- 回调函数必须标记为 MonoPInvokeCallbackAttribute;
- 回调函数必须是静态的, 现在还不支持实例函数 (这个限制以后会移除)
3. 没有 Remoting
Remoting 在 MonoTouch 上不可用。4. 运行时禁用的特性
以下特性在 iPhone 的 Mono 运行时被禁用:- 分析器 Profiler;
- 发射 Reflection.Emit;
- 发射保存 Reflection.Emit.Save;
- COM 绑定;
- JIT 引擎;
- 元数据验证(因为没有 JIT)
最新评论