asp.net core 之多語言國際化自定義資源文件

来源:http://www.cnblogs.com/passingwind/archive/2017/05/21/aspnet-core-localization-and-custom-resource-service-with-file.html
-Advertisement-
Play Games

先說說 asp.net core 預設的多語言和國際化。 官方文檔 一:基本使用方法 先要安裝 包 Microsoft.AspNetCore.Mvc.Localization (依賴 Microsoft.Extensions.Localization) 然後使用 資源文件保存不同的語言對應的數據。 ...


先說說 asp.net core 預設的多語言和國際化。 官方文檔

一:基本使用方法

先要安裝 包 Microsoft.AspNetCore.Mvc.Localization (依賴 Microsoft.Extensions.Localization)  然後使用 資源文件保存不同的語言對應的數據。

1,在視圖頁面註入 IViewLocalizer ,然後在需要的地方使用即可。 比如:

1 @inject IViewLocalizer Localizer
2 
3 <h2>@Localizer["hello"]</h2>

 

其中 中括弧中的字元 即是資源文件中的名稱, 運行後,輸出的即是 當前語言對應的資源文件下的設置的資源值。

那麼有個問題來了,資源文件怎麼設置?

1,預設情況下會去查找 設置的 LocalizationOptions.ResourcesPath  的值對應的文件夾,如果沒有設置,則去根目錄下查找。

在 Startup 中設置 ResourcesPath  。

services.AddLocalization(options => options.ResourcesPath = "Resources");

2,查找當前視圖文件對應的同名資源文件。 預設支持 使用 點 . 和路徑 path 查找兩種方式,當然也可以指定其中一個方式。 比如 當前視圖路徑是 views/account/login.cshtml ,那麼 查找的資源文件是  views/account/login.{CultureName}.resx 文件和 views.account.login.{CultureName}.resx 文件 

services.AddMvc()
         .AddViewLocalization()
          //.AddViewLocalization(Microsoft.AspNetCore.Mvc.Razor.LanguageViewLocationExpanderFormat.SubFolder)
          .AddDataAnnotationsLocalization();

3,如果是 model 類, 查找的路徑則變成了model 類對應的命名空間即typeof(model).FullName 全路徑。比如 ViewModels/account/login.{CultureName}.resx 文件和 ViewModels.account.login.{CultureName}.resx 文件 。同理 如果是在controller  那麼,資源文件 則是  Controllers.HomeController.{CultureName}.resx 或者 Controllers/HomeController.{CultureName}.resx


二:解析

那麼這個是如何實現的呢?如果我想使用 資料庫或者是 json 文件來存在這些資源文件。

在試圖文件中 註入的是 IViewLocalizer 介面,對應的實現是  ViewLocalizer 。ViewLocalizer 實現了IViewLocalizer 和IHtmlLocalizer 的定義,並且 IViewLocalizer 繼承自IHtmlLocalizer。  ViewLocalizer 會註入一個IHtmlLocalizerFactory,然後 用 IHtmlLocalizerFactory創建一個 IHtmlLocalizer 對應的實例。 在創建的時候 會帶入兩個參數 ,一個是 當前 試圖的路徑,一個是當前應用名稱。 

 

IHtmlLocalizer 定義如下:

 

所以在 IHtmlLocalizer的實例中, 既可以輕鬆的獲取對應的值。

因為 ViewLocalizer 會註入一個IHtmlLocalizerFactory 的實例。預設的實例 是  HtmlLocalizerFactory , 在 HtmlLocalizerFactory 的構造函數中會註入一個 IStringLocalizerFactory 的實例(位於Microsoft.Extensions.Localization.Abstractions)。

的定義是 

 

而  IHtmlLocalizerFactory 的定義是 

 

可以說  HtmlLocalizerFactory 是對 HtmlLocalizerFactory 的一個包裝。

 

查閱代碼知道 預設 IStringLocalizerFactory 實現是 ResourceManagerStringLocalizerFactory ,並且讀取資源文件均是這個實現來操作。

 

回到開頭的問題,假設我要使用 json 文件 代替 resx 文件。該如何實現呢,。?  有2種方法

1)只要實現對應的 IStringLocalizerFactory 並且代替預設的 ResourceManagerStringLocalizerFactory 。

2)重寫 ResourceManagerStringLocalizerFactory 。

1) 1,定義一個  JsonStringLocalizerFactory 並實現 IStringLocalizerFactory 。

public class JsonStringLocalizerFactory : IStringLocalizerFactory
    {
        private readonly string _applicationName;
        private readonly IHostingEnvironment _hostingEnvironment;
        private readonly LocalizationOptions _options;
        public JsonStringLocalizerFactory(IHostingEnvironment hostingEnvironment, IOptions<LocalizationOptions> localizationOptions)
        {
            if (localizationOptions == null)
            {
                throw new ArgumentNullException(nameof(localizationOptions));
            }
            this._hostingEnvironment = hostingEnvironment ?? throw new ArgumentNullException(nameof(hostingEnvironment));
            this._options = localizationOptions.Value;
            this._applicationName = hostingEnvironment.ApplicationName;
        }

        public IStringLocalizer Create(Type resourceSource)
        {
            TypeInfo typeInfo = IntrospectionExtensions.GetTypeInfo(resourceSource);
            //Assembly assembly = typeInfo.Assembly;
            //AssemblyName assemblyName = new AssemblyName(assembly.FullName);

            string baseResourceName = typeInfo.FullName;
            baseResourceName = TrimPrefix(baseResourceName, _applicationName + ".");

            return new JsonStringLocalizer(_hostingEnvironment, _options, baseResourceName, null);
        }

        public IStringLocalizer Create(string baseName, string location)
        {
            location = location ?? _applicationName;

            string baseResourceName = baseName;
            baseResourceName = TrimPrefix(baseName, location + ".");

            return new JsonStringLocalizer(_hostingEnvironment, _options, baseResourceName, null);
        }

        private static string TrimPrefix(string name, string prefix)
        {
            if (name.StartsWith(prefix, StringComparison.Ordinal))
            {
                return name.Substring(prefix.Length);
            }

            return name;
        }
    }

 

2, JsonStringLocalizer

public class JsonStringLocalizer : IStringLocalizer
    {
        private readonly ConcurrentDictionary<string, string> _all;

        private readonly IHostingEnvironment _hostingEnvironment;
        private readonly LocalizationOptions _options;

        private readonly string _baseResourceName;
        private readonly CultureInfo _cultureInfo;

        public LocalizedString this[string name] => Get(name);
        public LocalizedString this[string name, params object[] arguments] => Get(name, arguments);

        public JsonStringLocalizer(IHostingEnvironment hostingEnvironment, LocalizationOptions options, string baseResourceName, CultureInfo culture)
        {
            _options = options;
            _hostingEnvironment = hostingEnvironment;

            _cultureInfo = culture ?? CultureInfo.CurrentUICulture;
            _baseResourceName = baseResourceName + "." + _cultureInfo.Name;
            _all = GetAll();

        }

        public IEnumerable<LocalizedString> GetAllStrings(bool includeParentCultures)
        {
            return _all.Select(t => new LocalizedString(t.Key, t.Value, true)).ToArray();
        }

        public IStringLocalizer WithCulture(CultureInfo culture)
        {
            if (culture == null)
                return this;

            CultureInfo.CurrentUICulture = culture;
            CultureInfo.DefaultThreadCurrentCulture = culture;

            return new JsonStringLocalizer(_hostingEnvironment, _options, _baseResourceName, culture);
        }

        private LocalizedString Get(string name, params object[] arguments)
        {
            if (_all.ContainsKey(name))
            {
                var current = _all[name];
                return new LocalizedString(name, string.Format(_all[name], arguments));
            }
            return new LocalizedString(name, name, true);
        }

        private ConcurrentDictionary<string, string> GetAll()
        {
            var file = Path.Combine(_hostingEnvironment.ContentRootPath, _baseResourceName + ".json");
            if (!string.IsNullOrEmpty(_options.ResourcesPath))
                file = Path.Combine(_hostingEnvironment.ContentRootPath, _options.ResourcesPath, _baseResourceName + ".json");

            Debug.WriteLineIf(!File.Exists(file), "Path not found! " + file);

            if (!File.Exists(file))
                return new ConcurrentDictionary<string, string>();

            try
            {
                var txt = File.ReadAllText(file);

                return JsonConvert.DeserializeObject<ConcurrentDictionary<string, string>>(txt);
            }
            catch (Exception)
            {
            }

            return new ConcurrentDictionary<string, string>();
        }
    }

 

3,添加註入 

services.AddSingleton<IStringLocalizerFactory, JsonStringLocalizerFactory>();

 

4,json 文件

 

上面的代碼只是簡單的實現了 使用 點(.) 作為分隔符的json 文件作為資源文件。(其實上面的代碼運行後有個小問題)

 

代碼已經放到 Github 

 

2)。待實現~~~ 

 

鏈接:http://blog.wuliping.cn/post/aspnet-core-localization-and-custom-resource-service-with-file


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在劃分磁碟分區時,遇到錯誤“WARNING: Re-reading the partition table failed with error 22: Invalid argument” 如下所示: [root@DB-Server u02]# fdisk -l Disk /dev/sda: 500.... ...
  • 一:500錯誤 1、500 Internal Server Error 內部服務錯誤:顧名思義500錯誤一般是伺服器遇到意外情況,而無法完成請求。 2、500出錯的可能性: a、編程語言語法錯誤,web腳本錯誤 b、併發高時,因為系統資源限制,而不能打開過多的文件 3、一般解決思路: a、查看ngi ...
  • 安裝: [root@server ~]# yum install -y vsftpd [root@server ~]# rpm -ql vsftpd /etc/logrotate.d/vsftpd /etc/pam.d/vsftpd /etc/rc.d/init.d/vsftpd /etc/vsft ...
  • jdk版本:jdk-8u131-linux-x64.rpm 註:以下操作在root用戶或具有root許可權的用戶下操作 一、將 dk-8u131-linux-x64.rpm拷貝到/home目錄下 二、解壓rpm文件 三、環境變數的配置 註:環境變數的配置稍微麻煩一點,不過也不是特別難。 1.輸入以下命 ...
  • 最近公司需要優化導入的問題,由於之前使用的方式是生成 Insert 語句插入資料庫,數據量小的時候還行,但是隨著發展數據量漸漸大了,之前的方法性能就跟不上了,於是發現了 SqlBulkCopy 這個類。 使用 SqlBulkCopy 類只能向 SQL Server 表寫入數據。但是,數據源不限於 S ...
  • 一.寫在前面 半年以前,第一次在項目上實踐VueJs,由於在那之前,沒有Angular,avalon等框架的實踐經驗,所以在Vue的使用上,沒有給自己總結出更多的經驗和體驗。隨著項目進行和優化改版,無論是新代碼的增加還是舊代碼,在功能的實現和代碼的書寫上,Vue逐漸替代了Jquery,除了有些不容易 ...
  • CLR中說道,不要在構造函數中調用虛方法,原因是假如被實例化的類型重寫了虛方法,就會執行派生類型對虛方法的實現。但在這個時候,尚未完成對繼承層次結構中所有欄位的初始化。所以,調用虛方法會導致不可預測的行為。歸根結底,這是由於調虛方法時,直到運行時之前,都不會選擇執行該方法的實際類型。 在MSDN中, ...
  • 如果只是使用現有的WPF控制項的話,是很難滿足當前社會多複雜的業務。所以用戶自己訂製一系列控制項也是一種不可避免的情勢。WPF在控制方面分為倆種:用戶控制項和自定義控制項。相信看過前面章節的就明白他們倆者之間的差別。理解用戶控制項並不難——把現有的控制項組合在一起形成的控制項。而在筆者看來自定義控制項才是WPF最吸 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...