Android

Android通过.nomedia文件禁止多媒体库扫描指定文件夹下的多媒体文件 Android应用内存泄露分析、改善经验总结 修改Eclipse导入项目的默认工程名 自定义Android Studio工程模板 使用Nexus Repository搭建属于自己公司的私有maven服务器 Android Studio编译过程中mergeDebugResources时报“png-cruncher_*”异常的解决方案 Eclipse转Android Studio的过程中有必要弄明白的一些问题 Android开发经验总结 Android Studio使用过程中遇到的一些问题及解决方案 Android各个Support Library介绍 调用AsyncTask的excute方法不能立即执行程序的原因分析及改善方案 提升进入界面的速度 使用软引用解决Handler内存泄露和显示Popupwindow、Dialog时提示"Unable to add Window-token is null"的问题 SharedPreferences在多进程中的使用及注意事项 Android性能测试工具列表 Android View双缓冲绘制时清除Bitmap上的内容的方法 解决JPinyin在APK被加密后不能正常使用的问题 Android APP内存优化之图片优化 Android EditText的使用及值得注意的地方 Android应用内多进程的使用及注意事项 Android设置应用内文字的默认颜色和大小 关于APK瘦身值得分享的一些经验 Android通过ClipDrawable实现图片裁剪功能 Android通过广播更新文件和文件夹到媒体库 每个Android开发者都应该了解的资源列表 selector的使用方法及注意事项 通过批处理批量clone代码 Android清除数据、清除缓存、一键清理的区别 Android将数据库保存到SD卡的实现 Android多分辨率适配经验总结 通过观察者模式监听媒体库的变化实现APP本地数据自动更新 Android ADB命令大全(通过ADB命令查看wifi密码、MAC地址、设备信息、操作文件、查看文件、日志信息、卸载、启动和安装APK等) Android通过ADB查看wifi密码 Android一个APK多个入口(多个桌面图标)的实现 使用Python脚本批量卸载第三方应用和清除log缓存 Android CheckList Android模仿打字机效果的自定义View实现 在Activity的onCreate方法中显示PopupWindow导致异常的原因分析及解决方案 Android手写优化-更为平滑的签名效果实现 Android手写优化-平滑的签名效果实现 不要在Android的Application对象中缓存数据! 大量Android面试题目来袭 一种不需要Google账号、不需要关联手机、不需要在手机上安装Google的服务直接能够下载Google Play上APK的方法 在Android library中不能使用switch-case语句访问资源ID的原因分析及解决方案 Android程序和数据分离的实现方案 按Home按键退出应用后重新启动该应用无法返回到最后打开页面的解决方案 Eclipse下Android项目不能生成R.java的解决方法汇总 android:descendantFocusability属性在ListView中的妙用 去掉SrollView、GrdiView、ListView、ViewPager等滑动到边缘的光晕效果 Android开发经验谈-Eclipse使用技巧 Android开发经验谈-很少有人会告诉你的Android开发基本常识 Android开发经验谈-Android工程目录介绍 在Android的string.xml中使用转义字符实现想要的显示效果 修改ViewPager调用setCurrentItem时,滑屏的速度 Android监听Home按键消息 Android手写开源项目和资料搜集 Android通过资源文件名获取资源ID Android中Bitmap、Drawable、bytes数组之间相互转换 想过但未实现的一些Idea 读写文件编码方式不一致导致文件乱码的解决方案 Android字符串格式化开源库phrase介绍 Android实现带箭头的自定义Progressbar Android模拟键盘输入功能的实现 与Android应用程序相关的各种文件存储路径介绍 Android开发者网址导航

标签

Android 65

Android通过.nomedia文件禁止多媒体库扫描指定文件夹下的多媒体文件 Android应用内存泄露分析、改善经验总结 修改Eclipse导入项目的默认工程名 自定义Android Studio工程模板 使用Nexus Repository搭建属于自己公司的私有maven服务器 Android Studio编译过程中mergeDebugResources时报“png-cruncher_*”异常的解决方案 Eclipse转Android Studio的过程中有必要弄明白的一些问题 Android开发经验总结 Android Studio使用过程中遇到的一些问题及解决方案 Android各个Support Library介绍 调用AsyncTask的excute方法不能立即执行程序的原因分析及改善方案 提升进入界面的速度 使用软引用解决Handler内存泄露和显示Popupwindow、Dialog时提示"Unable to add Window-token is null"的问题 SharedPreferences在多进程中的使用及注意事项 Android性能测试工具列表 Android View双缓冲绘制时清除Bitmap上的内容的方法 解决JPinyin在APK被加密后不能正常使用的问题 Android APP内存优化之图片优化 Android EditText的使用及值得注意的地方 Android应用内多进程的使用及注意事项 Android设置应用内文字的默认颜色和大小 关于APK瘦身值得分享的一些经验 Android通过ClipDrawable实现图片裁剪功能 Android通过广播更新文件和文件夹到媒体库 每个Android开发者都应该了解的资源列表 selector的使用方法及注意事项 通过批处理批量clone代码 Android清除数据、清除缓存、一键清理的区别 Android将数据库保存到SD卡的实现 Android多分辨率适配经验总结 通过观察者模式监听媒体库的变化实现APP本地数据自动更新 Android ADB命令大全(通过ADB命令查看wifi密码、MAC地址、设备信息、操作文件、查看文件、日志信息、卸载、启动和安装APK等) Android通过ADB查看wifi密码 Android一个APK多个入口(多个桌面图标)的实现 使用Python脚本批量卸载第三方应用和清除log缓存 Android CheckList Android模仿打字机效果的自定义View实现 在Activity的onCreate方法中显示PopupWindow导致异常的原因分析及解决方案 Android手写优化-更为平滑的签名效果实现 Android手写优化-平滑的签名效果实现 不要在Android的Application对象中缓存数据! 大量Android面试题目来袭 一种不需要Google账号、不需要关联手机、不需要在手机上安装Google的服务直接能够下载Google Play上APK的方法 在Android library中不能使用switch-case语句访问资源ID的原因分析及解决方案 Android程序和数据分离的实现方案 按Home按键退出应用后重新启动该应用无法返回到最后打开页面的解决方案 Eclipse下Android项目不能生成R.java的解决方法汇总 android:descendantFocusability属性在ListView中的妙用 去掉SrollView、GrdiView、ListView、ViewPager等滑动到边缘的光晕效果 Android开发经验谈-Eclipse使用技巧 Android开发经验谈-很少有人会告诉你的Android开发基本常识 Android开发经验谈-Android工程目录介绍 在Android的string.xml中使用转义字符实现想要的显示效果 修改ViewPager调用setCurrentItem时,滑屏的速度 Android监听Home按键消息 Android手写开源项目和资料搜集 Android通过资源文件名获取资源ID Android中Bitmap、Drawable、bytes数组之间相互转换 想过但未实现的一些Idea 读写文件编码方式不一致导致文件乱码的解决方案 Android字符串格式化开源库phrase介绍 Android实现带箭头的自定义Progressbar Android模拟键盘输入功能的实现 与Android应用程序相关的各种文件存储路径介绍 Android开发者网址导航

Android字符串格式化开源库phrase介绍

2014年11月30日

引言

Phrase是一个字符串格式化的开源项目,整个项目只有一个类Phrase.java,相比于String.format,通过phrase格式化字符串代码更具可读性。

phrase项目介绍

  • 源码:phrase项目的源代码很简单,里面总共只有一个类:Phrase.java,代码如下:

      package com.eebbk.learnspell.util;
    	
      import java.util.HashMap;
      import java.util.HashSet;
      import java.util.Map;
      import java.util.Set;
    	
      import android.app.Fragment;
      import android.content.Context;
      import android.content.res.Resources;
      import android.text.SpannableStringBuilder;
      import android.view.View;
    	
      public final class Phrase {
        private final CharSequence pattern;
    	
        private final Set<String> keys = new HashSet<String>();
        private final Map<String, CharSequence> keysToValues = new HashMap<String, CharSequence>();
    	
        private CharSequence formatted;
    	
        private Token head;
    	
        private char curChar;
        private int curCharIndex;
    	
        private static final int EOF = 0;
    	
        public static Phrase from(Fragment f, int patternResourceId) {
          return from(f.getResources(), patternResourceId);
        }
    	
        public static Phrase from(View v, int patternResourceId) {
          return from(v.getResources(), patternResourceId);
        }
    	
        public static Phrase from(Context c, int patternResourceId) {
          return from(c.getResources(), patternResourceId);
        }
    	
        public static Phrase from(Resources r, int patternResourceId) {
          return from(r.getText(patternResourceId));
        }
    	
        public static Phrase from(CharSequence pattern) {
          return new Phrase(pattern);
        }
    	
        public Phrase put(String key, CharSequence value) {
          if (!keys.contains(key)) {
            throw new IllegalArgumentException("Invalid key: " + key);
          }
          if (value == null) {
            throw new IllegalArgumentException("Null value for '" + key + "'");
          }
          keysToValues.put(key, value);
    	
          formatted = null;
          return this;
        }
    	
        public Phrase put(String key, int value) {
          if (!keys.contains(key)) {
            throw new IllegalArgumentException("Invalid key: " + key);
          }
          keysToValues.put(key, Integer.toString(value));
    	
          formatted = null;
          return this;
        }
    	
        public Phrase putOptional(String key, CharSequence value) {
          return keys.contains(key) ? put(key, value) : this;
        }
    	
        public Phrase putOptional(String key, int value) {
          return keys.contains(key) ? put(key, value) : this;
        }
    	
        public CharSequence format() {
          if (formatted == null) {
            if (!keysToValues.keySet().containsAll(keys)) {
              Set<String> missingKeys = new HashSet<String>(keys);
              missingKeys.removeAll(keysToValues.keySet());
              throw new IllegalArgumentException("Missing keys: " + missingKeys);
            }
    	
            SpannableStringBuilder sb = new SpannableStringBuilder(pattern);
            for (Token t = head; t != null; t = t.next) {
              t.expand(sb, keysToValues);
            }
    	
            formatted = sb;
          }
          return formatted;
        }
    	
        @Override
        public String toString() {
          return pattern.toString();
        }
    	
        private Phrase(CharSequence pattern) {
          curChar = (pattern.length() > 0) ? pattern.charAt(0) : EOF;
          this.pattern = pattern;
    	
          Token prev = null;
          Token next;
          while ((next = token(prev)) != null) {
            if (head == null) head = next;
            prev = next;
          }
        }
    	
        private Token token(Token prev) {
          if (curChar == EOF) {
            return null;
          }
          if (curChar == '{') {
            char nextChar = lookahead();
            if (nextChar == '{') {
              return leftCurlyBracket(prev);
            } else if (nextChar >= 'a' && nextChar <= 'z') {
              return key(prev);
            } else {
              throw new IllegalArgumentException(
                  "Unexpected character '" + nextChar + "'; expected key.");
            }
          }
          return text(prev);
        }
    	
        private KeyToken key(Token prev) {
          StringBuilder sb = new StringBuilder();
    	
          consume();
          while ((curChar >= 'a' && curChar <= 'z') || curChar == '_') {
            sb.append(curChar);
            consume();
          }
    	
          if (curChar != '}') {
            throw new IllegalArgumentException("Missing closing brace: }");
          }
          consume();
    	
          if (sb.length() == 0) {
            throw new IllegalArgumentException("Empty key: {}");
          }
    	
          String key = sb.toString();
          keys.add(key);
          return new KeyToken(prev, key);
        }
    	
        private TextToken text(Token prev) {
          int startIndex = curCharIndex;
    	
          while (curChar != '{' && curChar != EOF) {
            consume();
          }
          return new TextToken(prev, curCharIndex - startIndex);
        }
    	
        private LeftCurlyBracketToken leftCurlyBracket(Token prev) {
          consume();
          consume();
          return new LeftCurlyBracketToken(prev);
        }
    	
        private char lookahead() {
          return curCharIndex < pattern.length() - 1 ? pattern.charAt(curCharIndex + 1) : EOF;
        }
    	
        private void consume() {
          curCharIndex++;
          curChar = (curCharIndex == pattern.length()) ? EOF : pattern.charAt(curCharIndex);
        }
    	
        private abstract static class Token {
          private final Token prev;
          private Token next;
    	
          protected Token(Token prev) {
            this.prev = prev;
            if (prev != null) prev.next = this;
          }
    	
          abstract void expand(SpannableStringBuilder target, Map<String, CharSequence> data);
    	
          abstract int getFormattedLength();
    	
          final int getFormattedStart() {
            if (prev == null) {
              return 0;
            } else {
              return prev.getFormattedStart() + prev.getFormattedLength();
            }
          }
        }
    	
        private static class TextToken extends Token {
          private final int textLength;
    	
          TextToken(Token prev, int textLength) {
            super(prev);
            this.textLength = textLength;
          }
    	
          @Override void expand(SpannableStringBuilder target, Map<String, CharSequence> data) {
    	    	
          }
    	
          @Override int getFormattedLength() {
            return textLength;
          }
        }
    	
        private static class LeftCurlyBracketToken extends Token {
          LeftCurlyBracketToken(Token prev) {
            super(prev);
          }
    	
          @Override void expand(SpannableStringBuilder target, Map<String, CharSequence> data) {
            int start = getFormattedStart();
            target.replace(start, start + 2, "{");
          }
    	
          @Override int getFormattedLength() {
            return 1;
          }
        }
    	
        private static class KeyToken extends Token {
          private final String key;
    	
          private CharSequence value;
    	
          KeyToken(Token prev, String key) {
            super(prev);
            this.key = key;
          }
    	
          @Override void expand(SpannableStringBuilder target, Map<String, CharSequence> data) {
            value = data.get(key);
    	
            int replaceFrom = getFormattedStart();
            int replaceTo = replaceFrom + key.length() + 2;
            target.replace(replaceFrom, replaceTo, value);
          }
    	
          @Override int getFormattedLength() {
            return value.length();
          }
        }
      }
    
  • 字符串格式化原理
    通过阅读Phrase.java的代码可知,它用”{“和”}”将需要格式化的内容包起来,然后用键值对给需要改变的内容传值,包起来的内容为键,值为动态设置的内容,比如:

      "Hi {first_name}, you are {age} years old."  
    

我们要最终的显示内容为:“Hi UperOne, you are 26 years old.”这里的first_name和age是键,值为UperOne和26。

使用方法

使用起来很简单,通过键值对的方式插值:

	CharSequence parseStr = Phrase.from("Hi {first_name}, you are {age} years old.")
					 .put("first_name", "UperOne")
					 .put("age", "26")
					 .format();
			
	mParseTxt.setText( parseStr );