Koncepcja: funkcje
Funkcje
Funkcje reprezentują zestawy funkcji, zachowania i zależności, które można selektywnie dodać do pakietu lub projektu podczas instalacji.
Zgodnie z projektem funkcje powinny być zgodne z następującymi zasadami:
- Addytywne: włączenie funkcji powinno spowodować brak nowej funkcjonalności w przeciwnym razie w pakiecie bez wyłączania innych funkcji.
- Nie wykluczające się: włączenie funkcji nie powinno uniemożliwiać instalowania innych funkcji.
Funkcje nie powinny być używane do definiowania alternatywnych zestawów funkcji. Na przykład biblioteka grafiki nie powinna używać funkcji do wyboru między ekskluzywnymi zapleczami grafiki, ponieważ nie jest możliwe zainstalowanie wszystkich z nich w tym samym czasie.
Funkcje mogą mieć następujący wpływ na zależności pakietu:
- Dodaj nowe zależności, w tym zależności od innych funkcji z tego samego pakietu.
- Włącz nowe funkcje na istniejących zależnościach.
Zestaw dostępnych funkcji jest definiowany "features"
przez pole .
Przykład 1. Wiele formatów plików
Biblioteka manipulowania obrazami może na przykład obsługiwać kilka różnych typów obrazów w zależności od różnych zestawów innych bibliotek.
{
"name": "my-image-lib",
"version": "0.1",
"features": {
"png": { "description": "Support PNG files", "dependencies": ["libpng"]},
"jpeg": { "description": "Support JPEG files", "dependencies": ["libjpeg-turbo"]},
"tiff": { "description": "Support TIFF files", "dependencies": ["libtiff"]},
}
}
Funkcje domyślne
Funkcje domyślne to zestaw funkcji do automatycznego aktywowania, jeśli projekt najwyższego poziomu nie zażąda jawnie kompilacji bez nich. Funkcje domyślne mają zapewnić minimalny poziom funkcjonalności niezależnie od tego, jak złożony i dostosowywalny jest wykres zależności projektu.
Uwaga
Funkcje domyślne nie są przeznaczone do modelowania "curation" ani "sugestii".
Rozważmy na przykład bibliotekę "extract-any"
, która obsługuje ponad 10 różnych formatów archiwum, w tym kilka, które są dość niejasne. Ponieważ wszystkie są opcjonalne, jeśli żadna z nich nie jest wybrana, biblioteka nie jest funkcjonalna: nie może wyodrębnić żadnych plików.
Funkcje domyślne zapewniają, że użytkownik, który po prostu dodaje "extract-any"
się do listy zależności w swoich vcpkg.json
aplikacjach, uzyska poziom podstawowej funkcjonalności, na przykład automatycznie wybierając .zip
i .tar.gz
dekompresory.
Przykład 2. Domyślne funkcje w działaniu
Gdy użytkownik dodaje "extract-any"
się do nich vcpkg.json
bez określania funkcji, funkcje domyślne (np. obsługa .zip
i .tar.gz
formaty) są automatycznie uwzględniane, zapewniając podstawowe funkcje.
{
"name": "my-application",
"version": "0.15.2",
"dependencies": [
"extract-any"
]
}
Jeśli użytkownik chce jawnie wyłączyć funkcje domyślne, może to zrobić, dodając "default-features": false
do zależności:
{
"name": "my-application",
"version": "0.15.2",
"dependencies": [
{
"name": "extract-any",
"default-features": false
}
]
}
Alternatywnie, jeśli używasz narzędzia vcpkg w trybie klasycznym, możesz wyłączyć funkcje domyślne za pośrednictwem core
funkcji. Na przykład vcpkg install extract-any[core]
instaluje extract-any
się bez żadnych domyślnych funkcji, ponieważ [core]
jawnie je wyklucza.
Aby uzyskać więcej informacji, zapoznaj się z artykułem dotyczącym funkcji domyślnych.
Rozwiązywanie zależności
W przypadku korzystania z narzędzia vcpkg rozpoznawanie zależności odgrywa kluczową rolę, zwłaszcza w przypadku pracy z funkcjami, które mają współzależności. Aby zilustrować, rozważmy następujący scenariusz obejmujący bibliotekę manipulowania obrazami:
{
"name": "my-image-lib",
"version": "0.1",
"features": {
"png": { "description": "Support PNG files", "dependencies": ["libpng"]},
"jpeg": { "description": "Support JPEG files", "dependencies": ["libjpeg-turbo"]},
"tiff": { "description": "Support TIFF files", "dependencies": ["libtiff"]},
}
}
W scenariuszach, w których różne biblioteki zależą od różnych funkcji wspólnej biblioteki, program vcpkg zapewnia, że wszystkie wymagane funkcje i zależności są brane pod uwagę. Jeśli na przykład library-a
wymaga png
funkcji i library-b
wymaga jpeg
funkcji z my-image-lib
programu , graf zależności będzie wyglądać następująco:
{
"name": "library-a",
"version": "1",
"dependencies": [{"name": "my-image-lib", "features": ["png"]}]
}
{
"name": "library-b",
"version": "1",
"dependencies": [{"name": "my-image-lib", "features": ["jpeg"]}]
}
{
"name": "project-using-a-and-b",
"version": "1",
"dependencies": [
"library-a",
"library-b"
]
}
Po rozwiązaniu tych zależności program vcpkg łączy wszystkie niezbędne funkcje i zależności w celu utworzenia kompleksowego planu instalacji. W tym przykładzie projekt jest zależny od library-a
library-b
planu instalacji, który obejmuje zarówno PNG
program , jak i JPEG
obsługę programu my-image-lib
, ale nie TIFF
:
libjpeg-turbo[core]
libpng[core]
library-a[core]
library-b[core]
my-image-lib[core,png,jpeg]
Ten mechanizm zapewnia, że kompilacja my-image-lib
programu jest zoptymalizowana pod kątem wymaganych funkcji, zapewniając obsługę PNG
i JPEG
jednocześnie wykluczając niepotrzebne TIFF
wsparcie.
Użycie zaawansowane
Przykład 3: wiele powiązanych projektów w jednym repozytorium
Gdy pojedyncze repozytorium zawiera kilka oddzielnych składników do kompilacji, takich jak aplikacje klienckie i serwerowe z kodem udostępnionym, deweloperzy każdego elementu mogą chcieć uniknąć instalowania kosztownych zależności wymaganych przez inne elementy.
{
"name": "my-game",
"dependencies": ["grpc"],
"features": {
"client": { "description": "Client Game Executable", "dependencies": ["sdl2", "bullet3"]},
"server": { "description": "Multiplayer Server Executable", "dependencies": ["proxygen"]},
"tests": { "description": "Build tests", "dependencies": ["gtest"] }
}
}
Następnie indywidualni deweloperzy mogą wybrać funkcje do zainstalowania:
- W wierszu polecenia instalacji przekaż
--x-feature=
- W przypadku korzystania z integracji narzędzia CMake ustaw
VCPKG_MANIFEST_FEATURES
przed poleceniemproject()
. - W przypadku korzystania z integracji z programem MSBuild przekaż za
--x-feature=
pośrednictwemVcpkgAdditionalInstallOptions
Aby uzyskać więcej informacji, zobacz następujące zasoby: