java.util.Formattable上的示例是否不正确?

Pok*_*u22 10 java formatting formatter formattable

使用给出java.util.Formattable示例(修改为实际设置构造函数中的值),事情似乎正常工作:

import java.nio.CharBuffer;
import java.util.Formatter;
import java.util.Formattable;
import java.util.Locale;
import static java.util.FormattableFlags.*;

public class StockName implements Formattable {
    private String symbol, companyName, frenchCompanyName;
    public StockName(String symbol, String companyName,
                     String frenchCompanyName) {
        this.symbol = symbol;
        this.companyName = companyName;
        this.frenchCompanyName = frenchCompanyName;
    }

    public void formatTo(Formatter fmt, int f, int width, int precision) {
        StringBuilder sb = new StringBuilder();

        // decide form of name
        String name = companyName;
        if (fmt.locale().equals(Locale.FRANCE))
            name = frenchCompanyName;
        boolean alternate = (f & ALTERNATE) == ALTERNATE;
        boolean usesymbol = alternate || (precision != -1 && precision < 10);
        String out = (usesymbol ? symbol : name);

        // apply precision
        if (precision == -1 || out.length() < precision) {
            // write it all
            sb.append(out);
        } else {
            sb.append(out.substring(0, precision - 1)).append('*');
        }

        // apply width and justification
        int len = sb.length();
        if (len < width)
            for (int i = 0; i < width - len; i++)
                if ((f & LEFT_JUSTIFY) == LEFT_JUSTIFY)
                    sb.append(' ');
                else
                    sb.insert(0, ' ');

        fmt.format(sb.toString());
    }

    public String toString() {
        return String.format("%s - %s", symbol, companyName);
    }
}
Run Code Online (Sandbox Code Playgroud)

运行

System.out.printf("%s", new StockName("HUGE", "Huge Fruit, Inc.", "Fruit Titanesque, Inc."));
Run Code Online (Sandbox Code Playgroud)

Huge Fruit, Inc.按预期打印.

但是,以下不起作用:

System.out.printf("%s", new StockName("PERC", "%Company, Inc.", "Fruit Titanesque, Inc."));
Run Code Online (Sandbox Code Playgroud)

它抛出一个java.util.MissingFormatArgumentException:

Exception in thread "main" java.util.MissingFormatArgumentException: Format specifier '%C'
        at java.util.Formatter.format(Formatter.java:2519)
        at java.util.Formatter.format(Formatter.java:2455)
        at StockName.formatTo(FormattableTest.java:44)
        at java.util.Formatter$FormatSpecifier.printString(Formatter.java:2879)
        at java.util.Formatter$FormatSpecifier.print(Formatter.java:2763)
        at java.util.Formatter.format(Formatter.java:2520)
        at java.io.PrintStream.format(PrintStream.java:970)
        at java.io.PrintStream.printf(PrintStream.java:871)
        at FormattableTest.main(FormattableTest.java:55)
Run Code Online (Sandbox Code Playgroud)

该示例用于Formatter.format添加文本,而format应该格式化格式字符串.当应该附加的文本包含百分比时,这会导致事情中断.

我应该怎么处理这个formatTo 我应该手动写入格式化程序的Appendable(formatter.out().append(text)可能会以IOException某种方式抛出)吗?我应该尝试逃避格式字符串(类似的东西formatter.format(text.replace("%","%%")),虽然这可能还不够)?我应该将简单的格式字符串传递给格式化程序(formatter.format("%s", text)但这似乎是多余的)?所有这些都应该有效,但语义上的正确方法是什么?

为了澄清,在这种假设情况下,给出的参数StockName是用户控制的并且可以是任意的; 我没有对它们的精确控制(我不能禁止输入%).但是,我可以编辑StockName.formatTo.

kri*_*aex 5

实际上很简单。您只能在格式化期间转义百分比字符,而不能在原始属性中转义:

  // apply precision
  if (precision == -1 || out.length() < precision) {
    // write it all
    sb.append(out.replaceAll("%", "%%"));
  }
  else {
    sb.append(out.substring(0, precision - 1).replaceAll("%", "%%")).append('*');
  }
Run Code Online (Sandbox Code Playgroud)

然后,如果您这样做:

StockName stockName = new StockName("HUGE", "%Huge Fruit, Inc.", "Fruit Titanesque, Inc.");
System.out.printf("%s%n", stockName);
System.out.printf("%s%n", stockName.toString());
System.out.printf("%#s%n", stockName);
System.out.printf("%-10.8s%n", stockName);
System.out.printf("%.12s%n", stockName);
System.out.printf(Locale.FRANCE, "%25s%n", stockName);
Run Code Online (Sandbox Code Playgroud)

输出看起来像这样:

%Huge Fruit, Inc.
HUGE - %Huge Fruit, Inc.
HUGE
HUGE      
%Huge Fruit*
   Fruit Titanesque, Inc.
Run Code Online (Sandbox Code Playgroud)