java 如何实现多语言配置i18n

java 实现多语言配置i18n

1、创建资源包和资源文件

一个资源包中的每个资源文件都必须拥有共同的基名。除了基名,每个资源文件的名称中还必须有标识其本地信息的附加部分。

例如:一个资源包的基名是“inspectionJsonMsg”,则与中文、英文环境相对应的资源文件名则为: "inspectionJsonMsg_zh_CN.properties" "inspectionJsonMsg_en_US.properties"

2、资源文件的书写格式

资源文件的内容通常采用"关键字=值"的形式,软件根据关键字检索值显示在页面上。一个资源包中的所有资源文件的关键字必须相同,值则为相应国家的文字。并且资源文件中采用的是properties格式文件,所以文件中的所有字符都必须是ASCII字码,属性(properties)文件是不能保存中文的,对于像中文这样的非ACSII字符,须先进行编码

例如:

国际化的中文环境的properties文件

国际化的英文环境的properties文件

3、在Spring配置文件中增加多语言配置

applicationContext.xml

<!--多语言的配置 开始-->
	<bean id="messageSource" class="org.springframework.context.support.ResourceBundleMessageSource">
		<property name="basenames">
			<list>
				<value>language/inspectionJsonMsg</value>
			</list>
		</property>

		<property name="defaultEncoding" value="UTF-8" />
	</bean>

	<bean id="localeResolver" class="org.springframework.web.servlet.i18n.CookieLocaleResolver">
		<property name="cookieName" value="spmsLanguage"/>
		<property name="cookieMaxAge" value="94608000"/>
		<property name="defaultLocale" value="zh_CN" />
	</bean>
	<!--多语言的配置 结束-->

4、如何使用

(1)编写常量类

public class InspectionConst {

	//设备查询成功
	public static final String INSPECTION_DEVICE_QUERY_SUCCESS = "ny.spms.java.inspection.device.query.success";
	//设备查询失败
	public static final String INSPECTION_DEVICE_QUERY_ERROR = "ny.spms.java.inspection.device.query.error";
}

(2)编写I18n工具类,I18nUtil.java

package com.hikvision.energy.util.i18n;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.MessageSource;
import org.springframework.context.MessageSourceResolvable;
import org.springframework.context.i18n.LocaleContextHolder;

import com.hikvision.energy.core.util.AppContext;

import java.util.Locale;

/**
 * 多语言查询value工具类
 *
 * @author wanjiadong
 * @date 2017-11-6
 *
 */
public class I18nUtil {

    private static MessageSource messageSource;

    private final static Logger log = LoggerFactory.getLogger(I18nUtil.class);

    static {
        messageSource = AppContext.getBean("messageSource");
    }
    /**
     * 根据CODE查询,默认无通配参数,Local跟随当前cookie
     * @Author: wanjiadong
     * @Description:
     * @Date: 2017-11-6
     * @param: code
     */
    public static String getMessage(String code){
        return getMessage(code,null,getLocal());
    }

    public static String getMessage(String code,Locale locale){
        return getMessage(code,null,locale);
    }

    /**
     * 根据CODE查询,自定义默认值,默认无通配参数,Local跟随当前cookie
     * @Author: wanjiadong
     * @Description:
     * @Date: 2017-11-6
     * @param: code
     */
    public static String getMessage(String code,String defaultMessage){
        return getMessage(code,null,defaultMessage,getLocal());
    }
    /**
     * 根据CODE和args查询,Local跟随当前cookie
     * @Author: wanjiadong
     * @Description:
     * @Date: 2017-11-6
     * @param: code
     * @param: args 通配符的参数
     */
    public static String getMessage(String code,Object[] args){
        return getMessage(code,args,getLocal());
    }

    public static String getMessage(String code,Object[] args,String defaultMessage,Locale locale){
        return messageSource.getMessage(code,args,defaultMessage,locale);
    }

    public static String getMessage(String code, Object[] args, Locale locale){
        try{
            return messageSource.getMessage(code,args,locale);
        }catch (Exception e){
            log.error("Query message value by key[{}] error. The reason is:"+e.getMessage(),code);
            return null;
        }

    }

    public static String getMessage(MessageSourceResolvable resolvable, Locale locale){
        try{
            return messageSource.getMessage(resolvable,locale);
        }catch (Exception e){
            log.error("Query message value error. The reason is:"+e.getMessage());
            return null;
        }

    }

    //解析用户区域
    public static Locale getLocal() {
        return LocaleContextHolder.getLocale();
    }
}

(3)具体使用:

I18nUtil.getMessage(InspectionConst.TEMPORARY_JOB_OVER_BEGIN_TIME, new Object[]{InspectionConst.TEMPORARA_TIME})

I18nUtil.getMessage(InspectionConst.INSPECTION_TEMPORARY_JOB_ADD_SUCCESS)

Java I18N国际化

i18n(其来源是英文单词internationalization的首末字符i和n,18为中间的字符数)是“国际化”的简称。对程序来说,在不修改内部代码的情况下,能根据不同语言及地区显示相应的界面。

这就要求,仅仅是页面语言的翻译是不够的,即便在一个国家在不同地区都可能会存在不同习惯方言操作习惯等等(例如我们国家搜索结果习惯左对齐,但是有的国家就习惯右对齐),这就导致了“本地化”(L10n)机制的出现。

国际化和本地化的出现,在语言、文化、书写习惯、阅读习惯、符合当地主题的全面配置,使得一个Web应用程序在运行时能够根据客户端请求所来自的国家和语言显示不同的用户界面。

【示例】

根据以上简介和查看的具体软件的配置,理解一下i18n吧,所以小编查看了一下QQ的i18n配置文件,如下:

<?xml version="1.0" encoding="utf-8" ?>
<i18n>
	<!-- 默认的语言,以主程序的优?-->
	<locale id="2052" />
	<!-- 语言文字列表 -->
	<StringBundle>
		<configfile name="GFStringBundle.xml"/>
		<configfile name="StringBundle.xml"/>
		<configfile name="UrlBundle.xml"/>
	</StringBundle>
	<!-- 地区信息,目前只需要一?-->
	<Location2>
		<configfile name="LocList.xml"/>
	</Location2>
	<LangList>
		<configfile name="LangList.xml"/>
	</LangList>
</i18n>

又查看了一下要读取的LangList.xml文件,很壮观,如下:

<?xml version="1.0" encoding="utf-8"?>
<Languages>
	<Language LCID="1078" Name="南非荷兰语" Code="af"/>
	<Language LCID="1052" Name="阿尔巴尼亚语" Code="sq"/>
	<Language LCID="1025" Name="阿拉伯语" Code="ar"/>
	<Language LCID="1067" Name="亚美尼亚语" Code="hy"/>
	<Language LCID="2092" Name="阿塞拜疆语" Code="az"/>
	<Language LCID="1059" Name="白俄罗斯语" Code="be"/>
	<Language LCID="5146" Name="波斯尼亚语" Code="bs-ba"/>
	<Language LCID="1026" Name="保加利亚语" Code="bg"/>
	<Language LCID="1109" Name="缅甸语" Code="my"/>
	<Language LCID="3076" Name="中文(繁体)" Code="zh-hk"/>
	<Language LCID="1027" Name="加泰罗尼亚语" Code="ca"/>
	<Language LCID="1050" Name="克罗地亚语" Code="hr"/>
	<Language LCID="1029" Name="捷克语" Code="cs"/>
	<Language LCID="1030" Name="丹麦语" Code="da"/>
	<Language LCID="1043" Name="荷兰语" Code="nl"/>
	<Language LCID="1033" Name="英语" Code="en-us"/>
	<Language LCID="1061" Name="爱沙尼亚语" Code="et"/>
	<Language LCID="1065" Name="波斯语" Code="fa"/>
	<Language LCID="1035" Name="芬兰语" Code="fi"/>
	<Language LCID="1036" Name="法语" Code="fr"/>
	<Language LCID="2108" Name="盖尔语" Code="gd"/>
	<Language LCID="1031" Name="德语" Code="de"/>
	<Language LCID="1032" Name="希腊语" Code="el"/>
	<Language LCID="1095" Name="古吉拉特语" Code="gu"/>
	<Language LCID="1037" Name="希伯来语" Code="he"/>
	<Language LCID="1081" Name="印地语" Code="hi"/>
	<Language LCID="1038" Name="匈牙利语" Code="hu"/>
	<Language LCID="1039" Name="冰岛语" Code="is"/>
	<Language LCID="1057" Name="印度尼西亚语" Code="id"/>
	<Language LCID="1040" Name="意大利语" Code="it"/>
	<Language LCID="1041" Name="日语" Code="ja"/>
	<Language LCID="1107" Name="高棉语" Code="km"/>
	<Language LCID="1042" Name="朝鲜语" Code="ko"/>
	<Language LCID="1108" Name="老挝语" Code="lo"/>
	<Language LCID="1062" Name="拉脱维亚语" Code="lv"/>
	<Language LCID="1063" Name="立陶宛语" Code="lt"/>
	<Language LCID="1071" Name="马其顿语" Code="mk"/>
	<Language LCID="1086" Name="马来西亚语" Code="ms"/>
	<Language LCID="2052" Name="中文(简体)" Code="zh-cn"/>
	<Language LCID="1104" Name="蒙古语" Code="mn"/>
	<Language LCID="1044" Name="挪威语" Code="no"/>
	<Language LCID="1045" Name="波兰语" Code="pl"/>
	<Language LCID="2070" Name="葡萄牙语" Code="pt"/>
	<Language LCID="1094" Name="旁遮普语" Code="pa"/>
	<Language LCID="1048" Name="罗马尼亚语" Code="ro"/>
	<Language LCID="1049" Name="俄语" Code="ru"/>
	<Language LCID="3098" Name="塞尔维亚语" Code="sr"/>
	<Language LCID="1113" Name="信德语" Code="sd"/>
	<Language LCID="1051" Name="斯洛伐克语" Code="sk"/>
	<Language LCID="1060" Name="斯洛文尼亚语" Code="sl"/>
	<Language LCID="1143" Name="索马里语" Code="so"/>
	<Language LCID="1034" Name="西班牙语" Code="es"/>
	<Language LCID="1089" Name="斯瓦西里语" Code="sw"/>
	<Language LCID="1053" Name="瑞典语" Code="sv"/>
	<Language LCID="1097" Name="泰米尔语" Code="ta"/>
	<Language LCID="1092" Name="鞑靼语" Code="tt"/>
	<Language LCID="1054" Name="泰语" Code="th"/>
	<Language LCID="1055" Name="土耳其语" Code="tr"/>
	<Language LCID="1058" Name="乌克兰语" Code="uk"/>
	<Language LCID="1056" Name="乌尔都语" Code="ur"/>
	<Language LCID="1066" Name="越南语" Code="vi"/>
	<Language LCID="1106" Name="威尔士语" Code="cy-gb"/>
	<Language LCID="1085" Name="意第绪语" Code="yi"/>
	<Language LCID="1130" Name="约鲁巴语" Code="yo"/>
</Languages>

【实践】

1.locale方法

Java.Util中有一个locale方法,该方法的其中一个构造方法为:

Locale(Stringlanguage,String country):根据语言和国家构造一个语言环境。每个国家都有一个locale信息,通过对象可以取得locale信息,locale信息来源于操作系统。如下:

package com.bjpowernode.i18n;

import java.util.Locale;

public class I18nSample {

 public static void main(String[] args) {

  Locale defaultLocale = Locale.getDefault();
  System.out.println("country=" + defaultLocale.getCountry());
  System.out.println("language=" + defaultLocale.getLanguage());

 }
}

运行结果为:

country=CN

language=zh

但是在我们下面的实践中,为了方便我们就直接进行设置了。

2.ResourceBundle资源包

我们需要将硬编码文本转移到外部的资源文件,对资源文件的命名必须使用一定的规则,一般是以“统一字符+locale信息”命名,如MessagesBundle_zh_CN.properties,MessagesBundle_en_US.properties。准备资源包的过程,就是把对应不同语言的用户所涉及的文本和图片保存在多个文本文件中,客户端根据不同的环境需要进行更换。这些文件被称为“属性文件”,所有属性文件合在一起被称为资源包(ResourceBundle)。

因为资源文件必须是Latin-1后者Unicode编码的字符,所以实践过程中,准备英文好说,准备中文的话,需要使用相应的Unicode编码,使用jdk自带的native2ascii.exe,将中文资源文件进行转换,如下:

如下:

也可以直接在资源文件中添加,会自动转换成Unicode编码,如下:

结果如下:

Demo中我们中文和英文准备的都是:k1=你好/hello,k2=再见/goodbye。

【实践代码】

package com.bjpowernode.i18n;

import java.text.MessageFormat;
import java.util.Locale;
import java.util.ResourceBundle;

public class I18nSample {

	public static void main(String[] args) {
		// 为了实验,就不每次都修改操作系统的语言了,直接设置locale信息
		// Locale currentLocale = new Locale("zh", "CN");
		Locale currentLocale = new Locale("en", "US");

		System.out.println("country=" + currentLocale.getCountry());
		System.out.println("language=" + currentLocale.getLanguage());

		// 根据locale信息寻找相应的资源包中的配置
		ResourceBundle rb = ResourceBundle.getBundle(
				"com.bjpowernode.resources.MessagesBundle", currentLocale);
		System.out.println(rb.getString("k1"));
		System.out.println(rb.getString("k2"));
		// 在配置文件中用占位符来加入自定义设置,如登陆时显示:你好,***
		MessageFormat mf = new MessageFormat(rb.getString("k1"));
		System.out.println(mf.format(new Object[] { "张三" }));
	}
}

运行结果为:

country=US

language=en

hello,{0}

good bye

hello,张三

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • Java SpringMVC实现国际化整合案例分析(i18n)

    所谓国际化就是支持多种语言,web应用在不同的浏览环境中可以显示出不同的语言,比如说汉语.英语等.下面我将以具体的实例来举例说明: (1)新建动态Java web项目,并导入几个SpringMVC必需的几个jar包,项目结构图和所需jar包如下: (2)配置web.xml: <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-ins

  • Springmvc国际化自动配置代码实现

    在 springmvc 中通过 LocalResover 获取区域信息对象 Local(对语言信息.国家代码等的封装),进行国际化配置,在 springboot 为我们提供了默认的区域信息解析器 @Configuration( proxyBeanMethods = false ) @ConditionalOnWebApplication( type = Type.SERVLET ) @ConditionalOnClass({Servlet.class, DispatcherServlet.cla

  • SpringBoot消息国际化配置实现过程解析

    一.目的 针对不同地区,设置不同的语言信息. SpringBoot国际化配置文件默认放在classpath:message.properties,如果自定义消息配置文件,需要application.properties或application.yml中设置spring.messages.basename的值. 二.步骤 在src/main/resources 下建i18n文件夹 在i18n文件夹中建立messages.properties 找不到语言配置时,使用此文件 hello=你好_默认 在

  • java 如何实现多语言配置i18n

    java 实现多语言配置i18n 1.创建资源包和资源文件 一个资源包中的每个资源文件都必须拥有共同的基名.除了基名,每个资源文件的名称中还必须有标识其本地信息的附加部分. 例如:一个资源包的基名是"inspectionJsonMsg",则与中文.英文环境相对应的资源文件名则为: "inspectionJsonMsg_zh_CN.properties" "inspectionJsonMsg_en_US.properties" 2.资源文件的书写格

  • Yii2简单实现多语言配置的方法

    本文实例讲述了Yii2简单实现多语言配置的方法,分享给大家供大家参考,具体如下: 1.配置文件中添加 'i18n' => [ 'translations' => [ '*' => [ 'class' => 'yii\i18n\PhpMessageSource', 'basePath' => '@app/messages', 'sourceLanguage' => 'en', 'fileMap' => [ 'custorm' => 'custorm.php',

  • 详解Java如何使用注解来配置Spring容器

    目录 介绍 @Bean and @Configuration AnnotationConfigApplicationContext实例化容器 通过使用 register(Class...) 以编程方式构建容器 @ComponentScan启用组件扫描 Bean的依赖 生命周期回调 Bean指定作用域 自定义bean名称 Bean别名 Bean注入之间的依赖 @Import @ImportResource @PropertySource 支持多个properties文件 ApplicationCo

  • Java 8 动态类型语言Lambda表达式实现原理解析

    Java 8支持动态语言,看到了很酷的Lambda表达式,对一直以静态类型语言自居的Java,让人看到了Java虚拟机可以支持动态语言的目标. import java.util.function.Consumer; public class Lambda { public static void main(String[] args) { Consumer<String> c = s -> System.out.println(s); c.accept("hello lambd

  • Java程序与C语言的区别浅析

    Java和C语言虽有相同性,但两者也有一定的不同.Java程序是面向对象的一种简单.分布式 .解释.健壮.安全.结构中立.可移植.高效能.多线程.动态的语言它是面向对象而C语言是面向过程的,这是最大的不同,对于学过C语言的我们来说,Java可以说是比较简单的编程语言. 在Java语言中,所有的变量必须要首先声明,然后才能使用. Java中可以有若干类但有且只有一个public,一个类中至少含有一个main方法,否则程序不能运行.而一个C语言由若干函数构成,其中必须且只有一个main的主函数,程序

  • 查看Java所支持的语言及相应的版本信息

    /* *Java语言作为第一种支持国际化的语言,在Internet从一开始就具有其他语言无与伦比的国际化的本质特性:*用Unicode来编写所有的字符串.可是理论与实际总是不能完全谋和的,因为Unicode本身也在不断变化.* 在J2SE(TM) 5.0中,主要的变化是对java.lang,java.text, java.util.regex等包进行调整,使整个字*符处理基于Unicode4.0的标准之上,同时加强了对增补字符的支持(欲了解更多信息请参照:* <Supplementary Cha

  • java实现Spring在XML配置java类的方法

    1. 创建自己的bean文件:beans.xml <?xml version="1.0" encoding="UTF-8"?> <busi-beans> <beans> <bean id="SysHelloImpl" type="com.cxm.test.SysHello"> <desc>test</desc> <impl-class>com.

  • java 使用memcached以及spring 配置memcached完整实例代码

    Memcached是一个高性能的分布式内存对象缓存系统,本文介绍了java 使用memcached以及spring 配置memcached完整实例代码,分享给大家 本文涉及以下内容: 1,要使用的jar包 2,java 使用memcached 3,spring 配置memcached 导入jar java_memcached-release_2.6.6.jar commons-pool-1.5.6.jar slf4j-api-1.6.1.jar slf4j-simple-1.6.1.jar 示例

  • Java加载property文件配置过程解析

    这篇文章主要介绍了java加载property文件配置过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 1 properties简介: properties是一种文本文件,内容格式为: key = value #单行注释 适合作为简单配置文件使用,通常作为参数配置.国际化资源文件使用. 对于复杂的配置,就需要使用XML.YML.JSON等了 2 java加载Properties: java加载properties主要通过2个util包下的

  • PHP项目多语言配置平台实现过程解析

    这个月用ant design的方式重构了我在15年的时候写的一个给主端项目写的多语言配置后台.发现当时的设计很"深",深到有些功能根本就没用过,根据这几年的使用情况,重构了这一版本,无论是体验上还是实用性上都有所提高.今天抽时间把当时的一些想当然的需求拿出来总结下吧: 使用的技术 我们的项目主要是PHP为主要开发语言.使用gettext扩展实现国际化的.即通过人工的方式按照规则写入.po文件,然后调用某函数生成.mo 文件,放到生产环境的文件夹下,然后在代码里初始化一些内容进行引用.

随机推荐