当前位置: 首页 > news >正文

【微软技术栈】C#.NET 中的本地化

本文内容

  1. 资源文件
  2. 注册本地化服务
  3. 使用 IStringLocalizer<T> 和 IStringLocalizerFactory
  4. 将其放在一起

本地化是针对应用支持的每个区域性,将应用资源转换为本地化版本的过程。 只有在完成本地化评审步骤,以验证全球化应用是否做好本地化准备后,才应继续执行本地化步骤。

可以开始进行本地化的应用程序分为两个概念块:一个是包含所有用户界面元素的块,另一个是包含可执行代码的块。 用户界面块仅包含可本地化的用户界面元素,如字符串、错误消息、对话框、菜单、嵌入的对象资源等区域性中性元素。 代码块仅包含所有支持的区域性要使用的应用代码。 公共语言运行时支持附属程序集资源模型,用于将应用的可执行代码与资源分隔开来。 

对于应用的每个本地化版本,请添加新的附属程序集,其中包含转换为目标区域性的相应语言的本地化用户界面块。 所有区域性的代码块应保持不变。 用户界面块的本地化版本和代码块组合生成了应用的本地化版本。

在本文中,你将了解如何使用 IStringLocalizer<T> 和 IStringLocalizerFactory 实现。 本文中的所有示例源代码都依赖于 Microsoft.Extensions.Localization 和 Microsoft.Extensions.Hosting NuGet 包。 

1、资源文件

隔离可本地化字符串的主要机制是使用资源文件。 资源文件是具有 .resx 文件扩展名的 XML 文件。 资源文件在执行使用应用程序之前被转换,换句话说,它们表示静态的已转换内容。 资源文件名通常包含区域设置标识符,并采用以下格式:

<FullTypeName><.Locale>.resx

其中:

  • <FullTypeName> 表示特定类型的可本地化资源。
  • 可选 <.Locale> 表示资源文件内容的区域设置。

1.1 指定区域设置

区域设置至少应该定义语言,但也可以定义区域性(区域语言),甚至是国家或地区。 这些段通常由 - 字符分隔。 通过添加区域性特异性,在为最佳匹配项设置优先级的地方应用“区域性回退”规则。 区域设置应该映射到已知语言标记。

1.2 区域性回退场景

假设你的本地化应用支持不同的塞尔维亚区域设置,并且其 MessageService 具有以下资源文件:

文件区域语言国家/地区代码
MessageService.sr-Cyrl-RS.resx(西里尔语,塞尔维亚)RS
MessageService.sr-Cyrl.resx西里尔语
MessageService.sr-Latn-BA.resx(拉丁语,波斯尼亚和黑塞哥维那)BA
MessageService.sr-Latn-ME.resx(拉丁语,黑山共和国)ME
MessageService.sr-Latn-RS.resx(拉丁语,塞尔维亚)RS
MessageService.sr-Latn.resx拉丁语
MessageService.sr.resx† 拉丁语
MessageService.resx

†语言的默认区域语言。

当应用运行时,将 CultureInfo.CurrentCulture 设置为 "sr-Cyrl-RS" 本地化的区域性,尝试按以下顺序解析文件:

  1. MessageService.sr-Cyrl-RS.resx
  2. MessageService.sr-Cyrl.resx
  3. MessageService.sr.resx
  4. MessageService.resx

但是,如果应用运行时,将 CultureInfo.CurrentCulture 设置为 "sr-Latn-BA" 本地化的区域性,则尝试按以下顺序解析文件:

  1. MessageService.sr-Latn-BA.resx
  2. MessageService.sr-Latn.resx
  3. MessageService.sr.resx
  4. MessageService.resx

如果没有相应的匹配项,“区域性回退”规则将忽略区域设置,这意味着如果找不到匹配项,则选择资源文件编号 4。 如果区域性设置为 "fr-FR",本地化最终会落到 MessageService.resx 文件,这会造成问题。 

1.3 资源查找

资源文件会在查找例程中自动解析。 如果项目文件名不同于项目的根命名空间,则程序集名称可能不同。 这可能会阻止资源查找成功。 要解决这种不匹配问题,请使用 RootNamespaceAttribute 向本地化服务提供提示。 提供以后,它将在资源查找期间使用。

示例项目名为 example.csproj,它会创建 example.dll 和 example.exe ,但是会使用 Localization.Example 命名空间。 应用 assembly 级别属性来更正这种不匹配问题:

[assembly: RootNamespace("Localization.Example")]

2、注册本地化服务

要注册本地化服务,请在服务配置期间调用其中一个 AddLocalization 扩展方法。 这将启用以下类型的依赖关系注入 (DI):

  • Microsoft.Extensions.Localization.IStringLocalizer<T>
  • Microsoft.Extensions.Localization.IStringLocalizerFactory

2.1 配置本地化选项

AddLocalization(IServiceCollection, Action<LocalizationOptions>) 重载接受类型为 Action<LocalizationOptions> 的 setupAction 参数。 这使你可以配置本地化选项。

HostApplicationBuilder builder = Host.CreateApplicationBuilder(args);builder.Services.AddLocalization(options =>
{options.ResourcesPath = "Resources";
});// Omitted for brevity.

资源文件可以存在于项目中的任何位置,但是有些常见做法已经被证明是成功的。 通常情况下,会选择阻力最小的路径。 上述 C# 代码:

  • 创建默认主机应用生成器。
  • 对服务集合调用 AddLocalization,并将 LocalizationOptions.ResourcesPath 指定为 "Resources"

这会使本地化服务在“Resources”目录中查找资源文件。

3、使用 IStringLocalizer<T> 和 IStringLocalizerFactory

在注册(并选择性配置)本地化服务后,可以将以下类型和 DI 结合使用:

  • IStringLocalizer<T>
  • IStringLocalizerFactory

要创建能够返回本地化字符串的消息服务,请考虑使用以下 MessageService

using System.Diagnostics.CodeAnalysis;
using Microsoft.Extensions.Localization;namespace Localization.Example;public sealed class MessageService
{private readonly IStringLocalizer<MessageService> _localizer = null!;public MessageService(IStringLocalizer<MessageService> localizer) =>_localizer = localizer;[return: NotNullIfNotNull(nameof(_localizer))]public string? GetGreetingMessage(){LocalizedString localizedString = _localizer["GreetingMessage"];return localizedString;}
}

在前述 C# 代码中:

  • 声明 IStringLocalizer<MessageService> _localizer 字段。
  • 构造函数采用 IStringLocalizer<MessageService> 参数并将其分配给 _localizer 字段。
  • GetGreetingMessage 方法调用 IStringLocalizer.Item[String],将 "GreetingMessage" 作为参数传递。

IStringLocalizer 还支持参数化字符串资源,请考虑使用以下 ParameterizedMessageService

using System.Diagnostics.CodeAnalysis;
using Microsoft.Extensions.Localization;namespace Localization.Example;public class ParameterizedMessageService
{private readonly IStringLocalizer _localizer = null!;public ParameterizedMessageService(IStringLocalizerFactory factory) =>_localizer = factory.Create(typeof(ParameterizedMessageService));[return: NotNullIfNotNull(nameof(_localizer))]public string? GetFormattedMessage(DateTime dateTime, double dinnerPrice){LocalizedString localizedString = _localizer["DinnerPriceFormat", dateTime, dinnerPrice];return localizedString;}
}

在前述 C# 代码中:

  • 声明 IStringLocalizer _localizer 字段。
  • 构造函数采用 IStringLocalizerFactory 参数,该参数用于从 ParameterizedMessageService 类型创建 IStringLocalizer,并将其分配给 _localizer 字段。
  • GetFormattedMessage 方法调用 IStringLocalizer.Item[String, Object[]],将 "DinnerPriceFormat"(一种 dateTime 对象)和 dinnerPrice 作为参数传递。

 重要

IStringLocalizerFactory 不是必需的。 而使用服务最好要求使用 IStringLocalizer<T>。

两个 IStringLocalizer.Item[] 索引器都返回 LocalizedString,它们具有向 string? 的隐式转换。

4、将其放在一起

若要举例说明使用两种消息服务以及本地化和资源文件的应用,请考虑使用以下 Program.cs 文件:

using System.Globalization;
using Localization.Example;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Localization;
using Microsoft.Extensions.Logging;
using static System.Console;
using static System.Text.Encoding;[assembly: RootNamespace("Localization.Example")]OutputEncoding = Unicode;if (args is { Length: 1 })
{CultureInfo.CurrentCulture =CultureInfo.CurrentUICulture =CultureInfo.GetCultureInfo(args[0]);
}HostApplicationBuilder builder = Host.CreateApplicationBuilder(args);builder.Services.AddLocalization();
builder.Services.AddTransient<MessageService>();
builder.Services.AddTransient<ParameterizedMessageService>();
builder.Logging.SetMinimumLevel(LogLevel.Warning);using IHost host = builder.Build();IServiceProvider services = host.Services;ILogger logger =services.GetRequiredService<ILoggerFactory>().CreateLogger("Localization.Example");MessageService messageService =services.GetRequiredService<MessageService>();
logger.LogWarning("{Msg}",messageService.GetGreetingMessage());ParameterizedMessageService parameterizedMessageService =services.GetRequiredService<ParameterizedMessageService>();
logger.LogWarning("{Msg}",parameterizedMessageService.GetFormattedMessage(DateTime.Today.AddDays(-3), 37.63));await host.RunAsync();

在前述 C# 代码中:

  • RootNamespaceAttribute 将 "Localization.Example" 设置为根命名空间。
  • Console.OutputEncoding 分配给 Encoding.Unicode。
  • 单个参数传递给 args 时,CultureInfo.CurrentCulture 和 CultureInfo.CurrentUICulture 分配有 arg[0]给定的 CultureInfo.GetCultureInfo(String) 的结果。
  • Host 使用默认值创建。
  • 本地化服务 MessageService 和 ParameterizedMessageService 注册到 DI 的 IServiceCollection
  • 为了消除干扰,日志记录被配置为忽略低于警告的任何日志级别。
  • MessageService 是从 IServiceProvider 实例解析的,其生成的消息被记录。
  • ParameterizedMessageService 是从 IServiceProvider 实例解析的,其生成的已设置格式的消息被记录。

每个 *MessageService 类都定义一组 .resx 文件,其中每个文件都有一个条目。 下面是 MessageService 资源文件的示例内容,从 MessageService.resx 开始:

<?xml version="1.0" encoding="utf-8"?>
<root><data name="GreetingMessage" xml:space="preserve"><value>Hi friends, the ".NET" developer community is excited to see you here!</value></data>
</root>

MessageService.sr-Cyrl-RS.resx:

<?xml version="1.0" encoding="utf-8"?>
<root><data name="GreetingMessage" xml:space="preserve"><value>Здраво пријатељи, ".NЕТ" девелопер заједница је узбуђена што вас види овде!</value></data>
</root>

MessageService.sr-Latn.resx:

<?xml version="1.0" encoding="utf-8"?>
<root><data name="GreetingMessage" xml:space="preserve"><value>Zdravo prijatelji, ".NET" developer zajednica je uzbuđena što vas vidi ovde!</value></data>
</root>

下面是 ParameterizedMessageService 资源文件的示例内容,从 ParameterizedMessageService.resx 开始:

<?xml version="1.0" encoding="utf-8"?>
<root><data name="DinnerPriceFormat" xml:space="preserve"><value>On {0:D} my dinner cost {1:C}.</value></data>
</root>

ParameterizedMessageService.sr-Cyrl-RS.resx:

<?xml version="1.0" encoding="utf-8"?>
<root><data name="DinnerPriceFormat" xml:space="preserve"><value>У {0:D} моја вечера је коштала {1:C}.</value></data>
</root>

ParameterizedMessageService.sr-Latn.resx:

<?xml version="1.0" encoding="utf-8"?>
<root><data name="DinnerPriceFormat" xml:space="preserve"><value>U {0:D} moja večera je koštala {1:C}.</value></data>
</root>

 提示

为简洁起见,有意省略所有资源文件 XML 注释、架构和 <resheader> 元素。

4.1 示例运行

以下示例运行显示给定目标区域设置的各种本地化输出。

以 "sr-Latn" 为例:

dotnet run --project .\example\example.csproj sr-Latnwarn: Localization.Example[0]Zdravo prijatelji, ".NET" developer zajednica je uzbuđena što vas vidi ovde!
warn: Localization.Example[0]U utorak, 03. avgust 2021. moja večera je koštala 37,63 ¤.

当省略运行项目的 .NET CLI 参数时,使用默认系统区域性,在本例中为 "en-US"

dotnet run --project .\example\example.csprojwarn: Localization.Example[0]Hi friends, the ".NET" developer community is excited to see you here!
warn: Localization.Example[0]On Tuesday, August 3, 2021 my dinner cost $37.63.

传递 "sr-Cryl-RS" 时,将找到正确的相应资源文件并应用本地化:

dotnet run --project .\example\example.csproj sr-Cryl-RSwarn: Localization.Example[0]Здраво пријатељи, ".NЕТ" девелопер заједница је узбуђена што вас види овде!
warn: Localization.Example[0]У уторак, 03. август 2021. моја вечера је коштала 38 RSD.

示例应用程序不提供 "fr-CA" 的资源文件,但在使用该区域性调用时,将使用非本地化的资源文件。

 警告

由于已找到区域性,但未找到正确的资源文件,因此应用格式设置时,最终会实现部分本地化:

dotnet run --project .\example\example.csproj fr-CAwarn: Localization.Example[0]Hi friends, the ".NET" developer community is excited to see you here!
warn: Localization.Example[0]On mardi 3 août 2021 my dinner cost 37,63 $.

相关文章:

【微软技术栈】C#.NET 中的本地化

本文内容 资源文件注册本地化服务使用 IStringLocalizer<T> 和 IStringLocalizerFactory将其放在一起 本地化是针对应用支持的每个区域性&#xff0c;将应用资源转换为本地化版本的过程。 只有在完成本地化评审步骤&#xff0c;以验证全球化应用是否做好本地化准备后&a…...

【qemu逃逸】华为云2021-qemu_zzz

前言 虚拟机用户名&#xff1a;root 无密码 设备逆向 经过逆向分析&#xff0c;可得实例结构体大致结构如下&#xff1a; 其中 self 指向的是结构体本身&#xff0c;cpu_physical_memory_rw 就是这个函数的函数指针。arr 应该是 PCI 设备类结构体没啥用&#xff0c;就直接用…...

vue递归获取树形菜单

文章目录 前言什么是递归&#xff1f; 一、数据集二、 递归函数三、打印树形结构展示 前言 什么是递归&#xff1f; 程序调用自身的编程技巧称为递归&#xff08; recursion&#xff09;。 递归 粗略的理解为 循环 &#xff0c;只不过 递归 是调用自身。 在实际使用中&#xf…...

[ubuntu]ubuntu22.04默认源和国内源

sudo vi /etc/apt/sources.list 请选择和系统对应的版本&#xff0c;查看系统版本命令&#xff1a; lsb_release -a Distributor ID: Ubuntu Description: Ubuntu 22.04 LTS Release: 22.04 Codename: jammy Ubuntu不同的版本配置的有…...

Map和ForEach的区别,For in和For of的区别

Map和ForEach的区别&#xff1a; 共同点&#xff1a;都可以遍历数组&#xff0c;this指向window&#xff0c;都不会改变原数组。 不同点&#xff1a;map返回一个数组&#xff0c;不会对空数组进行检测&#xff0c;如果是空数组map的话还是返回一个空数组&#xff0c;而空数组…...

json字符串属性名与实体类字段名转换

在项目开发过程中&#xff0c;会遇到实体类字段名与交互的json对象属性名不一致的情况&#xff0c;比如前段使用的是下划线方式定义&#xff0c;后端采用的是驼峰式定义&#xff0c;其他系统使用t表示一个时间戳&#xff0c;自己的系统使用timestamp定义。遇到这种情况&#xf…...

Vue Vuex模块化编码

正常写vuex的index的时候如果数据太多很麻烦&#xff0c;如有的模块是管理用户信息或修改课程等这两个是不同一个种类的&#xff0c;如果代码太多会造成混乱&#xff0c;这时候可以使用模块化管理 原始写法 如果功能模块太多很乱 import Vue from vue import Vuex from vuex …...

消费者忠诚度研究:群狼调研帮您制定忠诚客户计划

在当今竞争激烈的市场环境中&#xff0c;消费者忠诚度对于企业的成功至关重要。消费者忠诚度不仅关系到企业的市场份额和盈利能力&#xff0c;还直接影响着企业的品牌形象和声誉。群狼调研作为一家专业的市场研究机构&#xff0c;专注于消费者忠诚度研究&#xff0c;为企业提供…...

接口幂等性详解

1. 什么是幂等性 幂等性指的是对同一个操作的多次执行所产生的影响与一次执行的影响相同。无论操作执行多少次&#xff0c;系统状态都应该保持一致。 在计算机科学和网络领域中&#xff0c;幂等性通常用来描述服务或操作的特性。对于RESTful API或HTTP方法&#xff0c;一个幂…...

Java操作redis常见类型数据存储

一&#xff0c;Java连接Redis 1.1 导入依赖 打开IDEA在pom.xml导入依赖 注意&#xff1a;要在dependencies标签中导入 <dependency><groupId>redis.clients</groupId><artifactId>jedis</artifactId><version>2.9.0</version> &…...

【深度学习】pytorch——Autograd

笔记为自我总结整理的学习笔记&#xff0c;若有错误欢迎指出哟~ 深度学习专栏链接&#xff1a; http://t.csdnimg.cn/dscW7 pytorch——Autograd Autograd简介requires_grad计算图没有梯度追踪的张量ensor.data 、tensor.detach()非叶子节点的梯度计算图特点总结 利用Autograd实…...

【ARM 安全系列介绍 1 -- 奇偶校验与海明码校验详细介绍】

文章目录 奇偶校验介绍奇偶校验 python 实现奇偶校验C代码实现 海明码详细介绍 奇偶校验介绍 奇偶校验是一种错误检测方法&#xff0c;广泛应用于计算机内部以及数据通信领域。其基本原理是为了使得一组数据&#xff08;通常是一字节8位&#xff09;中的“1”的个数为偶数或奇…...

分享34个发布商会PPT,总有一款适合您

分享34个发布商会PPT&#xff0c;总有一款适合您 链接&#xff1a;https://pan.baidu.com/s/1jP9toqTZONWeDIcxvw1wxg?pwd8888 提取码&#xff1a;8888 Python采集代码下载链接&#xff1a;采集代码.zip - 蓝奏云 学习知识费力气&#xff0c;收集整理更不易。知识付费甚…...

047_第三代软件开发-日志分离

第三代软件开发-日志分离 文章目录 第三代软件开发-日志分离项目介绍日志分离用法 关键字&#xff1a; Qt、 Qml、 log、 日志、 分离 项目介绍 欢迎来到我们的 QML & C 项目&#xff01;这个项目结合了 QML&#xff08;Qt Meta-Object Language&#xff09;和 C 的强…...

ChinaSoft 论坛巡礼 | 系统与网络安全论坛

2023年CCF中国软件大会&#xff08;CCF ChinaSoft 2023&#xff09;由CCF主办&#xff0c;CCF系统软件专委会、形式化方法专委会、软件工程专委会以及复旦大学联合承办&#xff0c;将于2023年12月1-3日在上海国际会议中心举行。 本次大会主题是“智能化软件创新推动数字经济与社…...

Ubuntu Gitlab安javascript:void(‘numberedlist‘)装

原因&#xff1a; 代码越改越多&#xff0c;越难维护&#xff0c;开发代码和发布代码融为一体&#xff1b;2人以上开发&#xff0c;都会修改代码&#xff0c;修改次数一多&#xff0c;代码难以维护 其中&#xff1a;前往Gitlab官网&#xff1a;gitlab/gitlab-ce - Packages pa…...

11.4-GPT4AllTools版本已开始对小部分GPT3.5用户内测推送

OpenAI已经开始小规模推送GPT4 AllTools功能&#xff0c;部分GPT博主已经第一时间体验了此功能&#xff0c;此功能特色是整合目前的多模态功能以及文件上传和联网模块&#xff0c;无需切换&#xff0c;更要全面综合 可上传包括 PDF、数据文件在内的任意文档&#xff0c;并进行分…...

竞赛选题 深度学习手势检测与识别算法 - opencv python

文章目录 0 前言1 实现效果2 技术原理2.1 手部检测2.1.1 基于肤色空间的手势检测方法2.1.2 基于运动的手势检测方法2.1.3 基于边缘的手势检测方法2.1.4 基于模板的手势检测方法2.1.5 基于机器学习的手势检测方法 3 手部识别3.1 SSD网络3.2 数据集3.3 最终改进的网络结构 4 最后…...

语言模型AI——聊聊GPT使用情形与影响

GPT的出现象征着人工智能自然语言处理技术的一次巨大飞跃。从编程助手到写作利器&#xff0c;它的身影在各个行业中越来越常见。百度【文心一言】、CSDN【C知道】等基于GPT的产品相继推出&#xff0c;让我们看到了其广泛的应用前景。然而&#xff0c;随着GPT的普及&#xff0c;…...

浅谈事件冒泡和事件捕获

事件冒泡和事件捕获分别由微软和网景公司提出&#xff0c;这两个概念都是为了解决页面中事件流&#xff08;事件发生顺序&#xff09;的问题。 <div id"div1"><div id"div2"><div id"div3">click</div></div> <…...

SkyWalking 10.2.0 SWCK 配置过程

SkyWalking 10.2.0 & SWCK 配置过程 skywalking oap-server & ui 使用Docker安装在K8S集群以外&#xff0c;K8S集群中的微服务使用initContainer按命名空间将skywalking-java-agent注入到业务容器中。 SWCK有整套的解决方案&#xff0c;全安装在K8S群集中。 具体可参…...

云计算——弹性云计算器(ECS)

弹性云服务器&#xff1a;ECS 概述 云计算重构了ICT系统&#xff0c;云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台&#xff0c;包含如下主要概念。 ECS&#xff08;Elastic Cloud Server&#xff09;&#xff1a;即弹性云服务器&#xff0c;是云计算…...

【入坑系列】TiDB 强制索引在不同库下不生效问题

文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...

Vue3 + Element Plus + TypeScript中el-transfer穿梭框组件使用详解及示例

使用详解 Element Plus 的 el-transfer 组件是一个强大的穿梭框组件&#xff0c;常用于在两个集合之间进行数据转移&#xff0c;如权限分配、数据选择等场景。下面我将详细介绍其用法并提供一个完整示例。 核心特性与用法 基本属性 v-model&#xff1a;绑定右侧列表的值&…...

ssc377d修改flash分区大小

1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...

ServerTrust 并非唯一

NSURLAuthenticationMethodServerTrust 只是 authenticationMethod 的冰山一角 要理解 NSURLAuthenticationMethodServerTrust, 首先要明白它只是 authenticationMethod 的选项之一, 并非唯一 1 先厘清概念 点说明authenticationMethodURLAuthenticationChallenge.protectionS…...

【C语言练习】080. 使用C语言实现简单的数据库操作

080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...

CRMEB 框架中 PHP 上传扩展开发:涵盖本地上传及阿里云 OSS、腾讯云 COS、七牛云

目前已有本地上传、阿里云OSS上传、腾讯云COS上传、七牛云上传扩展 扩展入口文件 文件目录 crmeb\services\upload\Upload.php namespace crmeb\services\upload;use crmeb\basic\BaseManager; use think\facade\Config;/*** Class Upload* package crmeb\services\upload* …...

Java多线程实现之Thread类深度解析

Java多线程实现之Thread类深度解析 一、多线程基础概念1.1 什么是线程1.2 多线程的优势1.3 Java多线程模型 二、Thread类的基本结构与构造函数2.1 Thread类的继承关系2.2 构造函数 三、创建和启动线程3.1 继承Thread类创建线程3.2 实现Runnable接口创建线程 四、Thread类的核心…...

从 GreenPlum 到镜舟数据库:杭银消费金融湖仓一体转型实践

作者&#xff1a;吴岐诗&#xff0c;杭银消费金融大数据应用开发工程师 本文整理自杭银消费金融大数据应用开发工程师在StarRocks Summit Asia 2024的分享 引言&#xff1a;融合数据湖与数仓的创新之路 在数字金融时代&#xff0c;数据已成为金融机构的核心竞争力。杭银消费金…...