zoukankan      html  css  js  c++  java
  • 国际化的实现i18n--错误码国际化以及在springboot项目中使用

    国际化 ,英文叫 internationalization 单词太长 ,又被简称为 i18n(取头取尾中间有18个字母);

    主要涉及3个类: Locale用来设置定制的语言和国家代码 

           ResourceBundle 用来加载国际化资源文件

           MessageFormat 用来格式化占位符

    //先看结构:

    在创建国际化资源文件时,如我在resources文件下建了一个i18n文件夹,下放了3个资源文件,idea自动会生成一个Resource Bundle 'messages'文件夹,这个文件夹不是真实存在的,打开项目所在文件夹后,找不到它的;如图:

    资源文件的命名格式:前缀+"_"+"语言代码"+地区代码+".properites";(

    Locale[] locales = Locale.getAvailableLocales();//获取世界可用的语言地区表示

    )

    如上图所示,我定义了3个资源文件夹,其中一个没有语言代码和地区代码的那个是默认资源,也就是要找的资源文件都不存在时,就会去那里找;

    查找顺序,如果你指定Locale locale=new Locale("en_US"),那它就会找到messages_en_US.properties,如果你写错了一个单词Locale locale=new Locale("en_USAA");找不到的话,它就会找本地的语言(

    Locale locale2 = Locale.getDefault(); //获取默认语言地区

    ),如我们在中国就会找本地的中国语言包messages_zh_CN.properties,如果没有本地的语言包messages_zh_CN.properties,就会去messages.properties这里找

    资源文件的内容格式为:key=value,其中value值可以含有占位符,格式{数字},例如" 你的订单号{0}的订单,金额{1}元";

    key和value值在properties文件中都不用加双引号

    注意:

    资源文件中的所有字符都必须是 ascll 码 ,不能保存为中文的 ,Java 中提供 了 native2ascll 工具用于将中文转化为 ascll 码 。所以在编写 properties 文件的时候写的是中文 ,一回车就自动被编码了 。

    在idea工具中,可以设置显示为中文:

    后端做国际化用得比较多的情况是错误码国际化:具体代码如下:

     @Test
        public void test2(){
            Locale locale1 = new Locale("en_US");
            Locale locale2 = Locale.getDefault(); //获取默认语言地区
            System.out.println(locale2.getCountry()); //CN
            System.out.println(locale2.getDisplayCountry()); //中国
            System.out.println(locale2.getDisplayLanguage()); //中文
            System.out.println(locale2.getDisplayName()); //中文(中国)
            System.out.println(locale2.getDisplayLanguage(locale1)); //以某种语言显示语言,这里是Chinese
            System.out.println(locale2.getLanguage()); //zh 语言代表符
            System.out.println(locale2.toLanguageTag()); //语言标签 格式语言-国家 这里是zh-CN
            //Locale自定义了很多语言跟国家常量 如中国 和中文,德国和德文
            Locale china = Locale.CHINA; // zh-Cn
            Locale chinese = Locale.CHINESE; //ZH
            System.out.println(china.toLanguageTag()); //ZH-CN
            System.out.println(chinese.toLanguageTag()); //ZH
            Locale german = Locale.GERMAN; //de
            Locale germany = Locale.GERMANY; //de-DE
            System.out.println(german.toLanguageTag());//de
            System.out.println(germany.toLanguageTag());//de-DE
            Locale[] locales = Locale.getAvailableLocales();//获取世界可用的地区
            for (Locale locale : locales) {
                System.out.println(locale.toLanguageTag());
            }
            ResourceBundle resourceBundle = ResourceBundle.getBundle("i18n/messages",locale2); //这里的baseName也可以表示成i18n.messages形式,messages是虚拟的包名
    
            String s = resourceBundle.getString("103014"); //我在资源文件存了:103014=订单号为{0}的订单已经支付
            MessageFormat messageFormat = new MessageFormat(s);
            String format = messageFormat.format(new String[]{"100002222"});
            System.out.println(format);//输出值(订单号为{100002222}的订单已经支付)
    
    
        }
    
    //稍微封装:
     private String getErrorMessage(String language,String errorCode,String ...params){
            Locale locale=null;
            if(StringUtils.isEmpty(language)){
                locale=new Locale("zh_CN");
            }
    
            locale=new Locale(language);
            ResourceBundle bundle = ResourceBundle.getBundle("i18n/messages",locale);
            String msg = bundle.getString(errorCode);
            if(StringUtils.isEmpty(msg)){
                return null;
            }
            MessageFormat messageFormat = new MessageFormat(msg);
            String format = messageFormat.format(params);
            return format;
    
        }
      @Test
        public void testI18n() throws IOException {
       
          //key=value 103032=Order No.{0} and No.{1} has not been linked
            System.out.println(getErrorMessage("en_US", "103032", "dddd", "vvvvvv"));
    
        }//输出为Order No.dddd and No.vvvvvv has not been linked



    补充:注意下面的区别:下划线和横线用在不同方法,建议使用

    Locale locale = new Locale("en_US");

    Locale locale2 =Locale.forLanguageTag("en-US"); //推荐使用这种方式
     

     //springboot项目中使用国际化,非常简单:

    @RunWith(SpringRunner.class)
    @SpringBootTest
    public class JestTest {
        @Autowired
        private MessageSource messageSource;
    
        @Test
        public void testI18n() throws IOException {
    
            Locale locale2 = Locale.forLanguageTag("en-US");
    
            Object[] arr = {"dddd", "vvvvvv"};
            //key=value    103032=Order No.{0} and No.{1} has not been linked
            System.out.println(messageSource.getMessage("103032", arr, locale2));
            //输出结果:Order No.dddd and No.vvvvvv has not been linked
    
        }
    
    }
    需要在application.yml中配置:
    messages:
    always-use-message-format: false # Whether to always apply the MessageFormat rules, parsing even messages without arguments.
    basename: i18n/messages # Comma-separated list of basenames (essentially a fully-qualified classpath location), each following the ResourceBundle convention with relaxed support for slash based locations.
    cache-duration: # Loaded resource bundle files cache duration. When not set, bundles are cached forever. If a duration suffix is not specified, seconds will be used.
    encoding: UTF-8 # Message bundles encoding.
    fallback-to-system-locale: true # Whether to fall back to the system Locale if no files for a specific Locale have been found.
    use-code-as-default-message: false # Whether to use the message code as the default message instead of throwing a "NoSuchMessageException". Recommended during development only.
    如果报这个:org.springframework.context.NoSuchMessageException: No message found under code '103032' for locale 'en_us'
    非常有可能是上面的messages在application.yml的位置没配置对,messages要配置在spring的下一级,如图所示:

     

     错误配置如下:

                              

    
    
    
  • 相关阅读:
    jdk动态代理底层实现
    spring-FactoryBean
    大型网站技术架构 核心原理与案例分析 pdf
    实战JAVA虚拟机 JVM故障诊断与性能优化 pdf
    《实战Java高并发程序设计》pdf
    mysql-注意点
    Json入门
    inflate, findViewById与setContentView的区别与联系
    提高编程能力的7条建议
    JDBC之一:JDBC快速入门
  • 原文地址:https://www.cnblogs.com/yangxiaohui227/p/11180640.html
Copyright © 2011-2022 走看看