Respuestas a preguntas que probablemente tengas acerca de la creación y del consumo de las API de Windows Runtime con C++/WinRT.
Importante
Para obtener las notas de la versión sobre C++/WinRT, consulta Noticias y cambios en C++/WinRT 2.0.
Nota
Si tu pregunta es sobre un mensaje de error que has visto, consulta también el tema Solución de problemas de C++/WinRT.
¿Dónde puedo encontrar aplicaciones de ejemplo de C++/WinRT?
Consulte Aplicaciones de ejemplo de C++/WinRT.
¿Cómo redirijo el proyecto de C++/WinRT a una versión posterior del SDK de Windows?
Consulta Procedimientos para redirigir el proyecto de C++/WinRT a una versión posterior del SDK de Windows.
¿Por qué no se va a compilar mi nuevo proyecto, ahora que he cambiado a C++WinRT 2.0?
Para todo el conjunto de cambios (incluidos los cambios importantes), consulta Noticias y cambios, en C++/WinRT 2.0. Por ejemplo, si usas for
basado en intervalo en una colección de Windows Runtime, deberás aplicar ahora #include <winrt/Windows.Foundation.Collections.h>
.
¿Por qué no se va a compilar mi nuevo proyecto? Estoy usando Visual Studio 2017 (versión 15.8.0 o superior) y el SDK versión 17134
Si usas Visual Studio 2017 (versión 15.8.0 o superior) y te diriges al SDK de Windows versión 10.0.17134.0 (Windows 10, versión 1803), un proyecto recién creado de C++/WinRT puede generar un error al compilarse con el "error C3861: 'from_abi': no se encontró el identificador" y otros errores que se originan en base.h. La solución consiste en dirigirte a una versión posterior (más compatible) del SDK de Windows o establecer la propiedad del proyecto C/C++>Lenguaje>Modo de conformidad: No (además, si /permissive- aparece en la propiedad del proyecto C/C++>Línea de comandos en Opciones adicionales, elimínalo).
¿Cómo resuelvo el error de compilación "The C++/WinRT VSIX no longer provides project build support. Please add a project reference to the Microsoft.Windows.CppWinRT Nuget package"? (VSIX de C++WinRT ya no proporciona compatibilidad con la compilación del proyecto. Agregue una referencia de proyecto al paquete NuGet Microsoft.Windows.CppWinRT).
Instala el paquete NuGet Microsoft.Windows.CppWinRT en el proyecto. Para más información, consulta Versiones anteriores de la extensión VSIX.
¿Cómo personalizo la compatibilidad de la compilación en el paquete NuGet?
La compatibilidad de la compilación de C++/WinRT (props/targets) se documenta en readme del paquete NuGet Microsoft.Windows.CppWinRT.
¿Cuáles son los requisitos para la Extensión de Visual Studio (VSIX) para C++/WinRT?
Para la versión 1.0.190128.4 de la extensión de VSIX y posteriores, consulta Compatibilidad de Visual Studio para C++/WinRT. Para otras versiones, consulta Versiones anteriores de la extensión VSIX.
¿Qué es una clase en tiempo de ejecución?
Una clase en tiempo de ejecución es un tipo que puede activarse y consumirse a través de interfaces COM modernas, normalmente a través de límites de archivos ejecutables. Sin embargo, una clase en tiempo de ejecución también puede usarse dentro de la unidad de compilación que la implementa. Al declarar una clase en tiempo de ejecución en el lenguaje de definición de interfaz (IDL), puedes implementarla en C++ estándar con C++/WinRT.
¿Qué significa tipo proyectado y tipo de implementación?
Si solo vas a consumir una clase de Windows Runtime (clase en tiempo de ejecución), tratarás exclusivamente con tipos proyectados. C++/WinRT es una proyección de lenguaje, de modo que los tipos proyectados forman parte de la superficie de Windows Runtime que se proyecta en C++ con C++/WinRT. Para más información, consulta Consumir API con C++/WinRT.
El tipo de implementación contiene la implementación de una clase en tiempo de ejecución, por lo que solo está disponible en el proyecto que implementa la clase en tiempo de ejecución. Cuando estés trabajando en un proyecto que implemente clases en tiempo de ejecución (un proyecto de componente de Windows Runtime o un proyecto que use interfaz de usuario XAML), es importante que te sientas cómodo con la distinción entre tu tipo de implementación para una clase en tiempo de ejecución y el tipo proyectado que representa la clase en tiempo de ejecución proyectada en C++/WinRT. Para más información, consulta Crear API con C++/WinRT.
¿Tengo que declarar un constructor en el archivo IDL de mi clase en tiempo de ejecución?
Solo si la clase en tiempo de ejecución se ha diseñado para consumirse desde fuera de su unidad de compilación de implementación (es un componente de Windows Runtime destinado al consumo general por aplicaciones cliente de Windows Runtime). Para una información más completa sobre el propósito y las consecuencias de declarar constructores en IDL, consulta Constructores de clases en tiempo de ejecución.
¿Por qué el compilador me devuelve el error "C3779: consume_Something: function that returns 'auto' cannot be used before it is defined" (C3779: consume_Something: la función que devuelve 'auto' no se puede usar antes de definirse)?
Está usando un objeto de Windows Runtime sin haber incluido primero el archivo de encabezado del espacio de nombres correspondiente. Incluye el encabezado nombrado para el espacio de nombres de la API y vuelve a compilar. Para más información, consulta Encabezados de proyección de C++/WinRT.
¿Por qué el enlazador me da un "error LNK2019: símbolo externo sin resolver"?
Si el símbolo no resuelto es una función libre de Windows Runtime, como RoInitialize, tendrás que vincular explícitamente la biblioteca paraguas WindowsApp.lib en el proyecto. La proyección de C++/WinRT depende de algunas de estas funciones y puntos de entrada libres (no miembros). Si usas una de las plantillas de proyecto de la Extensión de Visual Studio (VSIX) para C++/WinRT para la aplicación, WindowsApp.lib
se vincula automáticamente. De lo contrario, puedes usar la configuración de vínculo del proyecto para incluirla o hacerlo en el código fuente.
#pragma comment(lib, "windowsapp")
Es importante que resuelvas los errores del enlazador que puedas mediante la vinculación de WindowsApp.lib en lugar de una biblioteca de vínculos estáticos alternativa, en caso contrario, la aplicación no pasará las pruebas del Kit de certificación de aplicaciones de Windows usadas por Visual Studio y Microsoft Store para validar los envíos (es decir que no será posible que tu aplicación se ingiera correctamente en Microsoft Store).
Si el símbolo sin resolver es un constructor, es posible que haya olvidado incluir el archivo de encabezado del espacio de nombres para la clase que se está construyendo. Incluya el encabezado correspondiente al espacio de nombres de la clase y vuelva a compilar. Para más información, consulta Encabezados de proyección de C++/WinRT.
¿Por qué se muestra la excepción "clase no registrada"?
En este caso, el síntoma es que, al construir una clase en tiempo de ejecución o acceder a un miembro estático, aparece una excepción en tiempo de ejecución con un valor HRESULT de REGDB_E_CLASSNOTREGISTERED.
Una causa de este error puede ser que no se puede cargar el componente de Windows Runtime. Asegúrate de que el archivo de metadatos (.winmd
) de Windows Runtime del componente tenga el mismo nombre que el binario del componente (el .dll
), que también es el nombre del proyecto y el nombre del espacio de nombres raíz. Asegúrate también de que el proceso de compilación haya copiado correctamente los metadatos de Windows Runtime y el binario en la carpeta Appx
de la aplicación de consumo. Y comprueba que el AppxManifest.xml
de la aplicación de consumo (también en la carpeta Appx
) contenga un elemento <InProcessServer> que declare correctamente la clase activable y el nombre del binario.
Construcción uniforme Este error también puede ocurrir si intenta crear una instancia de una clase en tiempo de ejecución implementada localmente a través de cualquiera de los constructores del tipo proyectado (distinto de su constructor std::nullptr_t). Para ello, necesitarás la característica C++/WinRT 2.0 que suele denominarse construcción uniforme. Si quieres participar en esa característica, consulta Participación en la construcción uniforme y acceso de implementación directa para obtener más información y ejemplos de código.
Si buscas una forma de crear una instancia de tus clases en tiempo de ejecución implementadas localmente que no requiera la construcción uniforme, consulta Controles de XAML; enlazar a una propiedad de C++/WinRT.
¿Debo implementar Windows::Foundation::IClosable? Y, si es así, ¿cómo?
Si tiene una clase en tiempo de ejecución que libera recursos en su destructor, y si dicha clase en tiempo de ejecución se ha diseñado para consumirse desde fuera de su unidad de compilación de implementación (es un componente de Windows Runtime destinado al consumo general por aplicaciones cliente de Windows Runtime), le recomendamos que también implemente IClosable para poder admitir el consumo de su clase en tiempo de ejecución con lenguajes que carecen de finalización determinista. Asegúrate de que tus recursos se liberan si se llama al constructor, a IClosable::Close o a ambos. Se puede llamar a IClosable::Close un número arbitrario de veces.
¿Tengo que llamar a IClosable::Close en las clases en tiempo de ejecución que consumo?
IClosable existe para admitir los lenguajes que carecen de finalización determinista. Por lo tanto, en general, no es necesario llamar a IClosable::Close desde C++/WinRT. Pero ten en cuenta estas excepciones a esa regla general.
- Hay casos muy excepcionales que implican carreras de apagado o interbloqueos, donde sí tienes que llamar a IClosable::Close. Si vas a usar tipos Windows.UI.Composition, por ejemplo, puede que te encuentres con casos en los que quieras deshacerte de objetos en una secuencia definida, como una alternativa para permitir que la destrucción del contenedor de C++/WinRT haga el trabajo por ti.
- Si no puedes garantizar que tienes la última referencia restante a un objeto (porque la pasaste a otras API, que podrían estar manteniendo una referencia), es conveniente llamar a IClosable::Close.
- En caso de duda, es seguro llamar a IClosable::Close manualmente, en lugar de esperar a que el contenedor lo llame en la destrucción.
Por lo tanto, si sabes que tienes la última referencia, puedes permitir que el destructor del contenedor realice su trabajo. Si necesitas cerrarlo antes de que desaparezca la última referencia, debes realizar la llamada Close. Para que no se apliquen excepciones, la llamada a Close debe ser de tipo resource-acquisition-is-initialization (RAII) (de este modo, el cierre se completa durante el desenredo). C++/WinRT no tiene ningún contenedor unique_close, pero puedes crear uno.
¿Puedo usar LLVM/Clang para compilar con C++/WinRT?
No admitimos la cadena de herramientas de LLVM y Clang para C++/WinRT, pero hacemos uso de ella internamente para validar la conformidad con los estándares de C++/WinRT. Por ejemplo, si quisieras emular lo que hacemos internamente, podrías intentar un experimento, como el que se describe a continuación.
Ve a LLVM Download Page (Página de descarga de LLVM), busca Download LLVM 6.0.0 (Descargar LLVM 6.0.0)>Pre-Built Binaries (Archivos binarios integrados previamente) y descarga Clang for Windows (64-bit) [Clang para Windows (64 bits)]. Durante la instalación, opta por agregar LLVM a la variable del sistema PATH para que puedas invocarlo desde un símbolo del sistema. Para los fines de este experimento, puedes hacer caso omiso de los errores de tipo "No se pudo encontrar el directorio de conjuntos de herramientas de MSBuild" o "Error de instalación de la integración de MSVC", si aparecen. Hay una gran variedad de formas para invocar a LLVM/Clang; el siguiente ejemplo constituye solo una de ellas.
C:\ExperimentWithLLVMClang>type main.cpp
// main.cpp
#pragma comment(lib, "windowsapp")
#pragma comment(lib, "ole32")
#include <winrt/Windows.Foundation.h>
#include <stdio.h>
#include <iostream>
using namespace winrt;
int main()
{
winrt::init_apartment();
Windows::Foundation::Uri rssFeedUri{ L"https://blogs.windows.com/feed" };
std::wcout << rssFeedUri.Domain().c_str() << std::endl;
}
C:\ExperimentWithLLVMClang>clang-cl main.cpp /EHsc /I ..\.. -Xclang -std=c++17 -Xclang -Wno-delete-non-virtual-dtor -o app.exe
C:\ExperimentWithLLVMClang>app
windows.com
Dado que C++/WinRT usa características del estándar de C++17, deberás utilizar los marcadores del compilador que sean necesarios para obtener dicha compatibilidad; estos marcadores difieren entre compiladores.
Visual Studio es la herramienta de desarrollo que admitimos y recomendamos para C++/WinRT. Consulta Compatibilidad de Visual Studio para C++/WinRT.
¿Por qué no tiene la función de implementación generada para una propiedad de solo lectura el calificador const?
Cuando se declara una propiedad de solo lectura en MIDL 3.0, puedes esperar que la herramienta cppwinrt.exe
te genere una función de implementación que es const
-qualified (una función const trata el puntero this como const).
Sin duda, recomendamos usar const siempre que sea posible, pero la propia herramienta cppwinrt.exe
no intenta razonar sobre qué funciones de implementación podrían ser const y cuáles no. Puedes elegir que cualquiera de las funciones de implementación sea const, como en este ejemplo.
struct MyStringable : winrt::implements<MyStringable, winrt::Windows::Foundation::IStringable>
{
winrt::hstring ToString() const
{
return L"MyStringable";
}
};
Puedes eliminar ese calificador const
en ToString si decides que necesitas modificar algún estado de objeto en su implementación. Pero haz que cada una de las funciones de miembros sean const o non-const, y no ambos. En otras palabras, no sobrecargues una función de implementación en const
.
Aparte de las funciones de implementación, otro lugar donde const aparece en la imagen es en las proyecciones de función de Windows Runtime. Observa este código.
int main()
{
winrt::Windows::Foundation::IStringable s{ winrt::make<MyStringable>() };
auto result{ s.ToString() };
}
Para la llamada a ToString anterior, el comando Ir a declaración en Visual Studio muestra que la proyección de Windows Runtime IStringable::ToString en C++/WinRT tiene este aspecto.
winrt::hstring ToString() const;
Las funciones de la proyección son const independientemente de cómo decidas calificar su implementación. En segundo plano, la proyección llama a la interfaz binaria de aplicación (ABI), lo que equivale a una llamada mediante un puntero de interfaz COM. El único estado con el que el objeto ToString previsto interactúa es ese puntero de interfaz COM y, ciertamente, no tiene necesidad de modificar ese puntero, por lo que la función es const. Esto te ofrece la garantía de que no cambiará nada sobre la referencia IStringable a la que estás llamando y garantiza que puedas llamar a ToString, incluso con una referencia const a IStringable.
Entiende que estos ejemplos de const
son detalles de implementación de proyecciones e implementaciones de C++/WinRT; constituyen la higiene de código para ayudarte. No existe nada como const
en COM ni en ABI de Windows Runtime (para funciones de miembros).
¿Tienes recomendaciones para reducir el tamaño del código para los archivos binarios de C++/WinRT?
Cuando se trabaja con objetos de Windows Runtime, debes evitar el patrón de codificación que se muestra a continuación, ya que puede tener un impacto negativo en la aplicación al generar más código binario de lo necesario.
anobject.b().c().d();
anobject.b().c().e();
anobject.b().c().f();
En el mundo de Windows Runtime, el compilador no puede almacenar en caché el valor de c()
o las interfaces para cada método al que se llama mediante un direccionamiento indirecto ('.'). A menos de que intervengas, esto da como resultado varias llamadas virtuales y sobrecarga de recuento de referencias. El patrón anterior podría generar fácilmente el doble de código de lo estrictamente necesario. En su lugar, es preferible que utilices el patrón que se muestra a continuación siempre que puedas. Genera mucho menos código y puede mejorar drásticamente el rendimiento del tiempo de ejecución.
auto a{ anobject.b().c() };
a.d();
a.e();
a.f();
El patrón recomendado mostrado anteriormente no solo se aplica a C++/WinRT, sino a todas las proyecciones de lenguaje de Windows Runtime.
¿Cómo convertir una cadena en un tipo (para la navegación, por ejemplo)?
Al final del ejemplo de código en la vista de navegación (que está principalmente en C#), hay un fragmento de código de C++/WinRT que muestra cómo hacerlo.
¿Cómo se pueden resolver las ambigüedades con GetCurrentTime o TRY?
El archivo de encabezado winrt/Windows.UI.Xaml.Media.Animation.h
declara un método denominado GetCurrentTime, mientras que windows.h
(mediante winbase.h
) define una macro denominada GetCurrentTime. Cuando los dos entran en conflicto, el compilador de C++ genera el error "error C4002: demasiados argumentos para la invocación de la macro como función GetCurrentTime".
De forma similar, winrt/Windows.Globalization.h
declara un método denominado TRY, mientras que afx.h
define una macro denominada TRY. Cuando estos entran en conflicto, el compilador de C++ genera el error "error C2334: tokens inesperados antes de '{'; se omite el cuerpo de función aparente".
Para solucionar uno o ambos problemas, puedes hacer lo siguiente.
#pragma push_macro("GetCurrentTime")
#pragma push_macro("TRY")
#undef GetCurrentTime
#undef TRY
#include <winrt/include_your_cppwinrt_headers_here.h>
#include <winrt/include_your_cppwinrt_headers_here.h>
#pragma pop_macro("TRY")
#pragma pop_macro("GetCurrentTime")
¿Cómo puedo acelerar la carga de símbolos?
En Visual Studio, Herramientas>Opciones>Depuración>Símbolos>, marque Load only specified modules (Cargar solo los módulos especificados). Después, puede hacer clic con el botón derecho en los DLL de la lista de pilas y cargar módulos individuales.
Nota
Si en este tema no hemos respondido a tu pregunta, podrás encontrar ayuda en la comunidad de desarrolladores de C++ de Visual Studio o mediante la etiqueta c++-winrt
en Stack Overflow.