Java 在定义字符串的时候不支持字符串插值, 即不能在字符串中捕获作用域中的变量, 用来组成当前字符串. 而这可以说是其他各种言都具备的基本特性. 例如 Bash 中可以这样
bash-3.2$ name=Yanbin
bash-3.2$ echo "Hello $name"
Hello Yanbin
Java 中不支持类似的方式
String name = "Yanbin";
String greeting = "Hello $name"; //Java 中无法把 $name 替换为 name 变量值 "Yanbin"
而使用 Java 必须用 String 的静态方法 format (注意是静态方法哦) 来间接的格式化出一个字符串, 可以这么写
String.format("Hello %s", "World");
String.format("%1$s %1$s %s %s %2$s", "aa", 10); //aa aa aa 10 10
我们看到在用 format 格式化时匹配参数会是一个不小的问题, 相同使用法的打印输出方法是 System.out.printf()
. 别说字符串插值了, 如果 format 方法是 String 的一个实例方法应用起来都会便利些, 如 "Hello %s".format("World")
会容易读些, Scala 的字符串实例就有 format
方法了.
也是因为 Java 对字符串插值的缺失造成了 Java 字符串本身不俱备基本的模板功能, 最简单的字符串模板都须用串替换操作, 或者需要依赖像 Velocity, FreeMarker 这样的模板组件. 有点不太明白引入字符串插值会对 Java 语言造成什么危害, 字符串插值应该是编译器的功能, 执行性能上也会优于字符的格式化或连接操作.
我们来大概领略一下几个常见语言是怎么支持字符插值的.
Swift
1 2 |
let extra = "little" println("Mary had a \(extra) lamb.") |
Scala
1 2 3 4 5 |
scala> val extra = "little" extra: String = little scala> s"Mary had a $extra lamb." res24: String = Mary had a little lamb. |
C#, 有点类似于 Scala 的做法
1 2 |
var extra = "little"; Console.WriteLine($"Mary had a {extra} lamb."); |
Python, 它有多种形式, f, % 操作或 format/substitute 方法, 不过个人看下面这种更趁手些
1 2 3 4 5 6 7 8 9 |
>>> extra = "little" >>> "Mary had a %(extra)s lamb." % locals() 'Mary had a little lamb.' >>> f"Mary had a {extra} lamb" 'Mary had a little lamb. >>> "Mary had a {} lamb.".format(extra) 'Mary had a little lamb. >>> "Mary had a %s lamb." % extra 'Mary had a little lamb. |
Ruby
1 2 3 4 5 6 |
irb(main):001:0> extra = 'little' => "little" irb(main):002:0> "Mary had a #{extra} lamb." => "Mary had a little lamb." irb(main):003:0> "Mary had a %s lamb." % extra => "Mary had a little lamb." |
Clojure 与 Scala 一样也是调用字符串的实例 format 方法, PHP, Perl, Kotlin, Groovy 等都是与 Bash 是一样的.
参考: https://rosettacode.org/wiki/String_interpolation_(included)