Usar bibliotecas C/C++ com o Xamarin
O Xamarin permite que os desenvolvedores criem aplicativos móveis nativos de plataforma cruzada com o Visual Studio. Geralmente, as associações C# são usadas para expor componentes de plataforma existentes aos desenvolvedores. No entanto, há momentos em que os aplicativos Xamarin precisam trabalhar com bases de código existentes. Às vezes, as equipes simplesmente não têm tempo, orçamento ou recursos para portar uma base de código grande, bem testada e altamente otimizada para C#.
O Visual C++ para desenvolvimento móvel entre plataformas permite que o código C/C++ e C# seja criado como parte da mesma solução, oferecendo muitas vantagens, incluindo uma experiência de depuração unificada. A Microsoft usou C/C++ e Xamarin dessa forma para entregar aplicativos como Hyperlapse Mobile e Pix Camera.
No entanto, em alguns casos, há um desejo (ou requisito) de manter as ferramentas e processos C/C++ existentes no lugar e manter o código da biblioteca desacoplado do aplicativo, tratando a biblioteca como se fosse semelhante a um componente de terceiros. Nessas situações, o desafio não é apenas expor os membros relevantes ao C#, mas gerenciar a biblioteca como uma dependência. E, claro, automatizar o máximo possível desse processo.
Este post descreve uma abordagem de alto nível para esse cenário e apresenta um exemplo simples.
C/C++ é considerada uma linguagem multiplataforma, mas muito cuidado deve ser tomado para garantir que o código-fonte seja realmente multiplataforma, usando apenas C/C++ suportado por todos os compiladores de destino e contendo pouca ou nenhuma plataforma condicionalmente incluída ou código específico do compilador.
Em última análise, o código deve ser compilado e executado com sucesso em todas as plataformas de destino, portanto, isso se resume à semelhança entre as plataformas (e compiladores) que estão sendo direcionadas. Os problemas ainda podem surgir de pequenas diferenças entre os compiladores e, portanto, testes completos (de preferência automatizados) em cada plataforma de destino se tornam cada vez mais importantes.
A ilustração abaixo representa a abordagem de quatro estágios usada para transformar o código-fonte C/C++ em uma biblioteca Xamarin de plataforma cruzada que é compartilhada via NuGet e, em seguida, é consumida em um aplicativo Xamarin.Forms.
As 4 etapas são:
- Compilando o código-fonte C/C++ em bibliotecas nativas específicas da plataforma.
- Encapsulando as bibliotecas nativas com uma solução do Visual Studio.
- Empacotando e enviando um pacote NuGet para o wrapper .NET.
- Consumindo o pacote NuGet de um aplicativo Xamarin.
O objetivo desta etapa é criar bibliotecas nativas que podem ser chamadas pelo wrapper C#. Isso pode ou não ser relevante, dependendo da sua situação. As muitas ferramentas e processos que podem ser trazidos à tona nesse cenário comum estão além do escopo deste artigo. As principais considerações são manter a base de código C/C++ sincronizada com qualquer código wrapper nativo, testes de unidade suficientes e automação de compilação.
As bibliotecas no passo a passo foram criadas usando o Visual Studio Code com um script de shell acompanhante. Uma versão estendida deste passo a passo pode ser encontrada no repositório GitHub do CAT móvel que discute essa parte do exemplo com mais profundidade. As bibliotecas nativas estão sendo tratadas como uma dependência de terceiros neste caso, no entanto, este estágio é ilustrado para o contexto.
Para simplificar, o passo a passo destina-se apenas a um subconjunto de arquiteturas. Para iOS, ele usa o utilitário lipo para criar um único binário gordo a partir dos binários específicos da arquitetura individual. O Android usará binários dinâmicos com uma extensão .so e o iOS usará um binário fat estático com uma extensão .a.
O próximo estágio é encapsular as bibliotecas nativas para que elas sejam facilmente usadas a partir do .NET. Isso é feito com uma solução do Visual Studio com quatro projetos. Um projeto compartilhado contém o código comum. Projetos direcionados a cada um dos Xamarin.Android, Xamarin.iOS e .NET Standard permitem que a biblioteca seja referenciada de maneira independente de plataforma.
O invólucro usa "a isca e o truque de trocar". Essa não é a única maneira, mas facilita a referência à biblioteca e evita a necessidade de gerenciar explicitamente implementações específicas da plataforma dentro do próprio aplicativo de consumo. O truque é essencialmente garantir que os destinos (.NET Standard, Android, iOS) compartilhem o mesmo namespace, nome de assembly e estrutura de classe. Como o NuGet sempre prefere uma biblioteca específica da plataforma, a versão do .NET Standard nunca é usada em tempo de execução.
A maior parte do trabalho nesta etapa se concentrará no uso de P/Invoke para chamar os métodos de biblioteca nativa e no gerenciamento das referências aos objetos subjacentes. O objetivo é expor a funcionalidade da biblioteca ao consumidor, abstraindo qualquer complexidade. Os desenvolvedores do Xamarin.Forms não precisam ter conhecimento prático sobre o funcionamento interno da biblioteca não gerenciada. Deve parecer que eles estão usando qualquer outra biblioteca C# gerenciada.
Em última análise, a saída desse estágio é um conjunto de bibliotecas .NET, uma por destino, juntamente com um documento nuspec que contém as informações necessárias para compilar o pacote na próxima etapa.
Estágio 3: Empacotando e enviando um pacote NuGet para o wrapper .NET
O terceiro estágio é a criação de um pacote NuGet usando os artefatos de compilação da etapa anterior. O resultado desta etapa é um pacote NuGet que pode ser consumido de um aplicativo Xamarin. O passo a passo usa um diretório local para servir como o feed NuGet. Na produção, essa etapa deve publicar um pacote em um feed NuGet público ou privado e deve ser totalmente automatizada.
Estágio 4: Consumindo o pacote NuGet de um aplicativo Xamarin.Forms
A etapa final é fazer referência e usar o pacote NuGet de um aplicativo Xamarin.Forms. Isso requer a configuração do feed NuGet no Visual Studio para usar o feed definido na etapa anterior.
Depois que o feed é configurado, o pacote precisa ser referenciado de cada projeto no aplicativo Xamarin.Forms de plataforma cruzada. O 'truque de isca e comutação' fornece interfaces idênticas, para que a funcionalidade da biblioteca nativa possa ser chamada usando código definido em um único local.
O repositório de código-fonte contém uma lista de leituras adicionais que inclui artigos sobre como configurar um feed NuGet privado no Azure DevOps e como enviar o pacote por push para esse feed. Embora exija um pouco mais de tempo de configuração do que um diretório local, esse tipo de feed é melhor em um ambiente de desenvolvimento de equipe.
As etapas fornecidas são específicas do Visual Studio para Mac, mas a estrutura também funciona no Visual Studio 2017 .
Para acompanhar, o desenvolvedor precisará:
Observação
É necessária uma Conta de Programador Apple ativa para implementar aplicações num iPhone.
A funcionalidade de biblioteca nativa é baseada no exemplo de Passo a passo: Criando e usando uma biblioteca estática (C++).
Este passo a passo ignora o primeiro estágio, criando as bibliotecas nativas, já que a biblioteca é fornecida como uma dependência de terceiros nesse cenário. As bibliotecas nativas pré-compiladas são incluídas junto com o código de exemplo ou podem ser baixadas diretamente.
O exemplo MathFuncsLib original inclui uma única classe chamada MyMathFuncs
com a seguinte definição:
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);
};
}
Uma classe adicional define funções de wrapper que permitem que um consumidor .NET crie, descarte e interaja com a classe nativa MyMathFuncs
subjacente.
#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);
}
Serão essas funções de wrapper que serão usadas no lado do Xamarin .
Este estágio requer as bibliotecas pré-compiladas descritas na seção anterior.
No Visual Studio para Mac, clique em Novo Projeto (na Página de Boas-vindas) ou Nova Solução (no menu Arquivo ).
Na janela Novo Projeto, escolha Projeto Compartilhado (de dentro da Biblioteca Multiplataforma>) e clique em Avançar.
Atualize os seguintes campos e clique em Criar:
- Nome do projeto: MathFuncs.Shared
- Nome da solução: MathFuncs
- Local: use o local de salvamento padrão (ou escolha uma alternativa)
- Criar um projeto dentro do diretório da solução: defina isso como marcado
No Gerenciador de Soluções, clique duas vezes no projeto MathFuncs.Shared e navegue até Configurações principais.
Remover . Compartilhado a partir do namespace padrão para que ele seja definido como MathFuncs somente e, em seguida, clique em OK.
Abra MyClass.cs (criado pelo modelo), renomeie a classe e o nome do arquivo para MyMathFuncsWrapper e altere o namespace para MathFuncs.
CONTROL + CLIQUE na solução MathFuncs, em seguida, escolha Adicionar novo projeto...no menu Adicionar.
Na janela Novo Projeto, escolha .NET Standard Library (de dentro da Biblioteca Multiplataforma>) e clique em Avançar.
Escolha .NET Standard 2.0 e clique em Avançar.
Atualize os seguintes campos e clique em Criar:
- Nome do projeto: MathFuncs.Standard
- Local: use o mesmo local de salvamento do projeto compartilhado
No Gerenciador de Soluções, clique duas vezes no projeto MathFuncs.Standard.
Navegue até Configurações principais e atualize o namespace padrão para MathFuncs.
Navegue até as configurações de Saída e atualize o nome do assembly para MathFuncs.
Navegue até as configurações do compilador, altere a configuração para liberar, definindo informações de depuração como Somente símbolos e clique em OK.
Exclua Class1.cs/Introdução do projeto (se um deles tiver sido incluído como parte do modelo).
CONTROL + CLICK na pasta Dependências/Referências do projeto e escolha Editar referências.
Selecione MathFuncs.Shared na guia Projetos e clique em OK.
Repita as etapas 7 a 17 (ignorando a etapa 9) usando as seguintes configurações:
NOME DO PROJETO NOME DO MODELO MENU NOVO PROJETO MathFuncs.Android Biblioteca de Classes Biblioteca Android > MathFuncs.iOS Biblioteca de vinculação Biblioteca do iOS > No Gerenciador de Soluções, clique duas vezes no projeto MathFuncs.Android e navegue até as configurações do compilador.
Com a Configuração definida como Depurar, edite Definir símbolos para incluir o Android;.
Altere a Configuração para Liberar e, em seguida, edite Definir símbolos para incluir também o Android;.
Repita as etapas 19 a 20 para MathFuncs.iOS, editando Definir símbolos para incluir iOS, em vez de Android, em ambos os casos.
Crie a solução na configuração Release (CONTROL + COMMAND + B) e valide se todos os três assemblies de saída (Android, iOS, .NET Standard) (nas respectivas pastas bin do projeto) compartilham o mesmo nome MathFuncs.dll.
Neste estágio, a solução deve ter três destinos, um para Android, iOS e .NET Standard e um projeto compartilhado que é referenciado por cada um dos três destinos. Eles devem ser configurados para usar o mesmo namespace padrão e assemblies de saída com o mesmo nome. Isso é necessário para a abordagem de "isca e interruptor" mencionada anteriormente.
O processo de adicionar as bibliotecas nativas à solução wrapper varia ligeiramente entre Android e iOS.
CONTROL + CLICK no projeto MathFuncs.Android e, em seguida, escolha Nova pasta no menu Adicionar nomeando-o lib.
Para cada ABI (Application Binary Interface), CONTROL + CLICK na pasta lib e, em seguida, escolha Nova pasta no menu Adicionar, nomeando-a após o respectivo ABI. Nesse caso:
- arm64-v8a
- armeabi-v7a
- x86
- x86_64
Observação
Para obter uma visão geral mais detalhada, consulte o tópico Arquiteturas e CPUs do guia do desenvolvedor NDK, especificamente a seção sobre como abordar código nativo em pacotes de aplicativos.
Verifique a estrutura de pastas:
- lib - arm64-v8a - armeabi-v7a - x86 - x86_64
Adicione as bibliotecas .so correspondentes a cada uma das pastas ABI com base no seguinte mapeamento:
arm64-v8a: lib/Android/arm64
armeabi-v7a: lib / Android / braço
x86: lib/Android/x86
x86_64: lib/Android/x86_64
Observação
Para adicionar arquivos, CONTROL + CLIQUE na pasta que representa o respectivo ABI e escolha Adicionar arquivos nomenu Adicionar . Escolha a biblioteca apropriada (no diretório PrecompiledLibs), clique em Abrir e, em seguida, clique em OK deixando a opção padrão para Copiar o arquivo para o diretório.
Para cada um dos arquivos .so , CONTROL + CLICK e escolha a opção EmbeddedNativeLibrary no menu Build Action .
Agora a pasta lib deve aparecer da seguinte maneira:
- lib
- arm64-v8a
- libMathFuncs.so
- armeabi-v7a
- libMathFuncs.so
- x86
- libMathFuncs.so
- x86_64
- libMathFuncs.so
CONTROL + CLIQUE no projeto MathFuncs.iOS e escolha Adicionar referência nativa no menu Adicionar .
Escolha a biblioteca libMathFuncs.a (de libs/ios no diretório PrecompiledLibs ) e clique em Abrir
CONTROL + CLIQUE no arquivo libMathFuncs (dentro da pasta Referências Nativas , em seguida, escolha a opção Propriedades no menu
Configure as propriedades de Referência Nativa para que elas sejam marcadas (mostrando um ícone de tick) no Painel de Propriedades:
- Forçar Carga
- É C++
- Link Inteligente
Observação
O uso de um tipo de projeto de biblioteca de vinculação junto com uma referência nativa incorpora a biblioteca estática e permite que ela seja vinculada automaticamente ao aplicativo Xamarin.iOS que faz referência a ela (mesmo quando ela é incluída por meio de um pacote NuGet).
Abra ApiDefinition.cs, exclua o código comentado do modelo (deixando apenas o
MathFuncs
namespace) e execute a mesma etapa para Structs.csObservação
Um projeto de biblioteca de vinculação requer esses arquivos (com as ações de compilação ObjCBindingApiDefinition e ObjCBindingCoreSource ) para compilar. No entanto, escreveremos o código, para chamar nossa biblioteca nativa, fora desses arquivos de uma forma que possa ser compartilhada entre os destinos da biblioteca Android e iOS usando P/Invoke padrão.
Agora, escreva o código C# para chamar a biblioteca nativa. O objetivo é ocultar qualquer complexidade subjacente. O consumidor não deve precisar de nenhum conhecimento prático dos internos da biblioteca nativa ou dos conceitos P/Invoke.
CONTROL + CLIQUE no projeto MathFuncs.Shared e escolha Adicionar arquivo...no menu Adicionar.
Escolha Classe vazia na janela Novo arquivo , nomeie-a MyMathFuncsSafeHandle e clique em Novo
Implemente a classe 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; } } }
Observação
Um SafeHandle é a maneira preferida de trabalhar com recursos não gerenciados em código gerenciado. Isso abstrai muito código clichê relacionado à finalização crítica e ao ciclo de vida do objeto. O proprietário desse identificador pode tratá-lo subsequentemente como qualquer outro recurso gerenciado e não precisará implementar o padrão descartável completo.
Abra MyMathFuncsWrapper.cs, alterando-o para uma classe estática interna
namespace MathFuncs { internal static class MyMathFuncsWrapper { } }
No mesmo arquivo, adicione a seguinte instrução condicional à classe:
#if Android const string DllName = "libMathFuncs.so"; #else const string DllName = "__Internal"; #endif
Observação
Isso define o valor da constante DllName com base em se a biblioteca está sendo criada para Android ou iOS. Isso é para abordar as diferentes convenções de nomenclatura usadas por cada respectiva plataforma, mas também o tipo de biblioteca que está sendo usada neste caso. O Android está usando uma biblioteca dinâmica e, portanto, espera um nome de arquivo incluindo extensão. Para iOS, '__Internal' é necessário, pois estamos usando uma biblioteca estática.
Adicione uma referência a System.Runtime.InteropServices na parte superior do arquivo MyMathFuncsWrapper.cs
using System.Runtime.InteropServices;
Adicione os métodos wrapper para lidar com a criação e o descarte da classe MyMathFuncs :
[DllImport(DllName, EntryPoint = "CreateMyMathFuncsClass")] internal static extern MyMathFuncsSafeHandle CreateMyMathFuncs(); [DllImport(DllName, EntryPoint = "DisposeMyMathFuncsClass")] internal static extern void DisposeMyMathFuncs(MyMathFuncsSafeHandle ptr);
Observação
Estamos passando nossa constante DllName para o atributo DllImport junto com o EntryPoint que informa explicitamente ao tempo de execução do .NET o nome da função a ser chamada dentro dessa biblioteca. Tecnicamente, não precisamos fornecer o valor EntryPoint se nossos nomes de método gerenciado forem idênticos ao não gerenciado. Se um não for fornecido, o nome do método gerenciado será usado como EntryPoint. No entanto, é melhor ser explícito.
Adicione os métodos wrapper para nos permitir trabalhar com a classe MyMathFuncs usando o seguinte código:
[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);
Observação
Estamos usando tipos simples para os parâmetros neste exemplo. Uma vez que o marshalling é uma cópia bitwise, neste caso, não requer nenhum trabalho adicional da nossa parte. Observe também o uso da classe MyMathFuncsSafeHandle em vez do IntPtr padrão. O IntPtr é mapeado automaticamente para o SafeHandle como parte do processo de empacotamento.
Verifique se a classe MyMathFuncsWrapper concluída aparece como abaixo:
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); } }
Abra a classe MyMathFuncsSafeHandle, navegue até o comentário TODO do espaço reservado dentro do método ReleaseHandle:
// TODO: Release the handle here
Substitua a linha TODO :
MyMathFuncsWrapper.DisposeMyMathFuncs(this);
Agora que o wrapper está concluído, crie uma classe MyMathFuncs que gerenciará a referência ao objeto MyMathFuncs C++ não gerenciado.
CONTROL + CLIQUE no projeto MathFuncs.Shared e escolha Adicionar arquivo...no menu Adicionar.
Escolha Classe vazia na janela Novo arquivo , nomeie-a MyMathFuncs e clique em Novo
Adicione os seguintes membros à classe MyMathFuncs :
readonly MyMathFuncsSafeHandle handle;
Implemente o construtor para a classe para que ele crie e armazene um identificador para o objeto MyMathFuncs nativo quando a classe é instanciada:
public MyMathFuncs() { handle = MyMathFuncsWrapper.CreateMyMathFuncs(); }
Implemente a interface IDisposable usando o seguinte código:
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); } // ... }
Implemente os métodos MyMathFuncs usando a classe MyMathFuncsWrapper para executar o trabalho real sob o capô passando o ponteiro que armazenamos para o objeto não gerenciado subjacente. O código deve ser o seguinte:
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); }
Para ter a biblioteca empacotada e distribuída via NuGet, a solução precisa de um arquivo nuspec . Isso identificará quais dos assemblies resultantes serão incluídos para cada plataforma suportada.
CONTROL + CLICK na solução MathFuncs, em seguida, escolha Adicionar pasta de solução no menu Adicionar nomeando-o SolutionItems.
CONTROL + CLIQUE na pasta SolutionItems e escolha Novo arquivo...no menu Adicionar.
Escolha Arquivo XML vazio na janela Novo arquivo , nomeie-o como MathFuncs.nuspec e clique em Novo.
Atualize o MathFuncs.nuspec com os metadados básicos do pacote a serem exibidos para o consumidor do NuGet . Por exemplo:
<?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>
Observação
Consulte o documento de referência nuspec para obter mais detalhes sobre o esquema usado para este manifesto.
Adicione um
<files>
elemento como filho do<package>
elemento (logo abaixo<metadata>
), identificando cada arquivo com um elemento separado<file>
:<files> <!-- Android --> <!-- iOS --> <!-- netstandard2.0 --> </files>
Observação
Quando um pacote é instalado em um projeto e há vários assemblies especificados pelo mesmo nome, o NuGet escolhe efetivamente o assembly mais específico para determinada plataforma.
Adicione os
<file>
elementos para os assemblies 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" />
Adicione os
<file>
elementos para os assemblies do 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" />
Adicione os
<file>
elementos para os assemblies 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" />
Verifique o manifesto 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>
Observação
Esse arquivo especifica os caminhos de saída do assembly de uma compilação Release , portanto, certifique-se de compilar a solução usando essa configuração.
Neste ponto, a solução contém 3 assemblies .NET e um manifesto nuspec de suporte.
A próxima etapa é empacotar e distribuir o pacote NuGet para que ele possa ser facilmente consumido pelo aplicativo e gerenciado como uma dependência. O empacotamento e o consumo poderiam ser feitos dentro de uma única solução, mas a distribuição da biblioteca via NuGet ajuda no desacoplamento e nos permite gerenciar essas bases de código de forma independente.
A forma mais simples de feed NuGet é um diretório local:
- No Finder, navegue até um diretório conveniente. Por exemplo, /Users.
- Escolha Nova pasta no menu Arquivo, fornecendo um nome significativo, como local-nuget-feed.
Defina a Configuração de compilação como Release e execute uma compilação usando COMMAND + B.
Abra o Terminal e altere o diretório para a pasta que contém o arquivo nuspec .
No Terminal, execute o comando nuget pack especificando o arquivo nuspec , a Versão (por exemplo, 1.0.0) e o OutputDirectory usando a pasta criada na etapa anterior, ou seja, local-nuget-feed. Por exemplo:
nuget pack MathFuncs.nuspec -Version 1.0.0 -OutputDirectory ~/local-nuget-feed
Confirme se MathFuncs.1.0.0.nupkg foi criado no diretório local-nuget-feed .
Uma técnica mais robusta é descrita em Introdução aos pacotes NuGet no Azure DevOps, que mostra como criar um feed privado e enviar o pacote (gerado na etapa anterior) para esse feed.
É ideal ter esse fluxo de trabalho totalmente automatizado, por exemplo, usando o Azure Pipelines. Para obter mais informações, consulte Introdução ao Azure Pipelines.
Para concluir o passo a passo, crie um aplicativo Xamarin.Forms para consumir o pacote recém-publicado no feed NuGet local.
Abra uma nova instância do Visual Studio para Mac. Isso pode ser feito a partir do Terminal:
open -n -a "Visual Studio"
No Visual Studio para Mac, clique em Novo Projeto (na Página de Boas-vindas) ou Nova Solução (no menu Arquivo ).
Na janela Novo Projeto, escolha Aplicativo de Formulários em Branco (de dentro do Aplicativo Multiplataforma>) e clique em Avançar.
Atualize os seguintes campos e clique em Avançar:
- Nome do aplicativo: MathFuncsApp.
- Identificador da Organização: use um namespace reverso, por exemplo, com.{your_org}.
- Plataformas de destino: use o padrão (destinos Android e iOS).
- Código compartilhado: defina isso como .NET Standard (uma solução de "Biblioteca Compartilhada" é possível, mas além do escopo deste passo a passo).
Atualize os seguintes campos e clique em Criar:
- Nome do projeto: MathFuncsApp.
- Nome da solução: MathFuncsApp.
- Local: use o local de salvamento padrão (ou escolha uma alternativa).
No Gerenciador de Soluções, CONTROL + CLIQUE no destino (MathFuncsApp.Android ou MathFuncs.iOS) para o teste inicial e escolha Definir como projeto de inicialização.
Escolha o dispositivo preferido ou emulador de simulador/.
Execute a solução (COMMAND + RETURN) para validar se o projeto Xamarin.Forms modelo é compilado e executado corretamente.
Observação
O iOS (especificamente o simulador) tende a ter o tempo de compilação/implantação mais rápido.
No Visual Studio, escolha Preferências (no menu do Visual Studio ).
Escolha Códigos-fonte na seção NuGet e clique em Adicionar.
Atualize os seguintes campos e clique em Adicionar Origem:
- Nome: forneça um nome significativo, por exemplo, Local-Packages.
- Local: especifique a pasta local-nuget-feed criada na etapa anterior.
Observação
Neste caso, não há necessidade de especificar um Nome de Usuário e Senha.
Clique em OK.
Repita as etapas a seguir para cada projeto (MathFuncsApp, MathFuncsApp.Android e MathFuncsApp.iOS).
- CONTROL + CLICK no projeto e escolha Adicionar pacotes NuGet...no menu Adicionar .
- Procure por MathFuncs.
- Verifique se a versão do pacote é 1.0.0 e os outros detalhes aparecem conforme o esperado, como o título e a descrição, ou seja, MathFuncs e Sample C++ Wrapper Library.
- Selecione o pacote MathFuncs e clique em Adicionar Pacote.
Agora, com uma referência ao pacote MathFuncs em cada um dos projetos, as funções estão disponíveis para o código C#.
Abra MainPage.xaml.cs de dentro do projeto Xamarin.Formscomum do MathFuncsApp (referenciado por MathFuncsApp.Android e MathFuncsApp.iOS).
Adicione instruções using para System.Diagnostics e MathFuncs na parte superior do arquivo:
using System.Diagnostics; using MathFuncs;
Declare uma instância da
MyMathFuncs
classe na parte superior daMainPage
classe:MyMathFuncs myMathFuncs;
Substitua os
OnAppearing
métodos eOnDisappearing
daContentPage
classe base:protected override void OnAppearing() { base.OnAppearing(); } protected override void OnDisappearing() { base.OnDisappearing(); }
Atualize o
OnAppearing
método para inicializar a variável declaradamyMathFuncs
anteriormente:protected override void OnAppearing() { base.OnAppearing(); myMathFuncs = new MyMathFuncs(); }
Atualize o
OnDisappearing
método para chamar oDispose
método emmyMathFuncs
:protected override void OnDisappearing() { base.OnAppearing(); myMathFuncs.Dispose(); }
Implemente um método privado chamado TestMathFuncs da seguinte maneira:
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}"); }
Por fim, chame
TestMathFuncs
no final doOnAppearing
método:TestMathFuncs();
Execute o aplicativo em cada plataforma de destino e valide a saída no Application Output Pad aparece da seguinte maneira:
1 + 2 = 3 1 - 2 = -1 1 * 2 = 2 1 / 2 = 0.5
Observação
Se você encontrar um 'DLLNotFoundException' ao testar no Android, ou um erro de compilação no iOS, certifique-se de verificar se a arquitetura da CPU do dispositivo/emulador/simulador que você está usando é compatível com o subconjunto que escolhemos suportar.
Este artigo explicou como criar um aplicativo Xamarin.Forms que usa bibliotecas nativas por meio de um wrapper .NET comum distribuído por meio de um pacote NuGet. O exemplo fornecido neste passo a passo é intencionalmente muito simplista para demonstrar mais facilmente a abordagem. Um aplicativo real terá que lidar com complexidades, como tratamento de exceções, retornos de chamada, empacotamento de tipos mais complexos e vinculação com outras bibliotecas de dependência. Uma consideração importante é o processo pelo qual a evolução do código C++ é coordenada e sincronizada com o wrapper e os aplicativos cliente. Esse processo pode variar dependendo se uma ou ambas as preocupações são de responsabilidade de uma única equipe. De qualquer forma, a automação é um benefício real. Abaixo estão alguns recursos que fornecem leitura adicional sobre alguns dos principais conceitos, juntamente com os downloads relevantes.