Porady: kierowanie tablic za pomocą międzyoperacyjności języka C++
W tym temacie przedstawiono jeden aspekt współdziałania języka Visual C++. Aby uzyskać więcej informacji, zobacz Using C++ Interop (Implicit PInvoke) (Używanie międzyoperacyjności języka C++ (niejawna funkcja PInvoke).
Poniższe przykłady kodu używają zarządzanych, niezarządzanych dyrektyw #pragma do implementowania funkcji zarządzanych i niezarządzanych w tym samym pliku, ale te funkcje współdziałają w taki sam sposób, jeśli są zdefiniowane w oddzielnych plikach. Pliki zawierające tylko funkcje niezarządzane nie muszą być kompilowane za pomocą polecenia /clr (kompilacja środowiska uruchomieniowego języka wspólnego).
Przykład: Przekazywanie tablicy zarządzanej do funkcji niezarządzanej
W poniższym przykładzie pokazano, jak przekazać tablicę zarządzaną do funkcji niezarządzanej. Funkcja zarządzana używa pin_ptr (C++/CLI) do pomijania odzyskiwania pamięci dla tablicy przed wywołaniem funkcji niezarządzanej. Dzięki zapewnieniu niezarządzanej funkcji z przypiętym wskaźnikiem do sterty GC, można uniknąć narzutu na utworzenie kopii tablicy. Aby zademonstrować, że funkcja niezarządzana uzyskuje dostęp do pamięci stert gC, modyfikuje zawartość tablicy, a zmiany są odzwierciedlane po wznowieniu kontroli przez funkcję zarządzaną.
// PassArray1.cpp
// compile with: /clr
#ifndef _CRT_RAND_S
#define _CRT_RAND_S
#endif
#include <iostream>
#include <stdlib.h>
using namespace std;
using namespace System;
#pragma unmanaged
void TakesAnArray(int* a, int c) {
cout << "(unmanaged) array received:\n";
for (int i=0; i<c; i++)
cout << "a[" << i << "] = " << a[i] << "\n";
unsigned int number;
errno_t err;
cout << "(unmanaged) modifying array contents...\n";
for (int i=0; i<c; i++) {
err = rand_s( &number );
if ( err == 0 )
a[i] = number % 100;
}
}
#pragma managed
int main() {
array<int>^ nums = gcnew array<int>(5);
nums[0] = 0;
nums[1] = 1;
nums[2] = 2;
nums[3] = 3;
nums[4] = 4;
Console::WriteLine("(managed) array created:");
for (int i=0; i<5; i++)
Console::WriteLine("a[{0}] = {1}", i, nums[i]);
pin_ptr<int> pp = &nums[0];
TakesAnArray(pp, 5);
Console::WriteLine("(managed) contents:");
for (int i=0; i<5; i++)
Console::WriteLine("a[{0}] = {1}", i, nums[i]);
}
Przykład: Przekazywanie tablicy niezarządzanej do funkcji zarządzanej
W poniższym przykładzie pokazano przekazywanie tablicy niezarządzanej do funkcji zarządzanej. Funkcja zarządzana uzyskuje bezpośredni dostęp do pamięci tablicy (w przeciwieństwie do tworzenia tablicy zarządzanej i kopiowania zawartości tablicy), co pozwala na odzwierciedlenie zmian wprowadzonych przez funkcję zarządzaną w funkcji niezarządzanej po odzyskaniu kontroli.
// PassArray2.cpp
// compile with: /clr
#include <iostream>
using namespace std;
using namespace System;
#pragma managed
void ManagedTakesAnArray(int* a, int c) {
Console::WriteLine("(managed) array received:");
for (int i=0; i<c; i++)
Console::WriteLine("a[{0}] = {1}", i, a[i]);
cout << "(managed) modifying array contents...\n";
Random^ r = gcnew Random(DateTime::Now.Second);
for (int i=0; i<c; i++)
a[i] = r->Next(100);
}
#pragma unmanaged
void NativeFunc() {
int nums[5] = { 0, 1, 2, 3, 4 };
printf_s("(unmanaged) array created:\n");
for (int i=0; i<5; i++)
printf_s("a[%d] = %d\n", i, nums[i]);
ManagedTakesAnArray(nums, 5);
printf_s("(ummanaged) contents:\n");
for (int i=0; i<5; i++)
printf_s("a[%d] = %d\n", i, nums[i]);
}
#pragma managed
int main() {
NativeFunc();
}
Zobacz też
Korzystanie z międzyoperacyjności języka C++ (niejawna funkcja PInvoke)