有关Android国际化的一点积累

    流行好用的软件应该能够适用于不同地区的市场。下面记录一些在项目中国际化的应用,有关图片和文本资源的自适应。

     Android采用XML资源文件来管理所有字符串消息,如果系统设置的Custom Locale,没有对应的本地化资源文件,那么程序就会取默认的res\values\strings.xml。在此我们默认的strings为英文,如下:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <string name="app_name">L10NTest</string>
    <string name="action_settings">Settings</string>
    <string name="hello_world">Hello world!</string>
    <string name="I_love_you">I love you.</string>
</resources>

    我们添加有关中文的字符串资源,这需要为values目录添加中文的语言国家版本。values文件夹的命名方式为:values-语言代码-r国家代码。

    如果想要支持中文的话,则需要在res目录下添加values-zh-rCN。该文件夹中的strings如下:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <string name="app_name">国际化测试</string>
    <string name="action_settings">设置</string>
    <string name="hello_world">世界,你好!</string>
    <string name="I_love_you">我爱你.</string>
</resources>

    如果希望应用的图片也能随语言环境改变,则需要为drawable目录添加其他的语言国家版本。drawable文件夹的命名方式为:drawable-语言代码-r国家代码。同样支持中文的话需要在res目录下添加drawable-zh-rCN。

    如果还需要为drawable目录按照分辨率提供文件夹,可以在其后追加分辨率后缀,如:drawable-zh-rCN-mdpi、drawable-en-rUS-xhdpi等等。

    主Activity中没有写任何东西,下面是它的XML布局:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center"
    android:orientation="vertical" >
    <ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:src="@drawable/country"
        tools:ignore="ContentDescription" />
 <!-- tools:ignore="ContentDescription",屏蔽掉对View设置备注说明的提示 -->
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/hello_world"
        android:textSize="20sp" />
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/I_love_you"
        android:textSize="20sp" />
</LinearLayout>

    在android ADT 16.0以后的控件中,没有文本描述的view如果不加android:contentDescription="@string/……"的话就会有黄 色的下划线,提示“[Accessibility] Missing contentDescription attribute on image”。加“ tools:ignore="ContentDescription"“便是屏蔽掉该提示。这句代码不加也没什么影响,在界面上不会有其他效果,当然也可以写一些字符串数据做提示,方便他人阅读。或者直接写成:

<ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:contentDescription="@null"
        android:src="@drawable/country" />

也可。

    以下是应用的res结构:

        wKiom1VI3QTQd4XcAAEtvUBA0Pg471.jpg

    我增加了”drawable-en-rUS“、”drawable-zh-rCN“和”values-zh-rCN“三个文件夹,其中前两个文件夹中放了名称都是”country“的图片,只是前者为美国国旗图片,后者为中国国旗图片。第三个文件夹中的strings文件上面已给出。

    运行程序,如果设置系统的语言和输入为中文,效果图如下:

        wKioL1VI4RaQEdNfAAVLO6c0vGY031.jpg

    如果设置为英文,效果图如下:

        wKioL1VI4bDR5LIhAAgH3Dcmpg4069.jpg

    我并没有增加有关英文的本地化资源文件,所以程序就会取res\values\下的strings.xml,当然也可以增加对应的文件夹为:values-en-rUS,其中放置相关文本资源。

    如果是其他国家的语言环境,那文件夹该如何命名呢,以字符串资源为例,如下:

        中文(中国):values-zh-rCN

        中文(台湾):values-zh-rTW

        中文(香港):values-zh-rHK

        英语(美国):values-en-rUS

        英语(英国):values-en-rGB

        英文(澳大利亚):values-en-rAU

        英文(加拿大):values-en-rCA

        英文(爱尔兰):values-en-rIE

        英文(印度):values-en-rIN

        英文(新西兰):values-en-rNZ

        英文(新加坡):values-en-rSG

        英文(南非):values-en-rZA

        阿拉伯文(埃及):values-ar-rEG

        阿拉伯文(以色列):values-ar-rIL

        保加利亚文:  values-bg-rBG

        加泰罗尼亚文:values-ca-rES

        捷克文:values-cs-rCZ

        丹麦文:values-da-rDK

        德文(奥地利):values-de-rAT

        德文(瑞士):values-de-rCH

        德文(德国):values-de-rDE

        德文(列支敦士登):values-de-rLI

        希腊文:values-el-rGR

        西班牙文(西班牙):values-es-rES

        西班牙文(美国):values-es-rUS

        芬兰文(芬兰):values-fi-rFI

        法文(比利时):values-fr-rBE

        法文(加拿大):values-fr-rCA

        法文(瑞士):values-fr-rCH

        法文(法国):values-fr-rFR

        希伯来文:values-iw-rIL

        印地文:values-hi-rIN

        克罗里亚文:values-hr-rHR

        匈牙利文:values-hu-rHU

        印度尼西亚文:values-in-rID

        意大利文(瑞士):values-it-rCH

        意大利文(意大利):values-it-rIT

        日文:values-ja-rJP

        韩文:values-ko-rKR

        立陶宛文:valueslt-rLT

        拉脱维亚文:values-lv-rLV

        挪威博克马尔文:values-nb-rNO

        荷兰文(比利时):values-nl-BE

        荷兰文(荷兰):values-nl-rNL

        波兰文:values-pl-rPL

        葡萄牙文(巴西):values-pt-rBR

        葡萄牙文(葡萄牙):values-pt-rPT

        罗马尼亚文:values-ro-rRO

        俄文:values-ru-rRU

        斯洛伐克文:values-sk-rSK

        斯洛文尼亚文:values-sl-rSI

        塞尔维亚文:values-sr-rRS

        瑞典文:values-sv-rSE

        泰文:values-th-rTH

        塔加洛语:values-tl-rPH

        土耳其文:values--r-rTR

        乌克兰文:values-uk-rUA

        越南文:values-vi-rVN

    事实上可以通过程序获取JAVA所有支持的语言和国家的,程序代码如下:

import java.util.Locale;
public class Test {
 public static void main(String[] args) {
  //调用Locale类的getAvailableLocales方法获取,该方法返回一个数组,其中包含JAVA支持的语言和国家
  Locale[] localeList = Locale.getAvailableLocales();
  //依次获取所支持的国家和语言
  for (int i = 0; i < localeList.length; i++) {
   System.out.println(localeList[i].getDisplayCountry() + "="
     + localeList[i].getCountry() + ""
     + localeList[i].getDisplayLanguage() + "="
     + localeList[i].getLanguage());
  }
 }
}

    输出结果如下(有些国家可能会使用多种语言):

马来西亚=MY马来文=ms
卡塔尔=QA阿拉伯文=ar
冰岛=IS冰岛文=is
芬兰=FI芬兰文=fi
=波兰文=pl
马耳他=MT英文=en
瑞士=CH意大利文=it
比利时=BE荷兰文=nl
沙特阿拉伯=SA阿拉伯文=ar
伊拉克=IQ阿拉伯文=ar
波多黎哥=PR西班牙文=es
智利=CL西班牙文=es
=芬兰文=fi
奥地利=AT德文=de
=丹麦文=da
英国=GB英文=en
巴拿马=PA西班牙文=es
=塞尔维亚文=sr
也门=YE阿拉伯文=ar
马其顿王国=MK马其顿文=mk
=马其顿文=mk
加拿大=CA英文=en
越南=VN越南文=vi
荷兰=NL荷兰文=nl
美国=US西班牙文=es
中国=CN中文=zh
洪都拉斯=HN西班牙文=es
美国=US英文=en
=法文=fr
=泰文=th
=阿拉伯文=ar
摩洛哥=MA阿拉伯文=ar
=拉托维亚文(列托)=lv
=德文=de
印度尼西亚=ID印度尼西亚文=in
=克罗地亚文=hr
南非=ZA英文=en
韩国=KR朝鲜文=ko
突尼斯=TN阿拉伯文=ar
=印度尼西亚文=in
=日文=ja
塞尔维亚=RS塞尔维亚文=sr
白俄罗斯=BY白俄罗斯文=be
台湾地区=TW中文=zh
苏丹=SD阿拉伯文=ar
=葡萄牙文=pt
=冰岛文=is
日本=JP日文=ja
玻利维亚=BO西班牙文=es
阿尔及利亚=DZ阿拉伯文=ar
=马来文=ms
阿根廷=AR西班牙文=es
阿拉伯联合酋长国=AE阿拉伯文=ar
加拿大=CA法文=fr
=斯洛文尼亚文=sl
=西班牙文=es
立陶宛=LT立陶宛文=lt
黑山=ME塞尔维亚文=sr
叙利亚=SY阿拉伯文=ar
俄罗斯=RU俄文=ru
比利时=BE法文=fr
西班牙=ES西班牙文=es
=保加利亚文=bg
以色列=IL希伯来文=iw
=瑞典文=sv
=英文=en
=希伯来文=iw
丹麦=DK丹麦文=da
哥斯达黎加=CR西班牙文=es
香港=HK中文=zh
=中文=zh
西班牙=ES加泰罗尼亚文=ca
泰国=TH泰文=th
乌克兰=UA乌克兰文=uk
多米尼加共和国=DO西班牙文=es
委内瑞拉=VE西班牙文=es
波兰=PL波兰文=pl
利比亚=LY阿拉伯文=ar
约旦=JO阿拉伯文=ar
=意大利文=it
=乌克兰文=uk
匈牙利=HU匈牙利文=hu
=爱尔兰文=ga
危地马拉=GT西班牙文=es
巴拉圭=PY西班牙文=es
保加利亚=BG保加利亚文=bg
克罗地亚=HR克罗地亚文=hr
波斯尼亚和黑山共和国=BA塞尔维亚文=sr
罗马尼亚=RO罗马尼亚文=ro
卢森堡=LU法文=fr
=挪威文=no
=立陶宛文=lt
新加坡=SG英文=en
厄瓜多尔=EC西班牙文=es
波斯尼亚和黑山共和国=BA塞尔维亚文=sr
尼加拉瓜=NI西班牙文=es
=斯洛伐克文=sk
=俄文=ru
=马耳他文=mt
萨尔瓦多=SV西班牙文=es
=荷兰文=nl
印度=IN印地文=hi
=爱沙尼亚文=et
希腊=GR希腊文=el
斯洛文尼亚=SI斯洛文尼亚文=sl
意大利=IT意大利文=it
日本=JP日文=ja
卢森堡=LU德文=de
瑞士=CH法文=fr
马耳他=MT马耳他文=mt
巴林=BH阿拉伯文=ar
=阿尔巴尼亚文=sq
=越南文=vi
黑山=ME塞尔维亚文=sr
巴西=BR葡萄牙文=pt
挪威=NO挪威文=no
=希腊文=el
瑞士=CH德文=de
新加坡=SG中文=zh
科威特=KW阿拉伯文=ar
埃及=EG阿拉伯文=ar
爱尔兰=IE爱尔兰文=ga
秘鲁=PE西班牙文=es
捷克共和国=CZ捷克文=cs
土耳其=TR土耳其文=tr
=捷克文=cs
乌拉圭=UY西班牙文=es
爱尔兰=IE英文=en
印度=IN英文=en
阿曼=OM阿拉伯文=ar
塞尔维亚及黑山=CS塞尔维亚文=sr
=加泰罗尼亚文=ca
=白俄罗斯文=be
=塞尔维亚文=sr
=朝鲜文=ko
阿尔巴尼亚=AL阿尔巴尼亚文=sq
葡萄牙=PT葡萄牙文=pt
拉脱维亚=LV拉托维亚文(列托)=lv
塞尔维亚=RS塞尔维亚文=sr
斯洛伐克=SK斯洛伐克文=sk
墨西哥=MX西班牙文=es
澳大利亚=AU英文=en
挪威=NO挪威文=no
新西兰=NZ英文=en
瑞典=SE瑞典文=sv
=罗马尼亚文=ro
黎巴嫩=LB阿拉伯文=ar
德国=DE德文=de
泰国=TH泰文=th
=土耳其文=tr
哥伦比亚=CO西班牙文=es
菲律宾=PH英文=en
爱沙尼亚=EE爱沙尼亚文=et
塞浦路斯=CY希腊文=el
=匈牙利文=hu
法国=FR法文=fr

    在做项目时,我遇到这样一种情况,字符串资源需要JAVA代码来控制,而不是在XML文件中获取,下面提供该方法:

    创建ConstantBase类来定义你需要的字符串,如下:

package com.example.l10ntest;
public class ConstantBase {
 public String PUT_NAME;
 public String SAY_LOVE;
 public String NEXT;
 public String Second_Page;
}

    中英文的字符串类分别命名如下,中文:

package com.example.l10ntest;
public class ConstantCH extends ConstantBase {
 public ConstantCH() {
  //此处定义需要的中文字符串
  PUT_NAME = "请输入你的名字。";
  SAY_LOVE = "我爱你";
  NEXT = "下一页";
  Second_Page = "第二页";
 }
}

    英文:

package com.example.l10ntest;
public class ConstantEN extends ConstantBase {
 public ConstantEN() {
  // 此处定义需要的英文字符串
  PUT_NAME = "Please put your name.";
  SAY_LOVE = "I love you";
  NEXT = "next";
  Second_Page = "The second page";
 }
}

    我在新创建的Constant类中写了本地语言的判断方法,该类如下:

package com.example.l10ntest;
import java.util.Locale;
public class Constant {
 public static ConstantBase string = produceConstant();
 public static ConstantBase produceConstant() {
  if (isLocalLanguageZH()) {
   string = new ConstantCH();
  } else {
   string = new ConstantEN();
  }
  return string;
 }
 //判断本地语言是否是中文
 public static boolean isLocalLanguageZH() {
  if (Locale.getDefault().getLanguage().contains("zh")
    && Locale.getDefault().getCountry().contains("CN")) {
   return true;
  }
  return false;
 }
}

    主类的XML文件做一下修改(暂时将ImageView屏蔽掉,后面增加了一个Button),如下:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center"
    android:orientation="vertical" >
    <!-- <ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:contentDescription="@null"
        android:src="@drawable/country" /> -->
      
    <!-- tools:ignore="ContentDescription",屏蔽掉对View设置备注说明的提示 -->
    <TextView
        android:id="@+id/putName"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@null"
        android:textSize="20sp" />
    <TextView
        android:id="@+id/iloveyou"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@null"
        android:textSize="20sp" />
    <Button
        android:id="@+id/next"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
</LinearLayout>

    在主类中,写如下代码:

((TextView) findViewById(R.id.putName))
    .setText(Constant.string.PUT_NAME);
  ((TextView) findViewById(R.id.iloveyou))
    .setText(Constant.string.SAY_LOVE);

    这样便可以通过JAVA代码来控制字符串的获取了。但是发现一个问题,就是当程序切换到后台时,你将手机终端的语言修改,比如当前中文修改为英文,再从后台将该程序拉起时,显示的文本信息并不会随着变换,如果以前是中文,修改系统语言后还是中文,国际化似乎不生效。最后发现只有将该程序进程杀死,再次启动,国际化才会生效。但是每次切换一下终端语言就人为杀进程重启应用,显然是不合理的。

    事实上当locale信息改变之后,系统会发广播消息Intent.ACTION_LOCALE_CHANGED,我们可以写接收该广播的代码,并做一些操作。

    不妨多写几个Activity页面做测试,我写了两个,MainActivity经过修改的XML前面已经给出了,MainActivity如下:

package com.example.l10ntest;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
public class MainActivity extends BaseActivity {
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  ((TextView) findViewById(R.id.putName))
    .setText(Constant.string.PUT_NAME);
  ((TextView) findViewById(R.id.iloveyou))
    .setText(Constant.string.SAY_LOVE);
  Button btn = (Button) findViewById(R.id.next);
  btn.setText(Constant.string.NEXT);
  btn.setOnClickListener(new OnClickListener() {
   
   @Override
   public void onClick(View v) {
    startActivity(new Intent(MainActivity.this,SecondActivity.class));
   }
  });
 }
}

    第二个Activity(SecondActivity)我没有写XML布局,代码如下:

package com.example.l10ntest;
import android.os.Bundle;
import android.view.Gravity;
import android.widget.TextView;
public class SecondActivity extends BaseActivity {
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  TextView textView = new TextView(this);
  textView.setGravity(Gravity.CENTER);
  textView.setText(Constant.string.Second_Page);
  textView.setTextSize(30);
  setContentView(textView);
 }
}

    事实上一个应用一般不会只有一个Activity的,我将两个Activity都继承了BaseActivity,在BaseActivity中对这两个Activity做统一管理,BaseActivity如下:

package com.example.l10ntest;
import android.app.Activity;
import android.os.Bundle;
public class BaseActivity extends Activity {
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  Utils.addToList(this);
 }
}

    只调用了Utils类里的一个方法,在Activity创建后,将它添加到List里面。Utils类代码如下:

package com.example.l10ntest;
import java.util.ArrayList;
import java.util.Iterator;
import android.app.Activity;
import android.util.Log;
public class Utils {
 private static String TAG = "Utils";
 // 创建存放Activity的集合
 private static ArrayList<Activity> activityList = new ArrayList<Activity>();
 public static void addToList(Activity activity) {
  Log.i(TAG, "activity name:" + activity.getClass().getName());
  activityList.add(activity);
 }
 public static void stopApp() {
  Iterator<Activity> itr = activityList.iterator();
  Activity a = null;
  while (itr.hasNext()) {
   a = itr.next();
   a.finish();
  }
  Log.i(TAG, "stopApp end.");
  System.exit(0);
 }
}

    这里面有两个方法,一个就是添加Activity做统一管理,第二个是停止程序的方法,先将List中的Activity依次杀死,再杀死进程退出程序。

    下面就是比较重要的接收系统广播的类了,代码如下:

package com.example.l10ntest;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
public class AppExitReceiver extends BroadcastReceiver {
 @Override
 public void onReceive(Context context, Intent intent) {
  if (intent == null || intent.getAction() == null)
   return;
  if (intent.getAction().equals(Intent.ACTION_LOCALE_CHANGED)) {
   Utils.stopApp();
  }
 }
}

    AndroidManifest.xml文件如下,配置了一个广播接收。

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.l10ntest"
    android:versionCode="1"
    android:versionName="1.0" >
    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="17" />
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.l10ntest.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:name="com.example.l10ntest.SecondActivity"
            android:label="@string/app_name" >
        </activity>
        <receiver android:name="com.example.l10ntest.AppExitReceiver" >
            <intent-filter>
                <action android:name="android.intent.action.LOCALE_CHANGED" />
            </intent-filter>
        </receiver>
    </application>
</manifest>

    整理一下思路:

    写了两个Activity,这两个Activity都用到了文本资源,如“((TextView) findViewById(R.id.iloveyou)).setText(Constant.string.SAY_LOVE);”和“textView.setText(Constant.string.Second_Page);”MainActivity中增加了跳转到SecondActivity的按钮。两个Activity都继承了BaseActivity,当它们创建后,会通过BaseActivity中的“Utils.addToList(this)”将它们依次添加到Utils类中的activityList里。

    在终端切换系统语言,发送Intent.ACTION_LOCALE_CHANGED广播后,我们创建AppExitReceiver类来接收该广播,并调用“Utils.stopApp()“来将创建的Activity依次关闭,并且退出应用程序,这样不管该程序从后台切回来还是点击该程序的图标,程序都会重新启动,显示的文本信息会随着系统语言的变换而变换。

你可能感兴趣的:(java,android,国际化)