Paketera och distribuera resurser i .NET-appar
Program förlitar sig på .NET Framework Resource Manager, som representeras av ResourceManager klassen, för att hämta lokaliserade resurser. Resource Manager förutsätter att en hubb- och ekermodell används för att paketera och distribuera resurser. Hubben är huvudsammansättningen som innehåller den icke-lokaliserade körbara koden och resurserna för en enda kultur, som kallas neutral eller standardkultur. Standardkulturen är återställningskulturen för programmet. det är den kultur vars resurser används om lokaliserade resurser inte kan hittas. Varje eker ansluter till en satellitsammansättning som innehåller resurser för en enda kultur, men som inte innehåller någon kod.
Det finns flera fördelar med den här modellen:
- Du kan stegvis lägga till resurser för nya kulturer när du har distribuerat ett program. Eftersom efterföljande utveckling av kulturspecifika resurser kan kräva en betydande tid kan du släppa huvudprogrammet först och leverera kulturspecifika resurser vid ett senare tillfälle.
- Du kan uppdatera och ändra ett programs satellitsammansättningar utan att kompilera om programmet.
- Ett program behöver bara läsa in de satellitsammansättningar som innehåller de resurser som behövs för en viss kultur. Detta kan avsevärt minska användningen av systemresurser.
Det finns dock även nackdelar med den här modellen:
- Du måste hantera flera uppsättningar resurser.
- Den initiala kostnaden för att testa ett program ökar eftersom du måste testa flera konfigurationer. Observera att det på lång sikt blir enklare och billigare att testa ett kärnprogram med flera satelliter än att testa och underhålla flera parallella internationella versioner.
Namngivningskonventioner för resurser
När du paketera programmets resurser måste du namnge dem med hjälp av de namngivningskonventioner för resurser som den vanliga språkkörningen förväntar sig. Körningen identifierar en resurs med dess kulturnamn. Varje kultur får ett unikt namn, vilket vanligtvis är en kombination av ett kulturnamn med två bokstäver som är associerat med ett språk och, om det behövs, ett subkulturnamn med två bokstäver som är associerat med ett land eller en region. Subkulturnamnet följer kulturnamnet, avgränsat med ett bindestreck (-). Exempel är ja-JP för japanska som talas i Japan, en-US för engelska som talas i USA, de-DE för tyska som talas i Tyskland, eller de-AT för tyska som talas i Österrike. Se kolumnen Språktagg i listan över språk-/regionnamn som stöds av Windows. Kulturnamn följer standarden som definieras av BCP 47.
Kommentar
Det finns vissa undantag för kulturnamn med två bokstäver, till exempel zh-Hans
för kinesiska (förenklad).
Mer information finns i Skapa resursfiler och Skapa satellitsammansättningar.
Återställningsprocessen för resursen
Hubb- och ekermodellen för paketering och distribution av resurser använder en återställningsprocess för att hitta lämpliga resurser. Om ett program begär en lokaliserad resurs som inte är tillgänglig söker den vanliga språkkörningen i kulturhierarkin efter en lämplig återställningsresurs som bäst matchar användarens programbegäran och genererar endast ett undantag som en sista utväg. Om en lämplig resurs hittas på varje nivå i hierarkin använder körningen den. Om resursen inte hittas fortsätter sökningen på nästa nivå.
För att förbättra uppslagsprestandan använder NeutralResourcesLanguageAttribute du attributet för huvudsammansättningen och skickar det namnet på det neutrala språk som fungerar med huvudsammansättningen.
Återställningsprocess för .NET Framework-resurser
Återställningsprocessen för .NET Framework-resurser omfattar följande steg:
Dricks
Du kanske kan använda konfigurationselementet <relativeBindForResources> för att optimera återställningsprocessen för resursen och processen med vilken körningen avsöker för resurssammansättningar. Mer information finns i Optimera återställningsprocessen för resurser.
Körningen kontrollerar först den globala sammansättningscachen för en sammansättning som matchar den begärda kulturen för ditt program.
Den globala sammansättningscacheminnet kan lagra resurssammansättningar som delas av många program. Detta gör att du inte behöver inkludera specifika uppsättningar resurser i katalogstrukturen för varje program som du skapar. Om körningen hittar en referens till sammansättningen söker den i sammansättningen efter den begärda resursen. Om den hittar posten i sammansättningen använder den begärda resursen. Om posten inte hittas fortsätter sökningen.
Körningen kontrollerar sedan katalogen för den pågående sammansättningen efter en underkatalog som matchar den begärda kulturen. Om den hittar underkatalogen söker den i underkatalogen efter en giltig satellitsammansättning för den begärda kulturen. Körningen söker sedan i satellitsammansättningen efter den begärda resursen. Om den hittar resursen i sammansättningen använder den den. Om den inte hittar resursen fortsätter sökningen.
Körningen frågar sedan Windows Installer för att avgöra om satellitsammansättningen ska installeras på begäran. I så fall hanterar den installationen, läser in sammansättningen och söker efter den eller den begärda resursen. Om den hittar resursen i sammansättningen använder den den. Om den inte hittar resursen fortsätter sökningen.
Körningen genererar AppDomain.AssemblyResolve händelsen för att indikera att den inte kan hitta satellitsammansättningen. Om du väljer att hantera händelsen kan händelsehanteraren returnera en referens till satellitsammansättningen vars resurser ska användas för sökningen. Annars returnerar
null
händelsehanteraren och sökningen fortsätter.Körningen söker sedan i den globala sammansättningscacheminnet igen, den här gången efter den överordnade sammansättningen av den begärda kulturen. Om den överordnade sammansättningen finns i den globala sammansättningscachen söker körningen efter den begärda resursen i sammansättningen.
Den överordnade kulturen definieras som lämplig reservkultur. Överväg föräldrar som reservkandidater, eftersom det är bättre att tillhandahålla en resurs än att kasta ett undantag. Med den här processen kan du också återanvända resurser. Du bör endast ta med en viss resurs på den överordnade nivån om den underordnade kulturen inte behöver lokalisera den begärda resursen. Om du till exempel tillhandahåller satellitsammansättningar för
en
(neutral engelska),en-GB
(engelska som talas i Storbritannien) ochen-US
(engelska som det talas i USA), skulle satellitenen
innehålla den gemensamma terminologin, ochen-GB
satelliterna ochen-US
kunde ge åsidosättningar endast för de termer som skiljer sig åt.Körningen kontrollerar sedan katalogen för den pågående sammansättningen för att se om den innehåller en överordnad katalog. Om det finns en överordnad katalog söker körningen i katalogen efter en giltig satellitsammansättning efter den överordnade kulturen. Om den hittar sammansättningen söker körningen efter den begärda resursen i sammansättningen. Om den hittar resursen använder den den. Om den inte hittar resursen fortsätter sökningen.
Körningen frågar sedan Windows Installer för att avgöra om den överordnade satellitsammansättningen ska installeras på begäran. I så fall hanterar den installationen, läser in sammansättningen och söker efter den eller den begärda resursen. Om den hittar resursen i sammansättningen använder den den. Om den inte hittar resursen fortsätter sökningen.
Körningen genererar AppDomain.AssemblyResolve händelsen för att indikera att den inte kan hitta en lämplig återställningsresurs. Om du väljer att hantera händelsen kan händelsehanteraren returnera en referens till satellitsammansättningen vars resurser ska användas för sökningen. Annars returnerar
null
händelsehanteraren och sökningen fortsätter.Körningen söker sedan efter överordnade sammansättningar, som i de föregående tre stegen, genom många potentiella nivåer. Varje kultur har bara en överordnad, som definieras av egenskapen, men en överordnad kan ha en egen överordnad CultureInfo.Parent . Sökningen efter överordnade kulturer stoppas när en kulturs Parent egenskap returnerar CultureInfo.InvariantCulture. För resursåterställning betraktas den invarianta kulturen inte som en överordnad kultur eller en kultur som kan ha resurser.
Om den kultur som ursprungligen angavs och alla föräldrar har genomsökts och resursen fortfarande inte hittas används resursen för standardkulturen (återställning). Vanligtvis ingår resurserna för standardkulturen i huvudprogramsammansättningen. Du kan dock ange värdet Satellite för egenskapen NeutralResourcesLanguageAttribute för attributet för Location att indikera att den ultimata återställningsplatsen för resurser är en satellitsammansättning i stället för huvudsammansättningen.
Kommentar
Standardresursen är den enda resurs som kan kompileras med huvudsammansättningen. Om du inte anger en satellitsammansättning med hjälp NeutralResourcesLanguageAttribute av attributet är det den ultimata återställningen (slutlig överordnad). Därför rekommenderar vi att du alltid inkluderar en standarduppsättning med resurser i huvudsammansättningen. Detta förhindrar att undantag utlöses. Genom att inkludera en standardresurs anger du en reserv för alla resurser och ser till att minst en resurs alltid finns för användaren, även om den inte är kulturellt specifik.
Om körningen inte hittar någon resurs för en standardkultur (återställning) genereras slutligen ett eller MissingSatelliteAssemblyException ett MissingManifestResourceException undantag för att indikera att resursen inte kunde hittas.
Anta till exempel att programmet begär en resurs lokaliserad för spanska (Mexiko) ( es-MX
kulturen). Körningen söker först i den globala sammansättningscachen efter den sammansättning som matchar es-MX
, men som inte hittar den. Körningen söker sedan i katalogen för den pågående sammansättningen efter en es-MX
katalog. Annars söker körningen i den globala sammansättningscachen igen efter en överordnad sammansättning som återspeglar lämplig återställningskultur – i det här fallet es
(spanska). Om den överordnade sammansättningen inte hittas söker körningen igenom alla potentiella nivåer av överordnade sammansättningar för es-MX
kulturen tills den hittar en motsvarande resurs. Om en resurs inte hittas använder körningen resursen för standardkulturen.
Optimera återställningsprocessen för .NET Framework-resurser
Under följande villkor kan du optimera den process genom vilken körningen söker efter resurser i satellitsammansättningar:
Satellitsammansättningar distribueras på samma plats som kodsammansättningen. Om kodsammansättningen installeras i global sammansättningscache installeras även satellitsammansättningar i den globala sammansättningscachen. Om kodsammansättningen installeras i en katalog installeras satellitsammansättningar i kulturspecifika mappar i katalogen.
Satellitsammansättningar installeras inte på begäran.
Programkoden hanterar inte händelsen AppDomain.AssemblyResolve .
Du optimerar avsökningen för satellitsammansättningar genom att inkludera elementet <relativeBindForResources> och ange dess enabled
attribut i true
programkonfigurationsfilen, som du ser i följande exempel.
<configuration>
<runtime>
<relativeBindForResources enabled="true" />
</runtime>
</configuration>
Den optimerade avsökningen för satellitsammansättningar är en opt-in-funktion. Det vill: Körningen följer de steg som beskrivs i Återställningsprocessen för resurser om inte elementet <relativeBindForResources> finns i programmets konfigurationsfil och dess enabled
attribut är inställt på true
. I så fall ändras avsökningsprocessen för en satellitsammansättning på följande sätt:
Körningen använder platsen för den överordnade kodsammansättningen för att avsöka satellitsammansättningen. Om den överordnade sammansättningen är installerad i den globala sammansättningscachen avsöker körningen i cacheminnet men inte i programmets katalog. Om den överordnade sammansättningen är installerad i en programkatalog avsöker runtime-avsökningarna i programkatalogen men inte i den globala sammansättningscacheminnet.
Körningen frågar inte Windows Installer om installation på begäran av satellitsammansättningar.
Om avsökningen för en viss resurssammansättning misslyckas genererar inte körningen AppDomain.AssemblyResolve händelsen.
Återställningsprocess för .NET Core-resurser
Återställningsprocessen för .NET Core-resurser omfattar följande steg:
Körningen försöker läsa in en satellitsammansättning för den begärda kulturen.
Kontrollerar katalogen för den pågående sammansättningen efter en underkatalog som matchar den begärda kulturen. Om den hittar underkatalogen söker den i underkatalogen efter en giltig satellitsammansättning för den begärda kulturen och läser in den.
Kommentar
På operativsystem med skiftlägeskänsliga filsystem (det vill säga Linux och macOS) är underkatalogsökningen för kulturnamn skiftlägeskänslig. Underkatalognamnet måste exakt matcha fallet CultureInfo.Name med (till exempel
es
elleres-MX
).Kommentar
Om programmeraren har härlett en anpassad sammansättningsbelastningskontext från AssemblyLoadContextär situationen komplicerad. Om den körande sammansättningen lästes in i den anpassade kontexten läser körningen in satellitsammansättningen i den anpassade kontexten. Informationen ligger utanför omfånget för det här dokumentet. Se AssemblyLoadContext.
Om en satellitmontering inte har hittats, AssemblyLoadContext genererar AssemblyLoadContext.Resolving händelsen för att indikera att den inte kan hitta satellitsammansättningen. Om du väljer att hantera händelsen kan händelsehanteraren läsa in och returnera en referens till satellitsammansättningen.
Om en satellitsammansättning fortfarande inte har hittats gör AssemblyLoadContext att AppDomain utlöser en AppDomain.AssemblyResolve händelse för att indikera att den inte kan hitta satellitsammansättningen. Om du väljer att hantera händelsen kan händelsehanteraren läsa in och returnera en referens till satellitsammansättningen.
Om en satellitsammansättning hittas söker körningen efter den begärda resursen. Om den hittar resursen i sammansättningen använder den den. Om den inte hittar resursen fortsätter sökningen.
Kommentar
För att hitta en resurs i satellitsammansättningen söker körningen efter resursfilen som begärs av ResourceManager för den aktuella CultureInfo.Name. I resursfilen söker den efter det begärda resursnamnet. Om någon av dem inte hittas behandlas resursen som inte hittad.
Körningen söker sedan igenom de överordnade kultursammansättningarna genom många potentiella nivåer, varje gång upprepa steg 1 och 2.
Den överordnade kulturen definieras som en lämplig reservkultur. Överväg föräldrar som reservkandidater, eftersom det är bättre att tillhandahålla en resurs än att kasta ett undantag. Med den här processen kan du också återanvända resurser. Du bör endast ta med en viss resurs på den överordnade nivån om den underordnade kulturen inte behöver lokalisera den begärda resursen. Om du till exempel tillhandahåller satellitsammansättningar för
en
(neutral engelska),en-GB
(engelska som talas i Storbritannien) ochen-US
(engelska som det talas i USA), innehåller satellitenen
den gemensamma terminologin, ochen-GB
satelliterna ochen-US
tillhandahåller åsidosättningar för endast de termer som skiljer sig åt.Varje kultur har bara en överordnad, som definieras av egenskapen, men en överordnad kan ha en egen överordnad CultureInfo.Parent . Sökningen efter överordnade kulturer stoppas när en kulturs Parent egenskap returnerar CultureInfo.InvariantCulture. För resursåterställning betraktas den invarianta kulturen inte som en överordnad kultur eller en kultur som kan ha resurser.
Om den kultur som ursprungligen angavs och alla föräldrar har genomsökts och resursen fortfarande inte hittas används resursen för standardkulturen (återställning). Vanligtvis ingår resurserna för standardkulturen i huvudprogramsammansättningen. Du kan dock ange värdet Satellite för egenskapen för Location att ange att den ultimata återställningsplatsen för resurser är en satellitsammansättning snarare än huvudsammansättningen.
Kommentar
Standardresursen är den enda resurs som kan kompileras med huvudsammansättningen. Om du inte anger en satellitsammansättning med hjälp NeutralResourcesLanguageAttribute av attributet är det den ultimata återställningen (slutlig överordnad). Därför rekommenderar vi att du alltid inkluderar en standarduppsättning med resurser i huvudsammansättningen. Detta förhindrar att undantag utlöses. Genom att inkludera en standardresursfil anger du en reserv för alla resurser och ser till att minst en resurs alltid finns för användaren, även om den inte är kulturellt specifik.
Slutligen, om körningen inte hittar en resursfil för en standardkultur (återställning) genereras ett eller MissingSatelliteAssemblyException ett MissingManifestResourceException undantag för att indikera att resursen inte kunde hittas. Om resursfilen hittas men den begärda resursen inte finns returnerar
null
begäran .
Ultimat reserv till satellitmontering
Du kan också ta bort resurser från huvudsammansättningen och ange att körningen ska läsa in de ultimata reservresurserna från en satellitsammansättning som motsvarar en specifik kultur. För att styra återställningsprocessen använder NeutralResourcesLanguageAttribute(String, UltimateResourceFallbackLocation) du konstruktorn och anger ett värde för parametern UltimateResourceFallbackLocation som anger om Resource Manager ska extrahera reservresurserna från huvudsammansättningen eller från en satellitsammansättning.
I följande .NET Framework-exempel används NeutralResourcesLanguageAttribute attributet för att lagra ett programs återställningsresurser i en satellitsammansättning för det franska språket (fr
). Exemplet har två textbaserade resursfiler som definierar en enskild strängresurs med namnet Greeting
. Den första, resources.fr.txt, innehåller en fransk språkresurs.
Greeting=Bon jour!
Den andra, resurser,ru.txt, innehåller en rysk språkresurs.
Greeting=Добрый день
Dessa två filer kompileras till .resources-filer genom att köra Resource File Generator (resgen.exe) från kommandoraden. För den franska språkresursen är kommandot:
resgen.exe resources.fr.txt
För den ryska språkresursen är kommandot:
resgen.exe resources.ru.txt
.resources-filerna bäddas in i dynamiska länkbibliotek genom att köra Assembly Linker (al.exe) från kommandoraden för den franska språkresursen enligt följande:
al /t:lib /embed:resources.fr.resources /culture:fr /out:fr\Example1.resources.dll
och för den ryska språkresursen enligt följande:
al /t:lib /embed:resources.ru.resources /culture:ru /out:ru\Example1.resources.dll
Programmets källkod finns i en fil med namnet Example1.cs eller Example1.vb. Den innehåller NeutralResourcesLanguageAttribute attributet för att indikera att standardprogramresursen finns i underkatalogen fr. Den instansierar Resource Manager, hämtar värdet för resursen Greeting
och visar den för konsolen.
using System;
using System.Reflection;
using System.Resources;
[assembly:NeutralResourcesLanguage("fr", UltimateResourceFallbackLocation.Satellite)]
public class Example
{
public static void Main()
{
ResourceManager rm = new ResourceManager("resources",
typeof(Example).Assembly);
string greeting = rm.GetString("Greeting");
Console.WriteLine(greeting);
}
}
Imports System.Reflection
Imports System.Resources
<Assembly: NeutralResourcesLanguage("fr", UltimateResourceFallbackLocation.Satellite)>
Module Example
Public Sub Main()
Dim rm As New ResourceManager("resources", GetType(Example).Assembly)
Dim greeting As String = rm.GetString("Greeting")
Console.WriteLine(greeting)
End Sub
End Module
Du kan sedan kompilera C#-källkoden från kommandoraden på följande sätt:
csc Example1.cs
Kommandot för Visual Basic-kompilatorn är mycket likt:
vbc Example1.vb
Eftersom det inte finns några resurser inbäddade i huvudsammansättningen behöver du inte kompilera med hjälp av växeln /resource
.
När du kör exemplet från ett system vars språk är något annat än ryska visas följande utdata:
Bon jour!
Förslag på paketeringsalternativ
Tids- eller budgetbegränsningar kan hindra dig från att skapa en uppsättning resurser för varje underkultur som programmet stöder. I stället kan du skapa en enda satellitsammansättning för en överordnad kultur som alla relaterade subkulturer kan använda. Du kan till exempel ange en enda engelsk satellitsammansättning (en) som hämtas av användare som begär regionspecifika engelska resurser och en enda tysk satellitsammansättning (de) för användare som begär regionspecifika tyska resurser. Till exempel skulle begäranden om tyska som talas i Tyskland (de-DE), Österrike (de-AT) och Schweiz (de-CH) återgå till den tyska satellitförsamlingen (de). Standardresurserna är den slutliga återställningen och bör därför vara de resurser som begärs av de flesta av programmets användare, så välj dessa resurser noggrant. Den här metoden distribuerar resurser som är mindre kulturellt specifika, men som avsevärt kan minska programmets lokaliseringskostnader.