Использование библиотек C/C++ в Xamarin
Обзор
Xamarin позволяет разработчикам создавать собственные кроссплатформенные мобильные приложения с использованием Visual Studio. Как правило, для предоставления разработчикам существующих компонентов платформы используются привязки C#. Однако бывают случаи, когда приложениям Xamarin требуется работать с существующими базами кода. Иногда у команд разработчиков просто нет времени, бюджета или ресурсов для переноса больших, хорошо протестированных и оптимизированных баз кода C#.
Visual C++ для разработки кроссплатформенных мобильных приложений позволяет создавать код C/C++ и C# в рамках одного решения, предлагая множество преимуществ, включая унифицированный процесс отладки. Корпорация Майкрософт использует C/C++ и Xamarin, чтобы предоставлять такие приложения, как Hyperlapse Mobile и Pix Camera.
Однако в некоторых случаях хочется (или требуется) сохранить существующие инструменты и процессы C/C++, а также отделить код библиотек от приложения, чтобы библиотеки использовались как сторонние компоненты. В таких ситуациях необходимо не только предоставить соответствующие элементы в C#, но и управлять библиотекой как зависимостью. И, конечно, следует автоматизировать как можно большую часть этого процесса.
В этой публикации рассматривается обобщенный подход к этому сценарию и демонстрируется простой пример.
Общие сведения
C/C++ считается кроссплатформенным языком, но необходимо уделить исходному коду особое внимание, чтобы он стал действительно кроссплатформенным, используя только версии C/C++, поддерживаемые всеми целевыми компиляторами, и практически не используя условно добавляемый код для определенных платформ или компиляторов.
В конечном итоге код должен успешно компилироваться и выполняться на всех целевых платформах, поэтому все сводится к унификации кода для нескольких целевых платформ (и компиляторов). Проблемы могут возникать и из-за незначительных различий между компиляторами, поэтому становится все более важным тщательное тестирование (желательно автоматическое) на каждой целевой платформе.
Обобщенный подход
На рисунке ниже представлен подход из четырех этапов, используемый для преобразования исходного кода C/C++ в кроссплатформенную библиотеку Xamarin, которая предоставляется посредством NuGet, а затем используется в приложении Xamarin.Forms.
Этот подход состоит из 4 этапов:
- компиляция исходного кода C/C++ в собственные библиотеки платформы;
- создание оболочки для собственных библиотек с помощью решения Visual Studio;
- упаковка и отправка пакета NuGet для оболочки .NET;
- использование пакета NuGet из приложения Xamarin.
Этап 1. Компиляция исходного кода C/C++ в собственные библиотеки для конкретной платформы
Цель этого этапа — создать собственные библиотеки, которые могут вызываться оболочкой C#. В зависимости от конкретной ситуации это может быть уместным или неуместным. Многие инструменты и процессы, которые могут быть включены в этот распространенный сценарий, выходят за рамки данной статьи. Основной идеей является обеспечение синхронизации базы кода C/C++ с любым собственным кодом оболочки, достаточное модульное тестирование и автоматизация сборки.
Библиотеки в пошаговом руководстве были созданы в Visual Studio Code с помощью сопутствующего сценария оболочки. Расширенную версию данного пошагового руководства можно найти в репозитории GitHub Mobile CAT, в котором более подробно рассматривается эта часть примера. В этом случае собственные библиотеки обрабатываются как сторонняя зависимость, однако этот этап показан для контекста.
Для простоты в этом пошаговом руководстве рассматривается только подмножество архитектур. В iOS используется служебная программа lipo для создания одного расширенного двоичного файла на основе отдельных двоичных файлов конкретной архитектуры. Android будет использовать динамические двоичные файлы с расширением ".so", а iOS будет использовать статический расширенный двоичный файл с расширением ".a".
Этап 2. Упаковка собственных библиотек с помощью решения Visual Studio
Следующий этап заключается в заключении в оболочку собственных библиотек, чтобы их можно было легко использовать в .NET. Для этого используется решение Visual Studio с четырьмя проектами. В общем проекте содержится общий код. Проекты, нацеленные на Xamarin.Android, Xamarin.iOS и .NET Standard, позволяют ссылаться на библиотеку независимым от платформы образом.
Оболочка использует прием с подменой. Это не единственный способ, но он упрощает добавление ссылки на библиотеку и позволяет избежать необходимости явно управлять реализациями, зависящими от платформы, в самом приложении. Этот прием гарантирует, что целевые платформы (.NET Standard, Android и iOS) совместно используют одно и то же пространство имен, имя сборки и структуру классов. Так как NuGet всегда предпочитает библиотеку для конкретной платформы, версия .NET Standard никогда не используется во время выполнения.
Большая часть усилий на этом шаге посвящена использованию P/Invoke для вызова методов собственных библиотек и управления ссылками на базовые объекты. Задача заключается в предоставлении функциональных возможностей библиотеки потребителю и абстрагировании любой сложности. Разработчикам для Xamarin.Forms не нужно знать, как работает неуправляемая библиотека изнутри. Они должны использовать ее, как любую другую управляемую библиотеку C#.
В конечном итоге результатом этого этапа будет набор библиотек .NET (по одной для каждой целевой платформы), а также NUSPEC-файл, содержащий сведения, необходимые для сборки пакета на следующем шаге.
Этап 3. Упаковка и отправка пакета NuGet для оболочки .NET
Третий этап — создание пакета NuGet с помощью артефактов сборки из предыдущего шага. Результатом этого этапа является пакет NuGet, который можно использовать из приложения Xamarin. В данном пошаговом руководстве в качестве веб-канала NuGet используется локальный каталог. В рабочей среде на этом шаге пакет должен публиковаться в общедоступном или частном веб-канале NuGet, и этот процесс должен быть полностью автоматизирован.
Этап 4. Использование пакета NuGet из приложения Xamarin.Forms
Последним шагом является добавление ссылки на пакет NuGet и его использование из приложения Xamarin.Forms. Для этого необходимо настроить веб-канал NuGet в Visual Studio, чтобы использовать веб-канал, определенный на предыдущем шаге.
После настройки веб-канала ссылку на пакет нужно добавить в каждый проект в кроссплатформенном приложении Xamarin.Forms. Прием с подменой обеспечивает идентичные интерфейсы, поэтому функции собственной библиотеки можно вызывать с помощью кода, определенного в одном месте.
Репозиторий исходного кода содержит список дополнительных статей о настройке частного веб-канала NuGet в Azure DevOps и о том, как отправить пакет в этот веб-канал. Хотя для настройки такого веб-канала потребуется немного больше времени, чем для настройки локального каталога, этот тип веб-канала лучше использовать в среде коллективной разработки.
Пошаговое руководство
Приведенные действия относятся к Visual Studio для Mac, но описанная структура подходит и для Visual Studio 2017.
Необходимые компоненты
Для работы разработчику потребуется:
Примечание.
Для развертывания приложений на iPhone требуется активная учетная запись разработчика Apple.
Создание собственных библиотек (этап 1)
Функции собственной библиотеки основаны на примере из пошагового руководства. Создание и использование статической библиотеки (C++).
В этом пошаговом руководстве пропускается первый этап — создание собственных библиотек, так как в этом сценарии библиотека предоставляется как сторонняя зависимость. Предварительно скомпилированные собственные библиотеки можно добавить вместе с примером кода или скачать напрямую.
Работа с собственной библиотекой
Исходный пример MathFuncsLib содержит отдельный класс с MyMathFuncs
со следующим определением.
namespace MathFuncs
{
class MyMathFuncs
{
public:
double Add(double a, double b);
double Subtract(double a, double b);
double Multiply(double a, double b);
double Divide(double a, double b);
};
}
Дополнительный класс определяет функции-оболочки, позволяющие потребителю .NET создавать, удалять базовый собственный класс MyMathFuncs
и взаимодействовать с ним.
#include "MyMathFuncs.h"
using namespace MathFuncs;
extern "C" {
MyMathFuncs* CreateMyMathFuncsClass();
void DisposeMyMathFuncsClass(MyMathFuncs* ptr);
double MyMathFuncsAdd(MyMathFuncs *ptr, double a, double b);
double MyMathFuncsSubtract(MyMathFuncs *ptr, double a, double b);
double MyMathFuncsMultiply(MyMathFuncs *ptr, double a, double b);
double MyMathFuncsDivide(MyMathFuncs *ptr, double a, double b);
}
Это функции-оболочки будут использоваться на стороне Xamarin.
Создание оболочки для собственной библиотеки (этап 2)
Для этого этапа требуются предварительно скомпилированные библиотеки, описанные в предыдущем разделе.
Создание решения Visual Studio
В Visual Studio для Mac щелкните Новый проект (на странице приветствия) или Новое решение (в меню Файл).
В окне "Новый проект" выберите "Общий проект" (из многоплатформной > библиотеки) и нажмите кнопку "Далее".
Заполните перечисленные ниже поля и нажмите кнопку Создать.
- Имя проекта: MathFuncs.Shared
- Имя решения: MathFuncs
- Расположение: используйте расположение сохранения по умолчанию (или выберите альтернативу)
- Создайте проект в каталоге решения: задайте для этого значение проверка
В обозревателе решений дважды щелкните проект MathFuncs.Shared и перейдите к разделу Основные параметры.
Удалите .Shared из поля Пространство имен по умолчанию, оставив только MathFuncs, а затем нажмите кнопку ОК.
Откройте MyClass.cs (созданный шаблоном), а затем переименуйте класс и файл в MyMathFuncsWrapper и измените пространство имен на MathFuncs.
Нажмите клавишу CONTROL и щелкните решение MathFuncs, а затем в меню Добавить выберите Добавить новый проект....
В окне "Новый проект" выберите библиотеку .NET Standard (из многоплатформной > библиотеки) и нажмите кнопку "Далее".
Выберите .NET Standard 2.0 и нажмите кнопку Далее.
Заполните перечисленные ниже поля и нажмите кнопку Создать.
- Имя проекта: MathFuncs.Standard
- Расположение: используйте то же расположение сохранения, что и общий проект
В обозревателе решений дважды щелкните проект MathFuncs.Standard.
Перейдите в раздел Основные параметры, а затем в поле Пространство имен по умолчанию укажите MathFuncs.
Перейдите к параметрам выходных данных, а затем измените имя сборки на MathFuncs.
Перейдите к параметрам компилятора, измените параметр Конфигурация на Выпуск, для параметра Отладочная информация выберите значение Только символы, после чего нажмите кнопку ОК.
Удалите Class1.cs/Getting Started из проекта (если один из этих компонентов был включен в шаблон).
Нажмите клавишу CONTROL и щелкните в проекте папку Dependencies/References, а затем выберите Изменить ссылки.
На вкладке Проекты выберите MathFuncs.Shared, а затем нажмите кнопку ОК.
Повторите шаги 7–17 (пропустив шаг 9), используя приведенные ниже конфигурации.
Имя проекта Имя шаблона Меню "Новый проект" MathFuncs.Android Библиотека классов Библиотека Android > MathFuncs.iOS Библиотеки привязок Библиотека iOS > В обозревателе решений дважды щелкните проект MathFuncs.Android и перейдите к параметрам компилятора.
Выбрав для параметра Конфигурация значение Отладка, измените значение Определить символы, чтобы добавить Android;.
Измените значение параметра Конфигурация на Выпуск, затем измените значение Определить символы, чтобы добавить Android;.
Повторите шаги 19–20 для MathFuncs.iOS, но добавляйте в значение Определите символыiOS. вместо Android;.
Выполните сборку решения с конфигурацией Выпуск (CONTROL + COMMAND + B) и убедитесь, что все три выходные сборки (для Android, iOS, .NET Standard, расположенные в соответствующих папках Bin проекта) используют один и тот же файл MathFuncs.dll.
На этом этапе в решении должны быть три цели, по одной для платформ Android, iOS и .NET Standard, а также общий проект, на который ссылается каждая из трех целей. Они должны быть настроены для использования одного и того же пространства имен по умолчанию и выходных сборок с одинаковым именем. Это необходимо для реализации приема с подменой, упомянутого ранее.
Добавление собственных библиотек
Процесс добавления собственных библиотек в решение-оболочку незначительно различается для Android и iOS.
Собственные ссылки для MathFuncs.Android
Нажмите клавишу CTRL и щелкните проект MathFuncs.Android, а затем в меню Добавить выберите Создать папку и укажите имя lib.
Для каждого экземпляра ABI (двоичный интерфейс приложения) нажмите клавишу CTRL и щелкните папку lib, а затем в меню Добавить выберите Создать папку и укажите имя, соответствующее выбранному ABI. В данном случае:
- arm64-v8a
- armeabi-v7a
- x86
- x86_64
Примечание.
Чтобы узнать больше, ознакомьтесь с разделом об архитектурах и ЦПруководства для разработчиков NDK, в частности с разделом о машинном коде в пакетах приложения.
Проверьте структуру папок.
- lib - arm64-v8a - armeabi-v7a - x86 - x86_64
Добавьте соответствующие библиотеки SO-файлов в каждую папку ABI, следуя приведенной ниже схеме:
arm64-v8a: lib/Android/arm64
armeabi-v7a: lib/Android/arm
x86: lib/Android/x86
x86_64: lib/Android/x86_64
Примечание.
Чтобы добавить файлы, нажмите клавишу CONTROL и щелкните папку соответствующего ABI, а затем в меню Добавить выберите Добавить файлы. Выберите соответствующую библиотеку (из каталога PrecompiledLibs), щелкните Открыть и нажмите кнопку ОК, оставив выбранным параметр по умолчанию Скопировать файл в каталог.
Нажмите клавишу CONTROL и щелкните каждый SO-файл, затем в меню Действие сборки выберите EmbeddedNativeLibrary.
Теперь папка lib должна выглядеть так:
- lib
- arm64-v8a
- libMathFuncs.so
- armeabi-v7a
- libMathFuncs.so
- x86
- libMathFuncs.so
- x86_64
- libMathFuncs.so
Собственные ссылки для MathFuncs.iOS
Нажмите клавишу CONTROL и щелкните проект MathFuncs.iOS, а затем в меню Добавить выберите Добавить собственную ссылку.
Выберите библиотеку libMathFuncs.a (из папки libs/ios в каталоге PrecompiledLibs), а затем щелкните Открыть.
Нажмите клавишу CONTROL и щелкните файл libMathFuncs (в папке Native References), затем выберите в меню пункт Свойства.
Настройте свойства собственной ссылки, чтобы они были выбраны (помечены галочкой) на Панели свойств.
- Принудительная загрузка
- Относится к C++
- Интеллектуальная связь
Примечание.
Использование типа проекта библиотеки привязки вместе с собственной ссылкой позволяет внедрить статическую библиотеку и автоматически связать ее с приложением Xamarin.iOS, которое ссылается на нее (даже если библиотека добавлена как пакет NuGet).
Откройте ApiDefinition.cs и удалите закомментированный шаблонный код (оставив только пространство имен
MathFuncs
), а затем выполните то же действие для файла Structs.cs.Примечание.
Эти файлы (как и действия сборки ObjCBindingApiDefinition и ObjCBindingCoreSource) нужны для сборки проекта библиотеки привязки. Однако мы напишем код, вызывающий нашу собственную библиотеку, за пределами этих файлов, чтобы его можно было совместно использовать в целевых объектах библиотек Android и iOS с помощью стандартного метода P/Invoke.
Написание кода управляемой библиотеки
Теперь напишите C# код для вызова собственной библиотеки. Наша задача — скрыть все сложности базовой структуры. Потребителю не нужны навыки работы с внутренними компонентами собственных библиотек или знания принципов работы P/Invoke.
Создание SafeHandle
Нажмите клавишу CONTROL и щелкните проект MathFuncs.Shared, а затем в меню Добавить выберите Добавить файл.
Выберите Пустой класс в окне Новый файл, назовите его MyMathFuncsSafeHandle, а затем щелкните Создать.
Реализуйте класс MyMathFuncsSafeHandle.
using System; using Microsoft.Win32.SafeHandles; namespace MathFuncs { internal class MyMathFuncsSafeHandle : SafeHandleZeroOrMinusOneIsInvalid { public MyMathFuncsSafeHandle() : base(true) { } public IntPtr Ptr => handle; protected override bool ReleaseHandle() { // TODO: Release the handle here return true; } } }
Примечание.
Рекомендуется использовать SafeHandle для работы с неуправляемыми ресурсами в управляемом коде. Это позволяет абстрагировать большой объем стандартного кода, связанного с критическим завершением и жизненным циклом объектов. Владелец этого дескриптора сможет впоследствии обрабатывать его так же, как и любой другой управляемый ресурс, и ему не придется реализовывать полный высвобождаемый шаблон.
Создание внутреннего класса-оболочки
Откройте MyMathFuncsWrapper.cs, изменив его на внутренний статический класс.
namespace MathFuncs { internal static class MyMathFuncsWrapper { } }
В том же самом файле добавьте в класс приведенный ниже условный оператор.
#if Android const string DllName = "libMathFuncs.so"; #else const string DllName = "__Internal"; #endif
Примечание.
Это позволит задать значение константы DllName в зависимости от того, для какой платформы выполняется сборка библиотеки, Android или iOS. Это позволяет соблюсти различные соглашения об именовании, используемые каждой платформой, а также учесть тип библиотеки, используемой в каждом случае. Android использует динамическую библиотеку, поэтому ожидает имя файла, включая расширение. Для iOS требуется имя с "__Internal", так как используется статическая библиотека.
Добавьте ссылку на System.Runtime.InteropServices в начало файла MyMathFuncsWrapper.cs.
using System.Runtime.InteropServices;
Добавьте методы-оболочки для создания и удаления класса MyMathFuncs.
[DllImport(DllName, EntryPoint = "CreateMyMathFuncsClass")] internal static extern MyMathFuncsSafeHandle CreateMyMathFuncs(); [DllImport(DllName, EntryPoint = "DisposeMyMathFuncsClass")] internal static extern void DisposeMyMathFuncs(MyMathFuncsSafeHandle ptr);
Примечание.
Мы передаем константу DllName в атрибут DllImport вместе с EntryPoint, который явно указывает среде выполнения .NET имя функции для вызова в этой библиотеке. Технически не нужно предоставлять значение EntryPoint, если имена управляемых методов идентичны неуправляемым. Если значение EntryPoint не указано, то вместо него будет использоваться имя управляемого метода. Однако лучше задать это значение явно.
Добавьте методы-оболочки, чтобы мы могли работать с классом MyMathFuncs, используя следующий код.
[DllImport(DllName, EntryPoint = "MyMathFuncsAdd")] internal static extern double Add(MyMathFuncsSafeHandle ptr, double a, double b); [DllImport(DllName, EntryPoint = "MyMathFuncsSubtract")] internal static extern double Subtract(MyMathFuncsSafeHandle ptr, double a, double b); [DllImport(DllName, EntryPoint = "MyMathFuncsMultiply")] internal static extern double Multiply(MyMathFuncsSafeHandle ptr, double a, double b); [DllImport(DllName, EntryPoint = "MyMathFuncsDivide")] internal static extern double Divide(MyMathFuncsSafeHandle ptr, double a, double b);
Примечание.
В этом примере мы используем простые типы для параметров. Так как маршалинг означает побитовое копирование, в этом случае с нашей стороны больше действий не требуется. Также обратите внимание на использование класса MyMathFuncsSafeHandle вместо стандартного IntPtr. IntPtr автоматически сопоставляется с SafeHandle в процессе маршалинга.
Убедитесь, что завершенный класс MyMathFuncsWrapper выглядит следующим образом.
using System.Runtime.InteropServices; namespace MathFuncs { internal static class MyMathFuncsWrapper { #if Android const string DllName = "libMathFuncs.so"; #else const string DllName = "__Internal"; #endif [DllImport(DllName, EntryPoint = "CreateMyMathFuncsClass")] internal static extern MyMathFuncsSafeHandle CreateMyMathFuncs(); [DllImport(DllName, EntryPoint = "DisposeMyMathFuncsClass")] internal static extern void DisposeMyMathFuncs(MyMathFuncsSafeHandle ptr); [DllImport(DllName, EntryPoint = "MyMathFuncsAdd")] internal static extern double Add(MyMathFuncsSafeHandle ptr, double a, double b); [DllImport(DllName, EntryPoint = "MyMathFuncsSubtract")] internal static extern double Subtract(MyMathFuncsSafeHandle ptr, double a, double b); [DllImport(DllName, EntryPoint = "MyMathFuncsMultiply")] internal static extern double Multiply(MyMathFuncsSafeHandle ptr, double a, double b); [DllImport(DllName, EntryPoint = "MyMathFuncsDivide")] internal static extern double Divide(MyMathFuncsSafeHandle ptr, double a, double b); } }
Завершение класса MyMathFuncsSafeHandle
Откройте класс MyMathFuncsSafeHandle и перейдите к заполняющему комментарию TODO в методе ReleaseHandle.
// TODO: Release the handle here
Замените строку TODO.
MyMathFuncsWrapper.DisposeMyMathFuncs(this);
Написание класса MyMathFuncs
Теперь, когда оболочка завершена, создайте класс MyMathFuncs, который будет управлять ссылкой на неуправляемый объект C++ MyMathFuncs.
Нажмите клавишу CONTROL и щелкните проект MathFuncs.Shared, а затем в меню Добавить выберите Добавить файл.
Выберите Пустой класс в окне Новый файл, назовите его MyMathFuncs, а затем щелкните Создать.
Добавьте следующие члены в класс MyMathFuncs.
readonly MyMathFuncsSafeHandle handle;
Реализуйте конструктор класса, чтобы он создавал и сохранял дескриптор для собственного объекта MyMathFuncs при создании экземпляра класса.
public MyMathFuncs() { handle = MyMathFuncsWrapper.CreateMyMathFuncs(); }
Реализуйте интерфейс IDisposable, используя следующий код.
public class MyMathFuncs : IDisposable { ... protected virtual void Dispose(bool disposing) { if (handle != null && !handle.IsInvalid) handle.Dispose(); } public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } // ... }
Реализуйте методы MyMathFuncs с помощью класса MyMathFuncsWrapper, чтобы выполнять внутренние операции, передавая указатель, сохраненный в базовом неуправляемом объекте. Код должен выглядеть следующим образом.
public double Add(double a, double b) { return MyMathFuncsWrapper.Add(handle, a, b); } public double Subtract(double a, double b) { return MyMathFuncsWrapper.Subtract(handle, a, b); } public double Multiply(double a, double b) { return MyMathFuncsWrapper.Multiply(handle, a, b); } public double Divide(double a, double b) { return MyMathFuncsWrapper.Divide(handle, a, b); }
Создание NUSPEC-файла
Для упаковки и распространения библиотеки с помощью NuGet решению требуется NUSPEC-файл. Он определит, какие из полученных сборок будут включаться для каждой поддерживаемой платформы.
Нажмите клавишу CONTROL и щелкните решение MathFuncs, а затем в меню Добавить выберите Добавить папку решения и укажите имя SolutionItems.
Нажмите клавишу CONTROL и щелкните папку SolutionItems, а затем в меню Добавить выберите Новый файл.
Выберите Empty XML File (Пустой XML-файл) в окне Новый файл, назовите его MathFuncs.nuspec, а затем щелкните Создать.
Измените MathFuncs.nuspec, добавив в него простые метаданные пакета, которые будут отображаться для потребителя NuGet. Рассмотрим пример.
<?xml version="1.0"?> <package> <metadata> <id>MathFuncs</id> <version>$version$</version> <authors>Microsoft Mobile Customer Advisory Team</authors> <description>Sample C++ Wrapper Library</description> <requireLicenseAcceptance>false</requireLicenseAcceptance> <copyright>Copyright 2018</copyright> </metadata> </package>
Примечание.
Дополнительные сведения о схеме, используемой для этого манифеста, см. в документе по NUSPEC.
Добавьте элемент
<files>
в качестве дочернего элемента<package>
(под<metadata>
), обозначая каждый файл с помощью отдельного элемента<file>
.<files> <!-- Android --> <!-- iOS --> <!-- netstandard2.0 --> </files>
Примечание.
При установке пакета в проект и наличии нескольких сборок с одним и тем же именем NuGet эффективно выберет сборку, наиболее подходящую для заданной платформы.
Добавьте элементы
<file>
для сборок Android.<file src="MathFuncs.Android/bin/Release/MathFuncs.dll" target="lib/MonoAndroid81/MathFuncs.dll" /> <file src="MathFuncs.Android/bin/Release/MathFuncs.pdb" target="lib/MonoAndroid81/MathFuncs.pdb" />
Добавьте элементы
<file>
для сборок iOS.<file src="MathFuncs.iOS/bin/Release/MathFuncs.dll" target="lib/Xamarin.iOS10/MathFuncs.dll" /> <file src="MathFuncs.iOS/bin/Release/MathFuncs.pdb" target="lib/Xamarin.iOS10/MathFuncs.pdb" />
Добавьте элементы
<file>
для сборок netstandard2.0.<file src="MathFuncs.Standard/bin/Release/netstandard2.0/MathFuncs.dll" target="lib/netstandard2.0/MathFuncs.dll" /> <file src="MathFuncs.Standard/bin/Release/netstandard2.0/MathFuncs.pdb" target="lib/netstandard2.0/MathFuncs.pdb" />
Проверьте NUSPEC-файл манифеста.
<?xml version="1.0"?> <package> <metadata> <id>MathFuncs</id> <version>$version$</version> <authors>Microsoft Mobile Customer Advisory Team</authors> <description>Sample C++ Wrapper Library</description> <requireLicenseAcceptance>false</requireLicenseAcceptance> <copyright>Copyright 2018</copyright> </metadata> <files> <!-- Android --> <file src="MathFuncs.Android/bin/Release/MathFuncs.dll" target="lib/MonoAndroid81/MathFuncs.dll" /> <file src="MathFuncs.Android/bin/Release/MathFuncs.pdb" target="lib/MonoAndroid81/MathFuncs.pdb" /> <!-- iOS --> <file src="MathFuncs.iOS/bin/Release/MathFuncs.dll" target="lib/Xamarin.iOS10/MathFuncs.dll" /> <file src="MathFuncs.iOS/bin/Release/MathFuncs.pdb" target="lib/Xamarin.iOS10/MathFuncs.pdb" /> <!-- netstandard2.0 --> <file src="MathFuncs.Standard/bin/Release/netstandard2.0/MathFuncs.dll" target="lib/netstandard2.0/MathFuncs.dll" /> <file src="MathFuncs.Standard/bin/Release/netstandard2.0/MathFuncs.pdb" target="lib/netstandard2.0/MathFuncs.pdb" /> </files> </package>
Примечание.
Этот файл указывает пути вывода сборки для конфигурации Выпуск, поэтому обязательно выполните сборку решения, используя эту конфигурацию.
На этом этапе решение содержит 3 сборки .NET и вспомогательный NUSPEC-файл манифеста.
Распространение оболочки .NET с помощью NuGet
Следующим шагом является упаковка и распространение пакета NuGet, чтобы его можно было легко использовать в приложении и управлять им как зависимостью. Упаковку и потребление можно выполнять в рамках одного решения, но распространение библиотеки с помощью NuGet поможет разделить эти базы кода и управлять ими независимо друг от друга.
Подготовка локального каталога пакетов
Самая простая форма веб-канала NuGet — это локальный каталог.
- В Finder перейдите к удобному каталогу. Например, /Users.
- В меню Файл выберите Создать папку и укажите понятное имя, например local-nuget-feed.
Создание пакета
Для параметра Конфигурация сборки выберите значение Выпуск и выполните сборку, нажав клавиши COMMAND + B.
Откройте Терминал и перейдите в каталог, содержащую NUSPEC-файл.
В Терминале выполните команду nuget pack, указав NUSPEC-файл, значение Version (например, 1.0.0) и OutputDirectory, указав папку, созданную на предыдущем шаге, то есть local-nuget-feed. Например:
nuget pack MathFuncs.nuspec -Version 1.0.0 -OutputDirectory ~/local-nuget-feed
Убедитесь, что в каталоге local-nuget-feed создан файл MathFuncs.1.0.0.nupkg.
[Дополнительно] Использование частного веб-канала NuGet в Azure DevOps
Более надежная методика описана в разделе о начале работы с пакетами NuGet в Azure DevOps, в котором показано, как создать частный веб-канал и отправить в него пакет, созданный на предыдущем шаге.
Рекомендуется полностью автоматизировать этот рабочий процесс, например с помощью Azure Pipelines. Чтобы узнать больше, ознакомьтесь с разделом Get started with Azure Pipelines (Приступая к работе с Azure Pipelines).
Использование оболочки .NET из приложения Xamarin.Forms
Чтобы выполнить данное пошаговое руководство, создайте приложение Xamarin.Forms для использования пакета, только что опубликованного в локальном веб-канале NuGet.
Создание проекта Xamarin.Forms
Откройте новый экземпляр Visual Studio для Mac. Это можно сделать из Терминала.
open -n -a "Visual Studio"
В Visual Studio для Mac щелкните Новый проект (на странице приветствия) или Новое решение (в меню Файл).
В окне "Новый проект" выберите пустое приложение Forms (из мультиплатформенных > приложений) и нажмите кнопку "Далее".
Заполните перечисленные ниже поля и нажмите кнопку Далее.
- Имя приложения: MathFuncsApp.
- Идентификатор организации: используйте обратное пространство имен, например com.{your_org}.
- Целевые платформы: используйте целевые объекты Android и iOS по умолчанию.
- Общий код: установите это значение в .NET Standard (возможно решение "Общая библиотека", но за пределами область этого пошагового руководства).
Заполните перечисленные ниже поля и нажмите кнопку Создать.
- Имя проекта: MathFuncsApp.
- Имя решения: MathFuncsApp.
- Расположение: используйте расположение сохранения по умолчанию (или выберите альтернативу).
В обозревателе решенийнажмите клавишу CONTROL и щелкните целевой объект (MathFuncsApp.Android или MathFuncsApp.iOS) для начального тестирования, а затем выберите Назначить в качестве автозагружаемого проекта.
Выберите предпочитаемое устройство или щелкните Симулятор/Эмулятор.
Запустите решение (нажмите клавиши COMMAND + RETURN), чтобы убедиться в том, что проект на основе шаблонаXamarin.Forms создан и работает нормально.
Примечание.
iOS (в частности симулятор), как правило, обеспечивает самую быструю сборку и развертывание.
Добавление локального веб-канала NuGet в конфигурацию NuGet
В Visual Studio выберите Настройка (в меню Visual Studio).
В разделе NuGet выберите Источники, а затем щелкните Добавить.
Заполните перечисленные ниже поля и нажмите кнопку Добавить источник.
- Имя: укажите понятное имя, например Local-Packages.
- Расположение. Укажите папку local-nuget-feed, созданную на предыдущем шаге.
Примечание.
В этом случае нет необходимости указывать имя пользователя и пароль.
Щелкните OK.
Указание ссылок на пакет
Повторите следующие шаги для каждого проекта (MathFuncsApp, MathFuncsApp.Android и MathFuncsApp.iOS).
- Нажмите клавишу CONTROL и щелкните проект, затем в меню Добавить выберите Add NuGet Packages (Добавить пакеты NuGet).
- Выполните поиск MathFuncs.
- Убедитесь, что версия пакета имеет значение 1.0.0 и другие сведения отображаются правильно, например, отображаются заголовок и описание, то есть MathFuncs и Sample C++ Wrapper Library.
- Выберите пакет MathFuncs, а затем щелкните Добавить пакет.
Использование функций библиотеки
Мы добавили ссылки на пакет MathFuncs в каждый проект, и теперь его функции можно использовать в коде C#.
Откройте MainPage.xaml.cs из общего проекта Xamarin.FormsMathFuncsApp (на который ссылаются как MathFuncsApp.Android, так и MathFuncsApp.iOS).
Добавьте операторы using для System.Diagnostics и MathFuncs в начало этого файла.
using System.Diagnostics; using MathFuncs;
Объявите экземпляр класса
MyMathFuncs
в начале классаMainPage
.MyMathFuncs myMathFuncs;
Переопределите методы
OnAppearing
иOnDisappearing
базового классаContentPage
.protected override void OnAppearing() { base.OnAppearing(); } protected override void OnDisappearing() { base.OnDisappearing(); }
Измените метод
OnAppearing
, чтобы инициализировать переменнуюmyMathFuncs
, объявленную ранее.protected override void OnAppearing() { base.OnAppearing(); myMathFuncs = new MyMathFuncs(); }
Теперь измените метод
OnDisappearing
, добавив вызов методаDispose
дляmyMathFuncs
.protected override void OnDisappearing() { base.OnAppearing(); myMathFuncs.Dispose(); }
Реализуйте частный метод TestMathFuncs следующим образом.
private void TestMathFuncs() { var numberA = 1; var numberB = 2; // Test Add function var addResult = myMathFuncs.Add(numberA, numberB); // Test Subtract function var subtractResult = myMathFuncs.Subtract(numberA, numberB); // Test Multiply function var multiplyResult = myMathFuncs.Multiply(numberA, numberB); // Test Divide function var divideResult = myMathFuncs.Divide(numberA, numberB); // Output results Debug.WriteLine($"{numberA} + {numberB} = {addResult}"); Debug.WriteLine($"{numberA} - {numberB} = {subtractResult}"); Debug.WriteLine($"{numberA} * {numberB} = {multiplyResult}"); Debug.WriteLine($"{numberA} / {numberB} = {divideResult}"); }
Наконец, добавьте вызов
TestMathFuncs
в конец методаOnAppearing
.TestMathFuncs();
Запустите приложение на каждой целевой платформе и проверьте выходные данные на панели выходных данных приложения, как показано ниже.
1 + 2 = 3 1 - 2 = -1 1 * 2 = 2 1 / 2 = 0.5
Примечание.
В случае обнаружения исключения DLLNotFoundException при тестировании в Android или при возникновении ошибки сборки в iOS убедитесь, что архитектура ЦП устройства, эмулятора или симулятора, которую вы используете, совместима с подмножеством архитектур, которые мы выбрали для поддержки.
Итоги
В этой статье объясняется, как создать приложение Xamarin.Forms, использующее собственные библиотеки через общую оболочку .NET, распространяемую как пакет NuGet. Пример, приведенный в этом пошаговом руководстве, намеренно очень упрощен в целях демонстрации метода. Реальное приложение потребует реализации сложных механизмов, таких как обработка исключений, обратные вызовы, маршалинг более сложных типов и привязка к другим библиотекам зависимостей. Основной идеей является процесс, с помощью которого развитие C++ кода координируется и синхронизируется с оболочкой и клиентскими приложениями. Этот процесс может отличаться в зависимости от того, отвечает ли за выполнение обеих задач одна группа разработчиков. В любом случае, автоматизация является реальным преимуществом. Ниже приведены ссылки на материалы для дальнейшего чтения, посвященные основным принципам, а также ссылки для скачивания соответствующих компонентов.
Файлы для загрузки
Примеры
- Использование Hyperlapse для разработки кроссплатформенных мобильных приложений на языке C++
- Microsoft PIX (C++ и Xamarin)