原文:http://zetcode.com/lang/java/strings/

在 Java 教程的这一部分中,我们将更详细地处理字符串数据。

字符串是计算机语言中非常重要的数据类型。 这就是为什么我们将一整章专门讨论 Java 中的字符串的原因。

Java 字符串定义

在 Java 中,字符串是 Unicode 字符序列。 字符串是对象。 有两个用于处理字符串的基本类:

  • String
  • StringBuilder

String是不可变的字符序列。 StringBuilder是可变的字符序列。 (还有一个StringBuffer类,可以由多个线程使用。如果不处理线程,则可以使用StringBuilder。)

字符串字面值源代码中用双引号引起来的一系列字符。 例如,"Java"是字符串字面值。 每当 Java 编译器在代码中遇到字符串字面值时,它都会使用其值创建一个String对象。

  1. String lang = "Java"; // same as String lang = new String("Java");

字符串字面值被许多编程语言使用。 这是一个既定的约定,还可以节省键入内容。

Java 初始化字符串

有多种创建字符串的方法,它们都是不可变的和可变的。 我们将展示其中的一些。

com/zetcode/StringInit.java

  1. package com.zetcode;
  2. public class StringInit {
  3. public static void main(String[] args) {
  4. char[] cdb = {'M', 'y', 'S', 'Q', 'L'};
  5. String lang = "Java";
  6. String ide = new String("NetBeans");
  7. String db = new String(cdb);
  8. System.out.println(lang);
  9. System.out.println(ide);
  10. System.out.println(db);
  11. StringBuilder sb1 = new StringBuilder(lang);
  12. StringBuilder sb2 = new StringBuilder();
  13. sb2.append("Fields");
  14. sb2.append(" of ");
  15. sb2.append("glory");
  16. System.out.println(sb1);
  17. System.out.println(sb2);
  18. }
  19. }

该示例显示了创建StringStringBuilder对象的几种方法。

  1. String lang = "Java";

最常见的方法是根据字符串字面值创建字符串对象。

  1. String ide = new String("NetBeans");

在这一行中,我们使用构建对象的常用方法创建一个字符串,即使用new关键字。

  1. String db = new String(cdb);

在这里,我们从字符数组创建一个字符串对象。

  1. StringBuilder sb1 = new StringBuilder(lang);

String创建一个StringBuilder对象。

  1. StringBuilder sb2 = new StringBuilder();
  2. sb2.append("Fields");
  3. sb2.append(" of ");
  4. sb2.append("glory");

我们创建一个空的StringBuilder对象。 我们将三个字符串附加到对象中。

  1. $ java StringInit.java
  2. Java
  3. NetBeans
  4. MySQL
  5. Java
  6. Fields of glory

运行示例可得出此结果。

Java 字符串是对象

字符串是对象; 它们不是原始数据类型。 字符串是StringStringBuilder类的实例。 由于它们是对象,因此有多种方法可用于完成各种工作。

com/zetcode/StringObjects.java

  1. package com.zetcode;
  2. public class StringObjects {
  3. public static void main(String[] args) {
  4. String lang = "Java";
  5. String bclass = lang.getClass().toString();
  6. System.out.println(bclass);
  7. String sup = lang.getClass().getSuperclass().toString();
  8. System.out.println(sup);
  9. if (lang.isEmpty()) {
  10. System.out.println("The string is empty");
  11. } else {
  12. System.out.println("The string is not empty");
  13. }
  14. int l = lang.length();
  15. System.out.println("The string has " + l + " characters");
  16. }
  17. }

在此程序中,我们演示了字符串是对象。 对象必须具有一个类名,一个父类,并且还必须具有一些可以调用的方法。

  1. String lang = "Java";

创建String类型的对象。

  1. String bclass = lang.getClass().toString();

我们确定lang变量所引用的对象的类名称。

  1. String sup = lang.getClass().getSuperclass().toString();

接收到我们对象的父类。 所有对象都有至少一个父对象-Object

  1. if (lang.isEmpty()) {
  2. System.out.println("The string is empty");
  3. } else {
  4. System.out.println("The string is not empty");
  5. }

对象具有各种方法。 有用的字符串方法之一是isEmpty()方法,它确定字符串是否为空。

  1. int l = lang.length();

length()方法返回字符串的大小。

  1. $ java StringObjects.java
  2. class java.lang.String
  3. class java.lang.Object
  4. The string is not empty
  5. The string has 4 characters

我们的字符串对象是String类的实例。 它具有Object父类。 该对象不为空,并且包含四个字符。

Java 可变&不可变字符串

String是不可变字符序列,而StringBuilder是可变字符序列。 下一个示例将显示差异。

com/zetcode/MutableImmutable.java

  1. package com.zetcode;
  2. public class MutableImmutable {
  3. public static void main(String[] args) {
  4. String name = "Jane";
  5. String name2 = name.replace('J', 'K');
  6. String name3 = name2.replace('n', 't');
  7. System.out.println(name);
  8. System.out.println(name3);
  9. StringBuilder sb = new StringBuilder("Jane");
  10. System.out.println(sb);
  11. sb.setCharAt(0, 'K');
  12. sb.setCharAt(2, 't');
  13. System.out.println(sb);
  14. }
  15. }

这两个对象都有替换字符串中字符的方法。

  1. String name = "Jane";
  2. String name2 = name.replace('J', 'K');
  3. String name3 = name2.replace('n', 't');

String上调用replace()方法将导致返回新的修改后的字符串。 原始字符串不变。

  1. sb.setCharAt(0, 'K');
  2. sb.setCharAt(2, 't');

StringBuildersetCharAt()方法将用新字符替换给定索引处的字符。 原始字符串被修改。

  1. $ java MutableImmutable.java
  2. Jane
  3. Kate
  4. Jane
  5. Kate

这是com.zetcode.MutableImmutable示例的输出。

Java 连接字符串

可以使用+运算符或concat()方法添加不可变的字符串。 它们将形成一个新字符串,该字符串是所有连接字符串的链。 可变字符串具有append()方法,该方法可以从任意数量的其他字符串中构建一个字符串。

com/zetcode/ConcatenateStrings.java

  1. package com.zetcode;
  2. public class ConcatenateStrings {
  3. public static void main(String[] args) {
  4. System.out.println("Return" + " of " + "the king.");
  5. System.out.println("Return".concat(" of ").concat("the king."));
  6. StringBuilder sb = new StringBuilder();
  7. sb.append("Return");
  8. sb.append(" of ");
  9. sb.append("the king.");
  10. System.out.println(sb);
  11. }
  12. }

该示例通过添加字符串来创建三个句子。

  1. System.out.println("Return" + " of " + "the king.");

通过使用+运算符形成一个新的字符串。

  1. System.out.println("Return".concat(" of ").concat("the king."));

concat()方法返回一个字符串,该字符串表示此对象的字符及其后的字符串参数的字符的连接。

  1. StringBuilder sb = new StringBuilder();
  2. sb.append("Return");
  3. sb.append(" of ");
  4. sb.append("the king.");

通过三次调用append()方法来创建StringBuilder类型的可变对象。

  1. $ java ConcatenateStrings.java
  2. Return of the king.
  3. Return of the king.
  4. Return of the king.

这是示例输出。

Java 字符串的引号

在某些情况下,例如使用直接语音,必须对内引号进行转义。

com/zetcode/Quotes.java

  1. package com.zetcode;
  2. public class Quotes {
  3. public static void main(String[] args) {
  4. System.out.println("There are may stars");
  5. System.out.println("He said: \"Which one are you looking at?\"");
  6. }
  7. }

我们使用\字符转义其他引号。

  1. $ java Quotes.java
  2. There are may stars
  3. He said: "Which one are you looking at?"

在这里,我们看到com.zetcode.Quotes程序的输出。

Java 多行字符串

Java 13 引入了文本块,该文本块允许定义多行字符串。 要创建多行字符串,我们使用三引号。

com/zetcode/MultilineStrings.java

  1. package com.zetcode;
  2. public class MultilineString {
  3. static String lyrics = """
  4. I cheated myself
  5. like I knew I would
  6. I told ya, I was trouble
  7. you know that I'm no good""";
  8. public static void main(String[] args) {
  9. System.out.println(lyrics);
  10. }
  11. }

我们有一条跨越四行的空想。

  1. $ java MultilineString.java
  2. I cheated myself
  3. like I knew I would
  4. I told ya, I was trouble
  5. you know that I'm no good

我们看到示例的输出。

在 Java 的早期版本中,我们需要执行连接操作并使用\n字符。

com/zetcode/MultilineString2.java

  1. package com.zetcode;
  2. public class MultilineString2 {
  3. static String lyrics = "I cheated myself\n" +
  4. "like I knew I would\n" +
  5. "I told ya, I was trouble\n" +
  6. "you know that I'm no good";
  7. public static void main(String[] args) {
  8. System.out.println(lyrics);
  9. }
  10. }

这四个字符串用+运算符连接。

Java 字符串元素

字符串是字符序列。 字符是字符串的基本元素。 以下两个示例显示了一些处理字符串字符的方法。

com/zetcode/StringElements.java

  1. package com.zetcode;
  2. public class StringElements {
  3. public static void main(String[] args) {
  4. char[] crs = {'Z', 'e', 't', 'C', 'o', 'd', 'e' };
  5. String s = new String(crs);
  6. char c1 = s.charAt(0);
  7. char c2 = s.charAt(s.length()-1);
  8. System.out.println(c1);
  9. System.out.println(c2);
  10. int i1 = s.indexOf('e');
  11. int i2 = s.lastIndexOf('e');
  12. System.out.println("The first index of character e is " + i1);
  13. System.out.println("The last index of character e is " + i2);
  14. System.out.println(s.contains("t"));
  15. System.out.println(s.contains("f"));
  16. char[] elements = s.toCharArray();
  17. for (char el : elements) {
  18. System.out.println(el);
  19. }
  20. }
  21. }

在第一个示例中,我们将使用不可变的字符串。

  1. char[] crs = {'Z', 'e', 't', 'C', 'o', 'd', 'e' };
  2. String s = new String(crs);

由字符数组构成一个新的不可变字符串。

  1. char c1 = s.charAt(0);
  2. char c2 = s.charAt(s.length()-1);

使用charAt()方法,我们获得字符串的第一个和最后一个char值。

  1. int i1 = s.indexOf('e');
  2. int i2 = s.lastIndexOf('e');

使用indexOf()lastIndexOf()方法,我们得到字符’e’的第一个和最后一个出现。

  1. System.out.println(s.contains("t"));

使用contains()方法,我们检查字符串是否包含t字符。 该方法返回一个布尔值。

  1. char[] elements = s.toCharArray();
  2. for (char el : elements) {
  3. System.out.println(el);
  4. }

toCharArray()方法从字符串创建一个字符数组。 我们遍历数组并打印每个字符。

  1. $ java StringElements.java
  2. Z
  3. e
  4. The first index of character e is 1
  5. The last index of character e is 6
  6. true
  7. false
  8. Z
  9. e
  10. t
  11. C
  12. o
  13. d
  14. e

This is the example output.

在第二个示例中,我们将处理StringBuilder类的元素。

com/zetcode/StringBuilderElements.java

  1. package com.zetcode;
  2. public class StringBuilderElements {
  3. public static void main(String[] args) {
  4. StringBuilder sb = new StringBuilder("Misty mountains");
  5. System.out.println(sb);
  6. sb.deleteCharAt(sb.length()-1);
  7. System.out.println(sb);
  8. sb.append('s');
  9. System.out.println(sb);
  10. sb.insert(0, 'T');
  11. sb.insert(1, 'h');
  12. sb.insert(2, 'e');
  13. sb.insert(3, ' ');
  14. System.out.println(sb);
  15. sb.setCharAt(4, 'm');
  16. System.out.println(sb);
  17. }
  18. }

形成可变的字符串。 我们通过删除,附加,插入和替换字符来修改字符串的内容。

  1. sb.deleteCharAt(sb.length()-1);

这行删除最后一个字符。

  1. sb.append('s');

删除的字符将附加回字符串。

  1. sb.insert(0, 'T');
  2. sb.insert(1, 'h');
  3. sb.insert(2, 'e');
  4. sb.insert(3, ' ');

我们在字符串的开头插入四个字符。

  1. sb.setCharAt(4, 'm');

最后,我们在索引 4 处替换一个字符。

  1. $ java StringBuilderElements.java
  2. Misty mountains
  3. Misty mountain
  4. Misty mountains
  5. The Misty mountains
  6. The misty mountains

从输出中,我们可以看到可变字符串是如何变化的。

回文示例

回文是一个单词,数字,短语或其他字符序列,它们的向前和向后读取相同,例如女士或赛车。 有很多方法可以检查字符串是否是回文。 以下示例是可能的解决方案之一。

com/zetcode/Palindrome.java

  1. package com.zetcode;
  2. // A palindrome is a word, number, phrase, or other sequence of characters
  3. // which reads the same backward as forward, such as madam or racecar
  4. public class Palindrome3 {
  5. public static void main(String[] args) {
  6. System.out.println(isPalindrome("radar"));
  7. System.out.println(isPalindrome("kayak"));
  8. System.out.println(isPalindrome("forest"));
  9. }
  10. private static boolean isPalindrome(String original) {
  11. char[] data = original.toCharArray();
  12. int i = 0;
  13. int j = data.length - 1;
  14. while (j > i) {
  15. if (data[i] != data[j]) {
  16. return false;
  17. }
  18. ++i;
  19. --j;
  20. }
  21. return true;
  22. }
  23. }

我们已经实现了isPalindrome()方法。

  1. private static boolean isPalindrome(String original) {
  2. char[] data = original.toCharArray();
  3. ...

我们将字符串转换为字符数组。

  1. int i = 0;
  2. int j = data.length - 1;
  3. while (j > i) {
  4. if (data[i] != data[j]) {
  5. return false;
  6. }
  7. ++i;
  8. --j;
  9. }
  10. return true;

我们遍历数组,并将左侧字符与右侧对应字符进行比较。 如果全部匹配,则返回true,否则返回false

  1. $ java Palindrome.java
  2. true
  3. true
  4. false

这是输出。

Java 子字符串

substring()方法返回字符串的一部分。 起始索引是包含的,结束索引是排斥的。 起始索引从零开始。

com/zetcode/Substrings.java

  1. package com.zetcode;
  2. public class Substrings {
  3. public static void main(String[] args) {
  4. String str = "bookcase";
  5. System.out.println(str.substring(0, 4));
  6. System.out.println(str.substring(4, str.length()));
  7. }
  8. }

该示例使用substring()方法创建两个子字符串。

  1. System.out.println(str.substring(0, 4));

在这里,我们得到"book"子字符串。 零是指字符串的第一个字符。

  1. System.out.println(str.substring(4, str.length()));

在此打印"case"子字符串。

  1. $ java Substrings.java
  2. book
  3. case

这是输出。

Java 分割字符串

split()方法将字符串分割成几部分; 它使用定界正则表达式作为参数。

com/zetcode/Splitting.java

  1. package com.zetcode;
  2. public class Splitting {
  3. public static void main(String[] args) {
  4. String s = "Today is a beautiful day.";
  5. String[] words = s.split(" ");
  6. for (String word : words) {
  7. System.out.println(word);
  8. }
  9. }
  10. }

该示例将句子拆分为单词。

  1. String s = "Today is a beautiful day.";

这是一个句子。 单词之间用空格隔开。

  1. String[] words = s.split(" ");

使用split()方法,我们将句子切成单词。 空格字符用作分隔符。 该方法返回一个字符串数组。

  1. for (String word : words) {
  2. System.out.println(word);
  3. }

我们遍历数组并打印其内容。

  1. $ java Splitting.java
  2. Today
  3. is
  4. a
  5. beautiful
  6. day.

这是输出。

Java 删除字符串字符

当我们将字符串拆分为单词时,某些单词的开头或结尾字符为逗号或点号。 在下一个示例中,我们显示如何删除此类字符。

com/zetcode/RemovingChars.java

  1. package com.zetcode;
  2. public class RemovingChars {
  3. public static void main(String[] args) {
  4. String str = "Did you go there? We did, but we had a \"great\" service there.";
  5. String[] parts = str.split(" ");
  6. for (String part: parts) {
  7. String word = removeChars(part);
  8. System.out.println(word);
  9. }
  10. }
  11. private static String removeChars(String part) {
  12. String word = part;
  13. if (part.endsWith(".") || part.endsWith("?") || part.endsWith(",")) {
  14. word = part.substring(0, part.length()-1);
  15. }
  16. if (part.startsWith("\"") && part.endsWith("\"")) {
  17. word = part.substring(1, part.length()-1);
  18. }
  19. return word;
  20. }
  21. }

该示例将字符串拆分为单词,并删除潜在的逗号,点,问号或双引号。

  1. String str = "Did you go there? We did, but we had a \"great\" service there.";

在此字符串中,我们有一个问号,逗号,引号和附加在单词上的圆点。

  1. private static String removeChars(String part) {

在此自定义方法内,我们从单词中删除了这些字符。

  1. if (part.endsWith(".") || part.endsWith("?") || part.endsWith(",")) {
  2. word = part.substring(0, part.length()-1);
  3. }

在此if语句中,我们删除结尾字符。 我们使用endsWith()方法来标识要删除的字符。 substring()方法返回不包含字符的字符串的一部分。

  1. if (part.startsWith("\"") && part.endsWith("\"")) {
  2. word = part.substring(1, part.length()-1);
  3. }

同样,我们删除起始字符。 起始字符通过startsWith()方法检查。

  1. $ java RemovingChars.java
  2. Did
  3. you
  4. go
  5. there
  6. We
  7. did
  8. but
  9. we
  10. had
  11. a
  12. great
  13. service
  14. there

这是示例的输出。 字符已成功删除。

Java 连接字符串

从 Java8 开始,我们有了join()方法来连接字符串。 请参阅 StringJoiner 教程,以了解有关在 Java 中连接字符串的更多信息。

com/zetcode/Joining.java

  1. package com.zetcode;
  2. public class Joining {
  3. public static void main(String[] args) {
  4. String joined = String.join(" ", "Today", "is", "Sunday");
  5. System.out.println(joined);
  6. }
  7. }

在示例中,我们将三个字符串合并为一个最终字符串。

  1. String joined = String.join(" ", "Today", "is", "Sunday");

join()方法的第一个参数是定界符,它将分隔最终字符串中的每个字符串。 其余参数是要连接的字符串。

Java 比较字符串

比较字符串有两种基本方法。 equals()方法比较两个字符串的内容,并返回一个布尔值,指示字符串是否相等。 equalsIgnoreCase()做相同的事情,除了它忽略大小写。

com/zetcode/ComparingStrings.java

  1. package com.zetcode;
  2. public class ComparingStrings {
  3. public static void main(String[] args) {
  4. String a = "book";
  5. String b = "Book";
  6. System.out.println(a.equals(b));
  7. System.out.println(a.equalsIgnoreCase(b));
  8. }
  9. }

我们使用上述方法比较两个字符串。

  1. String a = "book";
  2. String b = "Book";

我们定义了两个要比较的字符串。

  1. System.out.println(a.equals(b));

equals()方法返回false。 两个字符串的第一个字符不同。

  1. System.out.println(a.equalsIgnoreCase(b));

当我们忽略大小写时,字符串是相等的:equalsIgnoreCase()方法返回true

  1. $ java ComparingStrings.java
  2. false
  3. true

这是程序的输出。

如果我们正在将变量与字符串进行比较,则切记字符串应位于比较方法的左侧,这一点很重要。 否则我们可能会得到NullPointerException

com/zetcode/ComparingStrings2.java

  1. import java.util.Random;
  2. public class ComparingStrings2 {
  3. public static String readString() {
  4. Random r = new Random();
  5. boolean b = r.nextBoolean();
  6. if (b == true) {
  7. return "ZetCode";
  8. } else {
  9. return null;
  10. }
  11. }
  12. public static void main(String[] args) {
  13. String d = readString();
  14. if ("ZetCode".equals(d)) {
  15. System.out.println("Strings are equal");
  16. } else {
  17. System.out.println("Strings are not equal");
  18. }
  19. }
  20. }

在代码示例中,我们正确地比较了字符串,避免了可能的NullPointerException

  1. public static String readString() {
  2. Random r = new Random();
  3. boolean b = r.nextBoolean();
  4. if (b == true) {
  5. return "ZetCode";
  6. } else {
  7. return null;
  8. }
  9. }

readString()方法模拟方法调用可能导致空值的情况。 例如,如果我们尝试从数据库中读取值,则可能会发生这种情况。

  1. String d = readString();

d变量可以包含空值。

  1. if ("ZetCode".equals(d)) {

上一行是比较两个字符串的正确方法,其中一个字符串是已知字面值。 如果将d变量放在左侧,则如果d变量包含空值,则将导致NullPointerException

equals()方法比较两个字符串的字符。 ==运算符测试引用是否相等。 所有字符串字面值都将使用 Java 自动插入。 它们被放置在字符串池中。 这是在编译时发生的。 如果两个变量包含两个相等的字符串字面值,则它们实际上引用字符串池中的同一字符串对象。

com/zetcode/ComparingStrings3.java

  1. package com.zetcode;
  2. public class ComparingStrings3 {
  3. public static void main(String[] args) {
  4. boolean a = "ZetCode" == "ZetCode";
  5. boolean b = "ZetCode" == new String("ZetCode");
  6. boolean c = "ZetCode" == "Zet" + "Code";
  7. boolean d = "ZetCode" == new String("ZetCode").intern();
  8. boolean e = "ZetCode" == " ZetCode ".trim();
  9. System.out.println(a);
  10. System.out.println(b);
  11. System.out.println(c);
  12. System.out.println(d);
  13. System.out.println(e);
  14. }
  15. }

在此代码示例中,我们将字符串对象与==运算符进行比较。

  1. boolean a = "ZetCode" == "ZetCode";

这些字符串字面值是内部的。 因此,身份比较运算符返回true

  1. boolean b = "ZetCode" == new String("ZetCode");

使用new运算符创建的字符串不会被保留。 比较运算符导致错误的值。

  1. boolean c = "ZetCode" == "Zet" + "Code";

字符串在编译时连接在一起。 字符串字面值导致相同的对象。 结果是正确的。

  1. boolean d = "ZetCode" == new String("ZetCode").intern();

intern()对象将右侧的字符串对象放入池中。 因此,d 变量为布尔值 true。

  1. boolean e = "ZetCode" == " ZetCode ".trim();

在运行时调用trim()方法,生成一个不同的对象。 e变量具有布尔值false

  1. $ java ComparingStrings3.java
  2. true
  3. false
  4. true
  5. true
  6. false

这是示例的输出。

Java 格式化字符串

我们可以同时使用System.out.printf()System.out.format()方法来格式化 Java 中的字符串。 它们的工作原理相同。 这两种方法使用指定的格式字符串和参数将格式字符串写入输出流。 如果参数多于格式说明符,则多余的参数将被忽略。

  1. %[argument_index$][flags][width][.precision]conversion

常规,字符和数字类型的格式说明符具有此语法。

  1. %[argument_index$][flags][width]conversion

这是用于表示日期和时间的类型的语法。

格式说明符以%字符开头,以 1 或 2 个字符转换结尾,该转换符指定要生成的格式化输出的种类。 可选项目放在方括号之间。

arguments_index是一个十进制整数,指示参数在参数列表中的位置。 标志是一组修改输出格式的字符。 有效标志集取决于转换。 宽度是一个非负的十进制整数,指示要写入输出的最小字符数。 精度是通常用于限制字符数的非负十进制整数。 具体行为取决于转换。 所需的转换是一个字符,指示应如何格式化参数。

com/zetcode/Conversions.java

  1. package com.zetcode;
  2. public class Conversions {
  3. public static void main(String[] args) {
  4. System.out.format("There are %d %s.%n", 5, "pencils");
  5. System.out.printf("The rock weighs %f kilograms.%n", 5.345);
  6. }
  7. }

在此程序中,我们格式化两个简单的句子。

  1. System.out.format("There are %d %s.%n", 5, "pencils");

在此代码行中,我们有三个格式说明符。 每个说明符均以%字符开头。 d说明符格式化整数值。 s说明符需要字符串值。 %n输出平台特定的线路终端器; 它不需要参数。

  1. System.out.printf("The rock weighs %f kilograms.%n", 5.345);

f将浮点值格式化为十进制值。 System.out.printf()System.out.format()的工作相同。

  1. $ java Conversions.java
  2. There are 5 pencils.
  3. The rock weighs 5.345000 kilograms.

This is the output of the program.

com/zetcode/IndexPosition.java

  1. package com.zetcode;
  2. import java.util.Calendar;
  3. public class IndexPosition {
  4. public static void main(String[] args) {
  5. int x = 12;
  6. int y = 32;
  7. int z = 43;
  8. Calendar c = Calendar.getInstance();
  9. System.out.format("There are %d apples, %d oranges and "
  10. + "%d pears%n", x, y, z);
  11. System.out.format("There are %2$d apples, %3$d oranges and "
  12. + "%1$d pears%n", x, y, z);
  13. System.out.format("Year: %tY, Month: %<tm, Day: %<td%n", c);
  14. }
  15. }

该示例使用参数索引来引用包含在参数列表中的变量。

  1. System.out.format("There are %d apples, %d oranges and "
  2. + "%d pears%n", x, y, z);

如果不指定索引,则变量将自动与指定符匹配。 d说明符将整数值格式化为十进制值。

  1. System.out.format("There are %2$d apples, %3$d oranges and "
  2. + "%1$d pears%n", x, y, z);

1$引用x变量,2$引用y变量,3$引用z变量。

  1. System.out.format("Year: %tY, Month: %<tm, Day: %<td%n", c);

<标志使先前格式说明符的参数被重用。 这三个说明符均引用c变量。 tY转换字符给出的年份格式至少为四位数字,必要时带有前导零。tm给出一个月份,格式为两位数字,必要时以前导零,而td给出月份中的某一天,格式为两位数字,必要时带有前导零。

  1. $ java IndexPosition.java
  2. There are 12 apples, 32 oranges and 43 pears
  3. There are 32 apples, 43 oranges and 12 pears
  4. Year: 2016, Month: 09, Day: 07

This is the output of the program.

标志以特定方式修改格式。 有几个标志。 例如,+标志要求输出为所有正数包括一个正号。

com/zetcode/Flags.java

  1. package com.zetcode;
  2. public class Flags {
  3. public static void main(String[] args) {
  4. System.out.format("%+d%n", 553);
  5. System.out.format("%010d%n", 553);
  6. System.out.format("%10d%n", 553);
  7. System.out.format("%-10d%n", 553);
  8. System.out.format("%d%n", -553);
  9. System.out.format("%(d%n", -553);
  10. }
  11. }

该示例显示了一些字符串格式说明符的标志。

  1. System.out.format("%010d%n", 553);

0标志将使输出填充前导零到最小字段宽度。 我们的电话号码有三位数。 最小宽度为 10。因此,输出中有 7 个前导零。

  1. System.out.format("%10d%n", 553);

如果没有0标志,该数字将右对齐。

  1. System.out.format("%-10d%n", 553);

-标志将使数字左对齐。

  1. System.out.format("%d%n", -553);
  2. System.out.format("%(d%n", -553);

默认情况下,负数带有减号。 如果我们使用(标志,则负值将放在圆括号内。 (这用于会计。)

  1. $ java Flags.java
  2. +553
  3. 0000000553
  4. 553
  5. 553
  6. -553
  7. (553)

在这里,我们看到com.zetcode.Flags示例的输出。

width字段是要写入输出的最小字符数。 不能与行分隔符一起使用。

com/zetcode/WidthSpecifier.java

  1. package com.zetcode;
  2. public class WidthSpecifier {
  3. public static void main(String[] args) {
  4. System.out.println(1);
  5. System.out.println(16);
  6. System.out.println(1655);
  7. System.out.println(16567);
  8. System.out.println(166701);
  9. System.out.format("%10d%n", 1);
  10. System.out.format("%10d%n", 16);
  11. System.out.format("%10d%n", 1655);
  12. System.out.format("%10d%n", 16567);
  13. System.out.format("%10d%n", 166701);
  14. }
  15. }

首先,我们打印五个数字而不指定字段宽度。 输出的宽度等于要显示的字符数。 在第二种情况下,字段宽度为 10。5 个输出中的每个输出的最小长度为 10 个字符。 数字右对齐。

  1. System.out.format("%10d%n", 1);

数字 10 指出字符串输出必须至少包含十个字符。

  1. $ java WidthSpecifier.java
  2. 1
  3. 16
  4. 1655
  5. 16567
  6. 166701
  7. 1
  8. 16
  9. 1655
  10. 16567
  11. 166701

我们可以看到,在第二种情况下,数字是右对齐的。

对于不同的转换,精度字段具有不同的含义。 对于常规参数类型,精度是要写入输出的最大字符数。

com/zetcode/PrecisionSpecifier.java

  1. package com.zetcode;
  2. public class PrecisionSpecifier {
  3. public static void main(String[] args) {
  4. System.out.format("%.3g%n", 0.0000006);
  5. System.out.format("%.3f%n", 54.34263);
  6. System.out.format("%.3s%n", "ZetCode");
  7. }
  8. }

在三个不同的输出上演示了精度说明符。

  1. System.out.format("%.3g%n", 0.0000006);

如果使用g转换,则精度是四舍五入后得到的幅度中的位数总数。

  1. System.out.format("%.3f%n", 54.34263);

对于浮点值,精度是小数点分隔符后的位数。

  1. System.out.format("%.3s%n", "ZetCode");

对于字符串,它是最大可打印字符数。 七个字符中只有三个字符被打印到控制台。

  1. $ java PrecisionSpecifier.java
  2. 6.00e-07
  3. 54.343
  4. Zet

This is the example output.

下一个示例将格式化数字数据。

com/zetcode/FormatNumbers.java

  1. package com.zetcode;
  2. public class FormatNumbers {
  3. public static void main(String[] args) {
  4. System.out.format("%d%n", 12263);
  5. System.out.format("%o%n", 12263);
  6. System.out.format("%x%n", 12263);
  7. System.out.format("%e%n", 0.03452342263);
  8. System.out.format("%d%%%n", 45);
  9. }
  10. }

该示例演示了数字的标准格式说明符。

  1. System.out.format("%d%n", 12263);

d转换说明符会将整数值转换为十进制值。

  1. System.out.format("%o%n", 12263);

o转换说明符会将数字格式化为八进制。

  1. System.out.format("%x%n", 12263);

使用x说明符,结果被格式化为十六进制整数。

  1. System.out.format("%e%n", 0.03452342263);

使用e指示符,以科学计数法打印数字。

  1. System.out.format("%d%%%n", 45);

%%字符用于打印百分号。

  1. $ java FormatNumbers.java
  2. 12263
  3. 27747
  4. 2fe7
  5. 3.452342e-02
  6. 45%

该程序以不同的格式打印数字。

最后,我们将格式化日期和时间数据。

com/zetcode/FormatDateTime.java

  1. package com.zetcode;
  2. import java.time.LocalDateTime;
  3. public class FormatDateTime {
  4. public static void main(String[] args) {
  5. LocalDateTime ldt = LocalDateTime.now();
  6. System.out.format("%tF%n", ldt);
  7. System.out.format("%tD%n", ldt);
  8. System.out.format("%tT%n", ldt);
  9. System.out.format("%1$tA, %1$tb %1$tY%n", ldt);
  10. System.out.format("%1$td.%1$tm.%1$tY%n", ldt);
  11. }
  12. }

该示例演示了日期的标准格式说明符。 日期和时间格式字符串的转换部分以t字符开头。

  1. System.out.format("%tF%n", c);

作为tF转换的结果,此行以完整的 ISO 8601 格式打印日期。

  1. System.out.format("%1$td.%1$tm.%1$tY%n", c);

使用这些格式说明符,我们以斯洛伐克使用的格式打印日期。 这些部分由点字符分隔,并且日期在月份之前,月份在年份之前。 所有这三种格式说明符均引用c变量。

  1. $ java FormatDateTime.java
  2. 2019-10-02
  3. 10/02/19
  4. 15:45:02
  5. Wednesday, Oct 2019
  6. 02.10.2019

This is the output of the program.

Java 教程的这一部分更详细地介绍了字符串。