Kotlin有一个很棒的功能叫做字符串模板。
val i = 10
val s = "i = $i" // evaluates to "i = 10"
但是在模板中是否可以使用任何格式?例如,我想在Kotlin的字符串模板中格式化Double类型,至少设置小数点后的位数:
val pi = 3.14159265358979323
val s = "pi = $pi??" // How to make it "pi = 3.14"?
Kotlin有一个很棒的功能叫做字符串模板。
val i = 10
val s = "i = $i" // evaluates to "i = 10"
但是在模板中是否可以使用任何格式?例如,我想在Kotlin的字符串模板中格式化Double类型,至少设置小数点后的位数:
val pi = 3.14159265358979323
val s = "pi = $pi??" // How to make it "pi = 3.14"?
不幸的是,目前字符串模板中没有内置的格式化支持。作为解决方法,您可以使用类似以下的方式:
"pi = ${pi.format(2)}"
你需要自己定义的.format(n)
函数。
fun Double.format(digits: Int) = "%.${digits}f".format(this)
这仅适用于Kotlin/JVM。
显然,Kotlin目前缺少一些功能,我们将进行修复。
String.format()
的包装器)。val pi = 3.14159265358979323
val s = "pi = %.2f".format(pi)
Kotlin 的 String 类现在有一个 format 函数,它内部使用 Java 的 String.format
方法:
/**
* Uses this string as a format string and returns a string obtained by substituting the specified arguments,
* using the default locale.
*/
@kotlin.internal.InlineOnly
public inline fun String.Companion.format(format: String, vararg args: Any?): String = java.lang.String.format(format, *args)
使用方法
val pi = 3.14159265358979323
val formatted = String.format("%.2f", pi) ;
println(formatted)
>>3.14
很简单,使用:
val str: String = "%.2f".format(3.14159)
由于String.format
只是一个扩展函数(请参见此处),它在内部调用了java.lang.String.format
,如果您需要更多的灵活性,可以使用Java的DecimalFormat编写自己的扩展函数:
fun Double.format(fracDigits: Int): String {
val df = DecimalFormat()
df.setMaximumFractionDigits(fracDigits)
return df.format(this)
}
println(3.14159.format(2)) // 3.14
例如:
infix fun Double.f(fmt: String) = "%$fmt".format(this)
infix fun Double.f(fmt: Float) = "%${if (fmt < 1) fmt + 1 else fmt}f".format(this)
val pi = 3.14159265358979323
println("""pi = ${pi f ".2f"}""")
println("pi = ${pi f .2f}")
Kotlin 1.7.2
String的format
方法接受一个格式化字符串和任意数量的可为空的Any类型参数。
fun String.Companion.format(format: String, vararg args: Any?) : String
真正的代码可能看起来像这样:
fun main() {
val pi: Double = 3.141592653589
val gravity: Double = 9.80665
var str = String.Companion.format("%.2f %n%.2f", pi, gravity)
println(str)
}
%n
代表换行符,%f
代表十进制浮点数。
结果如下:
// 3.14
// 9.81
作为一个多平台的简单解决方案,您可以使用以下方法:
import kotlin.math.abs
import kotlin.math.pow
fun Number.simpleFormat(numberDigitsAfterSeparator: Int = 0, decimalSeparator: Char = ','): String {
if(numberDigitsAfterSeparator < 0)
throw IllegalArgumentException("numberDigitsAfterSeparator should be >= 0 but is $numberDigitsAfterSeparator")
val prefix = this.toInt()
if(numberDigitsAfterSeparator == 0)return "$prefix"
val sign = if(this.toDouble() >= 0.0) "" else "-"
val afterSeparatorPart = abs(this.toDouble() - prefix)
val suffixInt = (10.0.pow(numberDigitsAfterSeparator) * afterSeparatorPart).toInt()
val suffix = if(afterSeparatorPart >= 1.0) "$suffixInt" else addNullsBefore(suffixInt, numberDigitsAfterSeparator)
return "$sign${abs(prefix)}$decimalSeparator$suffix"
}
fun addNullsBefore(suffixInt: Int, numberDigitsAfterSeparator: Int): String {
var s = "$suffixInt"
val len = s.length
repeat(numberDigitsAfterSeparator - len) { _ -> s = "0$s" }
return s
}
测试:
fun main() {
println((-0.00001).simpleFormat(5))
println(3.1415927.simpleFormat(2))
println(3.99.simpleFormat(2))
println((-3.99).simpleFormat(2))
println(3.1415927.simpleFormat())
println(3.99.simpleFormat())
println((-3.99).simpleFormat())
println((-3.99).simpleFormat(-1))
}
带有输出:
-0,00001
3,14
3,99
-3,99
3
3
-3
Exception in thread "main" java.lang.IllegalArgumentException: numberDigitsAfterSeparator should be >= 0 but is -1..
这是一个针对Android TextView的Kotlin字符串格式化示例:
val format = String.format("<font color=#3177a3> test1: <b>%s</b><br> test2: <b>%s</b><br> test3: <b>%s</b></font>", "value1", "value2", "value3")
textView.text = format
它具有字符串格式化功能 示例:
fun printSum(a: Int, b: Int): Unit {
println("sum of $a and $b is ${a + b}")
}
.format
作为解决方法。不幸的是,在KMM中,String.format
仍然不可用,至少直到2022年。它不在stdlib.text
或kotlinx
中。字符串模板很酷,但是C风格格式化是基础,我认为复杂性可能基于变量列表的变化,但仍然令人失望。 - superarts.org