Android实现应用内置语言切换功能

发布时间 - 2026-01-10 22:56:35    点击率:

一、需求

有时候应用需要在内部切换语言但又不影响系统的语言,比如是应用现在是中文的,系统语言也是中文的,我把应用的切换成英文显示后系统语言还是中文的,系统语言切换后也不会被改变,还有就是有些机的系统是被改造精简过的,比如有些国产机的系统的语言就被精简剩中文和英文。支付宝、微信、Top Story都有在应用内部设置语言切换这样的功能。

二、实现效果

先看看实现效果吧。

三、实现

(一)添加多种语言的资源文件夹及文件

我这默认是英语再添加了个俄文(Google翻译的)和中文。

values/strings.xml

<resources> 
  <string name="app_name">SwitchLanguage</string> 
  <string name="helloworld">Hello World!</string> 
  <string name="language">Eng</string> 
  <string name="english">English</string> 
  <string name="chinese">中文</string> 
  <string name="russian">русский</string> 
  <string name="secondact">Second Activity</string> 
</resources> 

values-ru/strings.xml

<resources> 
  <string name="app_name">Переключение язык</string> 
  <string name="helloworld">привет мир!</string> 
  <string name="language">русский</string> 
  <string name="secondact">второй активность</string> 
</resources> 

values-zh/strings.xml

<resources> 
  <string name="app_name">切换语言</string> 
  <string name="helloworld">你好 世界!</string> 
  <string name="language">中文</string> 
  <string name="secondact">第二屏</string> 
</resources> 

(二)布局文件
activity_main.xml

默认标题栏的文字是切换语言后是不会被改变的,使用Toobar替换掉就可以了。

<?xml version="1.0" encoding="utf-8"?> 
<LinearLayout 
  android:id="@+id/activity_main" 
  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:orientation="vertical" 
  tools:context="com.ce.switchlanguage.MainActivity" 
  xmlns:app="http://schemas.android.com/apk/res-auto"> 
  <android.support.design.widget.AppBarLayout 
    android:layout_height="wrap_content" 
    android:layout_width="match_parent" 
    android:theme="@style/AppTheme.AppBarOverlay"> 
 
    <android.support.v7.widget.Toolbar 
      android:id="@+id/toolbar" 
      android:layout_width="match_parent" 
      android:layout_height="?attr/actionBarSize" 
      android:background="?attr/colorPrimary" 
      app:popupTheme="@style/AppTheme.PopupOverlay" 
      app:title="@string/app_name"/> 
 
  </android.support.design.widget.AppBarLayout> 
  <TextView 
    android:layout_width="wrap_content" 
    android:layout_height="wrap_content" 
    android:text="@string/helloworld"/> 
  <Button 
    android:id="@+id/change" 
    android:layout_width="wrap_content" 
    android:layout_height="wrap_content" 
    android:layout_gravity="center" 
    android:text="@string/language"/> 
</LinearLayout> 

styles.xml

设置语言后需要重启下activity,启动会有个效果,使用windowDisablePreview屏蔽掉它。

<resources> 
  <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar"> 
    <item name="colorPrimary">@color/colorPrimary</item> 
    <item name="colorPrimaryDark">@color/colorPrimaryDark</item> 
    <item name="colorAccent">@color/colorAccent</item> 
    <item name="windowActionBar">false</item> 
    <item name="windowNoTitle">true</item> 
    <item name="android:windowDisablePreview">true</item> 
  </style> 
  <style name="AppTheme.AppBarOverlay" parent="ThemeOverlay.AppCompat.Dark.ActionBar"/> 
  <style name="AppTheme.PopupOverlay" parent="ThemeOverlay.AppCompat.Dark"/> 
</resources> 

main_menu.xml

<?xml version="1.0" encoding="utf-8"?> 
<menu xmlns:android="http://schemas.android.com/apk/res/android" 
   xmlns:app="http://schemas.android.com/apk/res-auto"> 
  <item android:id="@+id/chinese" android:title="@string/chinese" app:showAsAction="never" /> 
  <item android:id="@+id/english" android:title="@string/english" app:showAsAction="never" /> 
  <item android:id="@+id/russian" android:title="@string/russian" app:showAsAction="never" /> 
</menu> 

(三)LocaleUtils

package com.ce.switchlanguage; 
 
import android.content.Context; 
import android.content.SharedPreferences; 
import android.content.res.Configuration; 
import android.os.Build; 
import android.util.DisplayMetrics; 
 
import com.google.gson.Gson; 
 
import java.util.Locale; 
 
public class LocaleUtils { 
  /** 
   * 中文 
   */ 
  public static final Locale LOCALE_CHINESE = Locale.CHINESE; 
  /** 
   * 英文 
   */ 
  public static final Locale LOCALE_ENGLISH = Locale.ENGLISH; 
  /** 
   * 俄文 
   */ 
  public static final Locale LOCALE_RUSSIAN = new Locale("ru"); 
  /** 
   * 保存SharedPreferences的文件名 
   */ 
  private static final String LOCALE_FILE = "LOCALE_FILE"; 
  /** 
   * 保存Locale的key 
   */ 
  private static final String LOCALE_KEY = "LOCALE_KEY"; 
 
  /** 
   * 获取用户设置的Locale 
   * @param pContext Context 
   * @return Locale 
   */ 
  public static Locale getUserLocale(Context pContext) { 
    SharedPreferences _SpLocale = pContext.getSharedPreferences(LOCALE_FILE, Context.MODE_PRIVATE); 
    String _LocaleJson = _SpLocale.getString(LOCALE_KEY, ""); 
    return jsonToLocale(_LocaleJson); 
  } 
  /** 
   * 获取当前的Locale 
   * @param pContext Context 
   * @return Locale 
   */ 
  public static Locale getCurrentLocale(Context pContext) { 
    Locale _Locale; 
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { //7.0有多语言设置获取顶部的语言 
      _Locale = pContext.getResources().getConfiguration().getLocales().get(0); 
    } else { 
      _Locale = pContext.getResources().getConfiguration().locale; 
    } 
    return _Locale; 
  } 
  /** 
   * 保存用户设置的Locale 
   * @param pContext Context 
   * @param pUserLocale Locale 
   */ 
  public static void saveUserLocale(Context pContext, Locale pUserLocale) { 
    SharedPreferences _SpLocal=pContext.getSharedPreferences(LOCALE_FILE, Context.MODE_PRIVATE); 
    SharedPreferences.Editor _Edit=_SpLocal.edit(); 
    String _LocaleJson = localeToJson(pUserLocale); 
    _Edit.putString(LOCALE_KEY, _LocaleJson); 
    _Edit.apply(); 
  } 
  /** 
   * Locale转成json 
   * @param pUserLocale UserLocale 
   * @return json String 
   */ 
  private static String localeToJson(Locale pUserLocale) { 
    Gson _Gson = new Gson(); 
    return _Gson.toJson(pUserLocale); 
  } 
  /** 
   * json转成Locale 
   * @param pLocaleJson LocaleJson 
   * @return Locale 
   */ 
  private static Locale jsonToLocale(String pLocaleJson) { 
    Gson _Gson = new Gson(); 
    return _Gson.fromJson(pLocaleJson, Locale.class); 
  } 
  /** 
   * 更新Locale 
   * @param pContext Context 
   * @param pNewUserLocale New User Locale 
   */ 
  public static void updateLocale(Context pContext, Locale pNewUserLocale) { 
    if (needUpdateLocale(pContext, pNewUserLocale)) { 
      Configuration _Configuration = pContext.getResources().getConfiguration(); 
      if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) { 
        _Configuration.setLocale(pNewUserLocale); 
      } else { 
        _Configuration.locale =pNewUserLocale; 
      } 
      DisplayMetrics _DisplayMetrics = pContext.getResources().getDisplayMetrics(); 
      pContext.getResources().updateConfiguration(_Configuration, _DisplayMetrics); 
      saveUserLocale(pContext, pNewUserLocale); 
    } 
  } 
  /** 
   * 判断需不需要更新 
   * @param pContext Context 
   * @param pNewUserLocale New User Locale 
   * @return true / false 
   */ 
  public static boolean needUpdateLocale(Context pContext, Locale pNewUserLocale) { 
    return pNewUserLocale != null && !getCurrentLocale(pContext).equals(pNewUserLocale); 
  } 
} 

Locale工具类,这里我用SharedPreferences来保存所设置的Locale,Locale是实现了Serializable的。

(四)Activity

package com.ce.switchlanguage; 
 
import android.content.Intent; 
import android.os.Bundle; 
import android.support.v7.app.AppCompatActivity; 
import android.support.v7.widget.Toolbar; 
import android.view.Menu; 
import android.view.MenuItem; 
 
public class MainActivity extends AppCompatActivity { 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 
    Toolbar _Toolbar =(Toolbar) findViewById(R.id.toolbar); 
    setSupportActionBar(_Toolbar); 
  } 
 
  @Override 
  public boolean onCreateOptionsMenu(Menu menu) { 
    getMenuInflater().inflate(R.menu.main_menu,menu); 
    return true; 
  } 
 
  @Override 
  public boolean onOptionsItemSelected(MenuItem item) { 
    int _ItemId=item.getItemId(); 
    switch (_ItemId) { 
      case R.id.chinese: 
        if (LocaleUtils.needUpdateLocale(this, LocaleUtils.LOCALE_CHINESE)) { 
          LocaleUtils.updateLocale(this, LocaleUtils.LOCALE_CHINESE); 
          restartAct(); 
        } 
        break; 
      case R.id.english: 
        if (LocaleUtils.needUpdateLocale(this, LocaleUtils.LOCALE_ENGLISH)) { 
          LocaleUtils.updateLocale(this, LocaleUtils.LOCALE_ENGLISH); 
          restartAct(); 
        } 
        break; 
      case R.id.russian: 
        if (LocaleUtils.needUpdateLocale(this, LocaleUtils.LOCALE_RUSSIAN)) { 
          LocaleUtils.updateLocale(this, LocaleUtils.LOCALE_RUSSIAN); 
          restartAct(); 
        } 
    } 
    return true; 
  } 
 
  /** 
   * 重启当前Activity 
   */ 
  private void restartAct() { 
    finish(); 
    Intent _Intent = new Intent(this, MainActivity.class); 
    startActivity(_Intent); 
    //清除Activity退出和进入的动画 
    overridePendingTransition(0, 0); 
  } 
} 

这里只有一个Activity所以切换的时候重启下当前Activity就好了,栈里还有其他Activity的自己再处理吧。

(五)Application

package com.ce.switchlanguage; 
 
import android.app.Application; 
import android.content.res.Configuration; 
import android.os.Build; 
 
import java.util.Locale; 
 
public class MyApplication extends Application { 
  @Override 
  public void onCreate() { 
    super.onCreate(); 
    Locale _UserLocale=LocaleUtils.getUserLocale(this); 
    LocaleUtils.updateLocale(this, _UserLocale); 
  } 
 
  @Override 
  public void onConfigurationChanged(Configuration newConfig) { 
    super.onConfigurationChanged(newConfig); 
    Locale _UserLocale=LocaleUtils.getUserLocale(this); 
    //系统语言改变了应用保持之前设置的语言 
    if (_UserLocale != null) { 
      Locale.setDefault(_UserLocale); 
      Configuration _Configuration = new Configuration(newConfig); 
      if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) { 
        _Configuration.setLocale(_UserLocale); 
      } else { 
        _Configuration.locale =_UserLocale; 
      } 
      getResources().updateConfiguration(_Configuration, getResources().getDisplayMetrics()); 
    } 
  } 
} 

在Application onCreate的时候更新下,在系统语言改变的时候也要保持之前设置的语言不变。

源码地址:Android应用内置语言切换

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# Android应用内置语言切换  # Android语言切换  # Android内置语言切换  # Android实现app应用多语言切换功能  # Android app应用多语言切换功能实现  # android动态设置app当前运行语言的方法  # 详解Android更改APP语言模式的实现过程  # android 使用kotlin 实现点击更换全局语言(中日英切换)  # Android 7.0以上版本实现应用内语言切换的方法  # Android 系统语言切换监听和设置实例代码  # Android实现系统语言切换功能  # Android App中进行语言的切换  # 俄文  # 英文  # 重启  # 转成  # 都有  # 有个  # 也要  # 不需要  # 我这  # 我把  # 有多  # 我用  # 只有一个  # 英语  # 又不  # 支付宝  # 大家多多  # 就可以  # 再添  # 标题栏 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?  Windows10电脑怎么设置虚拟光驱_Win10右键装载ISO镜像文件  如何快速搭建二级域名独立网站?  Laravel怎么判断请求类型_Laravel Request isMethod用法  深圳网站制作平台,深圳市做网站好的公司有哪些?  微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】  香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南  佛山企业网站制作公司有哪些,沟通100网上服务官网?  如何在宝塔面板中创建新站点?  javascript基于原型链的继承及call和apply函数用法分析  Laravel的HTTP客户端怎么用_Laravel HTTP Client发起API请求教程  奇安信“盘古石”团队突破 iOS 26.1 提权  大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?  如何用AWS免费套餐快速搭建高效网站?  Laravel Docker环境搭建教程_Laravel Sail使用指南  Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全  Laravel DB事务怎么使用_Laravel数据库事务回滚操作  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  Windows10怎样连接蓝牙设备_Windows10蓝牙连接步骤【教程】  javascript事件捕获机制【深入分析IE和DOM中的事件模型】  微信小程序 input输入框控件详解及实例(多种示例)  如何在七牛云存储上搭建网站并设置自定义域名?  宙斯浏览器怎么屏蔽图片浏览 节省手机流量使用设置方法  千库网官网入口推荐 千库网设计创意平台入口  Laravel队列任务超时怎么办_Laravel Queue Timeout设置详解  html5audio标签播放结束怎么触发事件_onended回调方法【教程】  原生JS获取元素集合的子元素宽度实例  今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】  Laravel如何保护应用免受CSRF攻击?(原理和示例)  香港网站服务器数量如何影响SEO优化效果?  瓜子二手车官方网站在线入口 瓜子二手车网页版官网通道入口  怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?  ChatGPT回答中断怎么办 引导AI继续输出完整内容的方法  php结合redis实现高并发下的抢购、秒杀功能的实例  Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程  韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南  Laravel如何处理异常和错误?(Handler示例)  新三国志曹操传主线渭水交兵攻略  Laravel怎么调用外部API_Laravel Http Client客户端使用  如何确认建站备案号应放置的具体位置?  Laravel Eloquent模型如何创建_Laravel ORM基础之Model创建与使用教程  jimdo怎样用html5做选项卡_jimdo选项卡html5实现与切换效果【指南】  JavaScript模板引擎Template.js使用详解  javascript基本数据类型及类型检测常用方法小结  PHP 实现电台节目表的智能时间匹配与今日/明日轮播逻辑  html如何与html链接_实现多个HTML页面互相链接【互相】  DeepSeek是免费使用的吗 DeepSeek收费模式与Pro版本功能详解  jQuery中的100个技巧汇总  PHP正则匹配日期和时间(时间戳转换)的实例代码