在Win8和WP8应用程序中共享静态库

时间:2013-07-01 03:45:43

标签: windows-8 windows-phone-8 portable-class-library

我有一个用C ++编写的静态库,我希望它可以用于用.NET编写的Windows 8和Windows Phone 8应用程序。

我想通过设置我的解决方案来最小化要维护的单独项目的数量:

尝试#1:只有UI项目是特定于平台的

MySolution.sln
|
+- Library                    (virtual folder to group projects below)
|  |
|  +- LegacyLib.vcxproj       Static Library project - Platform-independant, native C++ code using STL & CRT. Just enough C++/CX to make some types available to .NET and expose a few static methods.
|  |
|  +- LegacyPcl.csproj        Portable Class Library project - References LegacyLib, adds some types, wrappers, interfaces, etc. written in C#.
|
+- SampleApp                  (virtual folder to group projects below)
   |
   +- CommonAppSrc            Platform-independant C# source code (e.g. MVVM classes for Model and ViewModel).
   |
   +- SampleApp_Win8.csproj   Platform-specific App project - References LegacyPcl, includes files from CommonAppSrc using "Add as Link".
   |
   +- SampleApp_Wp8.csproj    Platform-specific App project - References LegacyPcl, includes files from CommonAppSrc using "Add as Link".

问题:似乎Portable Class Libraries不支持以任何形式包含本机C ++代码。

在Windows 8和Windows Phone 8应用程序中包含原生C ++的一种建议方法是创建 Windows运行时组件。不幸的是,针对Win8和Wp8的WRC项目是不同的项目类型; 没有选项可以创建面向多个平台的Windows运行时组件

因此,为每个目标平台创建Windows运行时组件项目会产生以下结果:

尝试#2 - UI项目是特定于平台的,LegacyLib包装器也是如此

MySolution.sln
|
+- Library                    (virtual folder to group projects below)
|  |
|  +- LegacyLib.vcxproj       Static Library project - Platform-independant, native C++ code using STL & CRT. Just enough C++/CX to make some types available to .NET and expose a few static methods.
|  |
|  +- LegacyWrc_Win8.csproj   Platform-specific Windows Runtime Component project - References LegacyLib, adds some types, wrappers, interfaces, etc. written in C#.
|  |
|  +- LegacyWrc_Wp8.csproj    Platform-specific Windows Runtime Component project - References LegacyLib, adds some types, wrappers, interfaces, etc. written in C#.
|
+- SampleApp                  (virtual folder to group projects below)
   |
   +- CommonAppSrc            C# source code (e.g. Model and ViewModel classes).
   |
   +- SampleApp_Win8.csproj   Platform-specific App project - References LegacyWrc_Win8, includes files from CommonAppSrc using "Add as Link".
   |
   +- SampleApp_Wp8.csproj    Platform-specific App project - References LegacyWrc_Wp8, includes files from CommonAppSrc using "Add as Link".

问题:当我尝试从LegacyWrc_Win8项目中引用LegacyLib时;我看到以下警告:

  

建议不要添加对“LegacyLib”的引用,因为它与Windows应用商店应用程序不兼容。

构建环境说要使LegacyLib静态库与我的Win8 Windows运行时组件兼容,我必须在LegacyLib项目配置中启用 Windows应用商店应用支持(忽略警告只会导致稍后出错):

  • 项目属性> 配置属性> 一般> 项目默认值
  • Windows应用商店应用支持设置为

接下来,当我尝试使用LegacyWrc_Wp8项目执行相同操作时,出现以下错误:

  

无法添加对“LegacyLib”的引用,因为这两个项目针对不同的平台。

因此,使静态库与Windows 8兼容,使其与Windows 8 Phone 不兼容

看起来根本无法构建可由这两个不同平台使用的单个静态库,即使库本身不包含特定于平台的代码。

我不一定要维护源代码的并行版本,但我必须创建/维护两个独立的项目,每个项目都构建一个特定于平台的静态库版本。

我的解决方案现在有一个特定于平台的每个项目版本:

尝试#3 - 每个项目都是特定于平台的

MySolution.sln
|
+- Library                        (virtual folder to group projects below)
|  |
|  +- NativeCode                  (virtual folder to group projects below)
|  |  |
|  |  +- CommonLibSrc             Native C++ source code using STL & CRT. Just enough C++/CX to make some types available to .NET and expose a few static methods.
|  |  |
|  |  +- LegacyLib_Win8.vcxproj   Platform-specific Static Library project - Includes files from CommonLibSrc using "Add as Link".
|  |  |
|  |  +- LegacyLib_Wp8.vcxproj    Platform-specific Static Library project - Includes files from CommonLibSrc using "Add as Link".
|  |
|  +- DotNetWrapper               (virtual folder to group projects below)
|     |
|     +- CommonWrcSrc             Platform-independant C# source code (types, wrappers, interfaces, etc. for LegacyLib support) for the Windows Runtime Component projects.
|     |
|     +- LegacyWrc_Win8.csproj    Platform-specific Windows Runtime Component project - References LegacyLib_Win8, includes files from CommonWrcSrc using "Add as Link".
|     |
|     +- LegacyWrc_Wp8.csproj     Platform-specific Windows Runtime Component project - References LegacyLib_Wp8, includes files from CommonWrcSrc using "Add as Link".
|
+- SampleApp                      (virtual folder to group projects below)
   |
   +- CommonAppSrc                Platform-independant C# source code (e.g. Model, ViewModel).
   |
   +- SampleApp_Win8.csproj       Platform-specific App project - References LegacyWrc_Win8, includes files from CommonAppSrc using "Add as Link".
   |
   +- SampleApp_Wp8.csproj        Platform-specific App project - References LegacyWrc_Wp8, includes files from CommonAppSrc using "Add as Link".

问题:目前为止我看不到(除了这个丑陋且维护费用更高的事实)。

BTW,我确实研究过使用Build Configurations的可能性,但它们已经被用于Debug / Release和x86 / ARM的组合。将Win8 / Wp8添加到混合中只会使配置再次加倍,并且您已经转移,而不是减少维护负担。

我错过了什么吗?看起来这是一件很常见的事情。有没有其他人经历过这个并提出替代/更好的方法来做到这一点?

编辑完整性

这是我最终做的......

MySolution.sln
|
+- LegacyApiLib               (virtual folder to group projects below)
|  |
|  +- CommonCppSource         Common C++ source code used by LegacyWrc_*.vcxproj projects
|  |                          below
|  |
|  +- LegacyPcl.csproj        Portable class library with C# API interfaces only; no
|  |                          implementation classes.
|  |
|  +- LegacyWrc_Win8.vcxproj  Platform-specific Windows Runtime Component project - includes
|  |                          legacy C++ files, and uses C++/CX to expose static methods and
|  |                          classes to .NET in Win8.
|  |
|  +- LegacyWrc_Wp8.vcxproj   Platform-specific Windows Runtime Component project - includes
|                             legacy C++ files, and uses C++/CX to expose static methods and
|                             classes to .NET in WP8.
|
+- SampleApp                  (virtual folder to group projects below)
   |
   +- SampleAppBL.csproj      Portable class library containing the app's business logic.
   |                          References LegacyPcl.csproj for legacy API interface definitions.
   |                          Business logic is written against those interfaces; never against
   |                          specific classes. The Win8 or WP8 apps will reference the
   |                          appropriate Windows Runtime Component project, create or register
   |                          the concrete classes that implement the legacy API interface, and
   |                          either pass it into the BL or register it with IOC mechanism. In
   |                          this way, SampleAppBL.csproj never has to know about (or
   |                          reference) any of the WRC projects. 
   |
   +- SampleAppUI_Win8.csproj Platform-specific App project - References LegacyWrc_Win8.vcxproj
   |                          and SampleAppBL. Contains minimal platform-specific Win8 UI code.
   |
   +- SampleAppUI_Wp8.csproj  Platform-specific App project - References LegacyWrc_Wp8.vcxproj
                              and SampleAppBL. Contains minimal platform-specific WP8 UI code.

我意识到LegacyApiLib导出的少数辅助类可以用托管C ++而不是C#编写。这意味着我可以将它们与WRC项目捆绑在一起,这简化了一些事情。

1 个答案:

答案 0 :(得分:2)

正如您所发现的,本地库需要为每个平台单独编译,因此无法直接从可移植类库中引用。

我建议使用抽象模式,允许您通过可移植类库共享大部分.NET代码(即模型和视图模型)。在PCL中,为您在本机库中公开的功能创建接口或抽象类。然后在特定于平台的项目中实现这些项目,将调用转发到本机库。最后使用依赖注入来连接这些接口的实现,以便您可以从共享PCL中调用它们。

以下是一些博文: