本文不讨论创建可执行的exe程序,主要想说明怎么在silverlight程序里面调用由原生代码所编写的DLL(C++ / ARM).
原生代码可以调用更多的API,但是这并不是说你就能随意获得那些你没有权限的资源,比如,你可以使用CopyFile这个API,但是如果你试图把文件Copy到\Windows文件夹,就会得到一个0x4ec的错误代码,系统会禁止你这样做.所以,你的程序也只能在沙箱的环境下运行.
本文中所涉及的知识包含 C++,COM交互,Windows phone 程序设计.希望你在开发你的程序前能保证熟悉这些技术.因为原生代码还不能调试,所以你只能使用返回错误信息的方式来确保你的程序能正确运行.
需要注意的是: 如果你有些任务需要执行很长时间,它们在调试的时候能很好的运行,但是在实际运行的时候,你最好采用一个线程来做这些事情.因为在非调试状态下检测程序会检测你的程序,一但你的程序锁定超过10秒,那么系统会自动退出这个程序.
有人建议原生代码所写的DLL需要签名,其实这并不是必须的.在Mango设备里面可以使用未签名的库.
讨论一下互操作锁.详细讨论可以参见这个贴子.互操作锁在WP7.5里面出现.最直观的表现就是你的程序如果使用了ID_CAP_INTEROPSERVICES,那么所使用的设备必须得解锁.
下面就是一个详细的操作步骤:
所需要的软件请点击名称下载:
1. 安装 Visual Studio 2008 及 最新的补丁包,确保安装 C++.
2. 安装 Windows Modile 6 Professional SDK Resfresh.
3. 安装 Visual Studio 2010 和 最新的补丁包.
4. 安装 Windows Phone SDK 7.1
5. 下载Microsoft.Phone.InteropServices.zip. 下载解压后要确定文件是非锁定状态,解锁可以按以下操作,文件是点击右键,选择属性,点击解锁.
6. 把Microsoft.Phone.InteropServices.dll放到 C:\Program Files\Reference Assemblies\Microsoft\Framework\Silverlight\v4.0\Profile\WindowsPhone71 ,如果是64位系统就放到 C:\Program Files (x86)\Reference Assemblies\Microsoft\Framework\Silverlight\v4.0\Profile\WindowsPhone71.
7. 打开 Visual Studio 2010 的命令行工具,转到Microsoft.Phone.InteropServices.dll放置的地方,输入以下代码,请确定注册成功.否则引用这个DLL时会报没有引用命名.
SN -Vr Microsoft.Phone.InteropServices.dll
8. 在DLL的目录下有一个RedistList文件夹,里面有一FrameworkList.xml,加下面代码
<File AssemblyName="Microsoft.Phone.InteropServices" Version="7.0.0.0" Culture="neutral" ProcessorArchitecture="MSIL" InGac="false" />
9. 安装 zune
10. 打开 VS2008,创建新项目.
11. 选择 Visual C++ / Smart Device / ATL Smart Device 项目,不能选择MFC.
12. 点击下一步
13. 取消 Pocket PC 2003,加入 Windows Mobile 6 Pro SDK,点击下一步
14. 点击完成
15. 编译设置为 Release.
16. 在工程属性 / 属性配置 / C/C++ / 预处理 / 预处理定义里面加入
_CE_ALLOW_SINGLE_THREADED_OBJECTS_IN_MTA |
17. 在工程里面加入一个新类 , 选择 "Simple ATL object"
18. 对话框内 "Short name"是 Com类的名字,其他可以保持默认,点击完成.
19. 文件添加就完成了.包含这些文件 : 头文件(.h) ,代码文件 (.cpp) , Com定义文件(.idl), Com类的头文件(.h),Com类的代码文件(.cpp).
20. 把Com类的基类改成 IUnknown.
21. 在Com类的头文件中,删除下面这一句
COM_INTERFACE_ENTRY(IDispatch) |
22. 在IDL文件中,把IDispatch改成IUnknown.
23. 现在可以向你的Com类加入方法了.所有方法都必须以HRESULT为返回值.这个值用来判断函数是否执行成成功.成功可以返回 S_OK,如果有错误代码,则把错误代码与0x80070000进行逻辑或的结果做为返回值.如果你想返回一个变量,你需要在IDL文件里面声明他.参数以COM为边界,参见这里和这里查看COM支持的参数类型.
24. 在COM类中加入如下代码:
STDMETHODIMP CNative::TestMethod1() { BOOL result = ::CopyFile(L"\\Windows\\0000_System.Windows.xaml", L"\\Windows\\Test.xaml", TRUE); //这里会抛出一个异常 if (result) return S_OK; else return 0x80070000 | ::GetLastError(); } STDMETHODIMP CNative::TestMethod2(BSTR InputString, BSTR* OutputString) { size_t size = 1000; // in chars TCHAR* msg = new TCHAR[size]; wcscpy_s(msg, size, L"\0"); LPWSTR value = new WCHAR[20]; _itow((int)wcslen(InputString), value, 10); wcscat_s(msg, size, L"Length of string is: "); wcscat_s(msg, size, value); *OutputString = SysAllocString(msg); delete[] msg; delete[] value; return S_OK; }
25. Com类的头文件中加入下面代码,放在END_COM_MAP()后面
1 STDMETHOD(TestMethod1)(); 2 STDMETHOD(TestMethod2)(BSTR InputString, BSTR* OutputString);
26. 在IDL文件里面加如如下代码,关于参数定义,可以查看 点击我吧
1 HRESULT TestMethod1(); 2 HRESULT TestMethod2(BSTR InputString, BSTR* OutputString);
27. 记下IDL文件里面的接口GUID(uuid标识),类标识GUID.
28. 在VS2010里面创建一个新的WP工程.
29. VS2008里面编译生成DLL,然后把DLL拷贝到WP工程目录下.
30. 在WP工程下创建WPInteropManifest.xml文件,内容为
1 <?xml version="1.0" encoding="UTF-8"?>2 <Interop>3 </Interop>
31. 更改WPInteropManifest.xml文件的编译规则为"Content","Copy if newer".
32. 更改COM输出的DLL编译规则为"Content","Copy if newer"
33. WP工程添加引用"Microsoft.Phone.InteropServices"
34. 打开WMAppManifest.xml文件,添加
<Capability Name="ID_CAP_INTEROPSERVICES" />
35. 添加一个代码文件 输入以下内容:
using System.Runtime.InteropServices; [ComImport, ClassInterface(ClassInterfaceType.None), Guid("YOUR-COCLASS-GUID-GOES-HERE")] public class CNative { } [ComImport, Guid("YOUR-INTERFACE-GUID-GOES-HERE"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] public interface INative { void TestMethod1(); [return : MarshalAs(UnmanagedType.BStr)] string TestMethod2([MarshalAs(UnmanagedType.BStr)] string InputString); }
36.创建一个测试代码文件,输入以入内容
uint retval = Microsoft.Phone.InteropServices.ComBridge.RegisterComDll("Your Com DLL.dll", new Guid("YOUR-COCLASS-GUID-GOES-HERE")); INative MyNativeCodeInstance = (INative)new CNative(); string result1 = "OK"; try { MyNativeCodeInstance.TestMethod1(); //这里抛出一个异常 } catch (Exception ex) { result1 = ex.Message; } string result2 = MyNativeCodeInstance.TestMethod2("Hello, Mango!"); MessageBox.Show(result1 + Environment.NewLine + result2);
37. 运行程序,测试代码.
38. 注意,当使用高级功能时.我们需要Marshal-class,比如操作内存等.此时需要使用Microsoft.Phone.InteropServices内的Marshal类,如果使用System.Runtime.InteropServices命名空间下的此类,会抛出一个MethodAccessException异常.
关于简单的调用方法就说到这里.希望大家都能搞出更好的自制程序.如果翻译或者描述有不准确的地方,希望大家指正,谢谢!!
原贴地址: http://forum.xda-developers.com/showthread.php?t=1299134