import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.function.Function;
import java.util.stream.Collectors;
class CommonCharacters {
@SuppressWarnings("boxing")
// Requirement
//
// Always return lowercase versions of common characters. e.g.:
//
// OK: (a, a) -> a
// OK: (a, A) -> a
// OK: (A, A) -> a
// No: (A, A) -> A
// No: (aA, aA) -> aA
//
// Requirement
//
// Return common characters joined in a String, preserving the order in
// which they appeared in the longest argument, or in the first argument if
// the arguments are of the same length.
//
// Requirement
//
// Handle "characters" (i.e. code points) outside the Basic Multilingual
// Plane (BMP), including characters from Supplementary Planes.
// There should be no `char' or `Character' based "false positives". i.e.:
//
// String string1 = "\uD835\uDC00", string2 = "\uD835\uDC01";
//
// string1 and string2 share no characters in the intended acceptation of
// "character".
String shortestArgument, longestArgument
; if (string1.length() < string2.length()) {
shortestArgument = string1;
longestArgument = string2;
} else {
shortestArgument = string2;
longestArgument = string1;
}
// @formatter:off
Set<Integer> codePointsSeen =
shortestArgument.codePoints()
.boxed()
.collect(Collectors.toSet());
List<Integer> codePointsInCommon = new ArrayList<>();
for (Iterator<Integer> iterator = longestArgument.codePoints()
.distinct()
.iterator();
iterator.hasNext() &&
codePointsInCommon.size() < codePointsSeen.size();) {
// @formatter:on
Integer codePoint
= iterator.
next(); int lowerCaseCodePoint
= Character.
toLowerCase(codePoint
); if (codePointsSeen.contains(lowerCaseCodePoint)) {
codePointsInCommon.add(lowerCaseCodePoint);
}
}
StringBuilder stringBuilder = new StringBuilder();
for (Integer codePoint
: codePointsInCommon
) { stringBuilder.appendCodePoint(codePoint);
}
return stringBuilder.toString();
}
@SuppressWarnings("boxing")
public static void main
(String[] args
) { // @formatter:off
String[][] testArgumentsAndExpectedValues
= { { "" , "" , "" },
{ "a" , "" , "" },
{ "" , "a" , "" },
{ "aa" , "" , "" },
{ "" , "aa" , "" },
{ "a" , "a" , "a" },
{ "aa" , "b" , "" },
{ "b" , "aa" , "" },
{ "ab" , "ba" , "ab" },
{ "aba" , "ab" , "ab" },
{ "aba" , "ba" , "ab" },
{ "aba" , "aab" , "ab" },
{ "a" , "A" , "a" },
{ "A" , "a" , "a" },
{ "A" , "A" , "a" },
{ "ab" , "AB" , "ab" },
{ "AB" , "ab" , "ab" },
{ "aB" , "Ab" , "ab" },
{ "aB" , "Ba" , "ab" },
{ "aB" , "Ba" , "ab" },
{ "a" , "\uD835\uDC1A" , "" },
{ "\uD835\uDC1A" , "\uD835\uDC1A" , "\uD835\uDC1A" },
{ "\uD835\uDC00" , "\uD835\uDC00" , "\uD835\uDC00" },
{ "\uD835\uDC1A" , "\uD835\uDC00" , "" },
{ "\uD835\uDC00" , "\uD835\uDC01" , "" },
{ "\uD801\uDC2B" , "\uD801\uDC2B" , "\uD801\uDC2B" },
{ "\uD801\uDC03" , "\uD801\uDC03" , "\uD801\uDC2B" },
{ "\uD801\uDC2B" , "\uD801\uDC03" , "\uD801\uDC2B" },
{ "\uD83D\uDE80" , "\uD83D\uDE80" , "\uD83D\uDE80" },
{ "a" , "aaaaaaaaaaaaaaaaa" , "a" },
// The last test should still work, and work fast, with a second
// argument string starting with "a" and ending _many_ characters later
// The last test values doe not test it, but illustrate the scenario
};
int maximumTestArgumentLength =
Arrays.
stream(testArgumentsAndExpectedValues
) .
flatMap(testValues
-> Arrays.
stream(testValues
) .limit(2))
.max()
.getAsInt();
// @formatter:on
int maximumQuotedTestArgumentLength = maximumTestArgumentLength + 2;
for (int i = 0; i < testArgumentsAndExpectedValues.length; i++) {
String[] currentTestValues
= testArgumentsAndExpectedValues
[i
]; String string1
= currentTestValues
[0]; String string2
= currentTestValues
[1]; String expectedResult
= currentTestValues
[2]; String actualResult
= commonCharactersOf
(string1, string2
); boolean testSuccessful = expectedResult.equals(actualResult);
if (testSuccessful) {
// continue; // TODO: uncomment to filter out successful tests
}
Function
<String, String
> quoteString
= s
-> '"' + s
+ '"'; // @formatter:off
+ "%5s: "
+ "(%-" + maximumQuotedTestArgumentLength + "s"
+ " , "
+ "%-" + maximumQuotedTestArgumentLength + "s)"
+ " -> "
+ "%s "
+ "(%s)"
+ "%n";
System.
out.
printf(outputFormat,
i,
testSuccessful ? "Success" : "Failure",
quoteString.apply(string1),
quoteString.apply(string2),
quoteString.apply(actualResult),
quoteString.apply(expectedResult));
// @formatter:on
}
}
}
    import java.util.ArrayList;
    import java.util.Arrays;
    import java.util.Iterator;
    import java.util.List;
    import java.util.Set;
    import java.util.function.Function;
    import java.util.stream.Collectors;
    class CommonCharacters {
      @SuppressWarnings("boxing")
      private static String commonCharactersOf(String string1, String string2) {
        // Requirement
        //
        // Always return lowercase versions of common characters. e.g.:
        //
        // OK: (a, a) -> a
        // OK: (a, A) -> a
        // OK: (A, A) -> a
        // No: (A, A) -> A
        // No: (aA, aA) -> aA
        //
        // Requirement
        //
        // Return common characters joined in a String, preserving the order in
        // which they appeared in the longest argument, or in the first argument if
        // the arguments are of the same length.
        //
        // Requirement
        //
        // Handle "characters" (i.e. code points) outside the Basic Multilingual
        // Plane (BMP), including characters from Supplementary Planes.
        // There should be no `char' or `Character' based "false positives". i.e.:
        //
        // String string1 = "\uD835\uDC00", string2 = "\uD835\uDC01";
        //
        // string1 and string2 share no characters in the intended acceptation of
        // "character".
        String shortestArgument, longestArgument;
        if (string1.length() < string2.length()) {
          shortestArgument = string1;
          longestArgument = string2;
        } else {
          shortestArgument = string2;
          longestArgument = string1;
        }
        // @formatter:off
        Set<Integer> codePointsSeen =
            shortestArgument.codePoints()
            .boxed()
            .map(Character::toLowerCase)
            .collect(Collectors.toSet());
        List<Integer> codePointsInCommon = new ArrayList<>();
        for (Iterator<Integer> iterator = longestArgument.codePoints()
                                                         .distinct()
                                                         .iterator();
             iterator.hasNext() &&
             codePointsInCommon.size() < codePointsSeen.size();) {
        // @formatter:on
          Integer codePoint = iterator.next();
          int lowerCaseCodePoint = Character.toLowerCase(codePoint);
          if (codePointsSeen.contains(lowerCaseCodePoint)) {
            codePointsInCommon.add(lowerCaseCodePoint);
          }
        }
        StringBuilder stringBuilder = new StringBuilder();
        for (Integer codePoint : codePointsInCommon) {
          stringBuilder.appendCodePoint(codePoint);
        }
        return stringBuilder.toString();
      }
      @SuppressWarnings("boxing")
      public static void main(String[] args) {
        // @formatter:off
        String[][] testArgumentsAndExpectedValues = {
            { ""                     , ""                  , ""             },
            { "a"                    , ""                  , ""             },
            { ""                     , "a"                 , ""             },
            { "aa"                   , ""                  , ""             },
            { ""                     , "aa"                , ""             },
            { "a"                    , "a"                 , "a"            },
            { "aa"                   , "b"                 , ""             },
            { "b"                    , "aa"                , ""             },
            { "ab"                   , "ba"                , "ab"           },
            { "aba"                  , "ab"                , "ab"           },
            { "aba"                  , "ba"                , "ab"           },
            { "aba"                  , "aab"               , "ab"           },
            { "a"                    , "A"                 , "a"            },
            { "A"                    , "a"                 , "a"            },
            { "A"                    , "A"                 , "a"            },
            { "ab"                   , "AB"                , "ab"           },
            { "AB"                   , "ab"                , "ab"           },
            { "aB"                   , "Ab"                , "ab"           },
            { "aB"                   , "Ba"                , "ab"           },
            { "aB"                   , "Ba"                , "ab"           },
            { "a"                    , "\uD835\uDC1A"      , ""             },
            { "\uD835\uDC1A"         , "\uD835\uDC1A"      , "\uD835\uDC1A" },
            { "\uD835\uDC00"         , "\uD835\uDC00"      , "\uD835\uDC00" },
            { "\uD835\uDC1A"         , "\uD835\uDC00"      , ""             },
            { "\uD835\uDC00"         , "\uD835\uDC01"      , ""             },
            { "\uD801\uDC2B"         , "\uD801\uDC2B"      , "\uD801\uDC2B" },
            { "\uD801\uDC03"         , "\uD801\uDC03"      , "\uD801\uDC2B" },
            { "\uD801\uDC2B"         , "\uD801\uDC03"      , "\uD801\uDC2B" },
            { "\uD83D\uDE80"         , "\uD83D\uDE80"      , "\uD83D\uDE80" },
            { "a"                    , "aaaaaaaaaaaaaaaaa" , "a"            },
            // The last test should still work, and work fast, with a second
            // argument string starting with "a" and ending _many_ characters later
            // The last test values doe not test it, but illustrate the scenario
        };
        int maximumTestArgumentLength =
            Arrays.stream(testArgumentsAndExpectedValues)
                  .flatMap(testValues -> Arrays.stream(testValues)
                                               .limit(2))
                  .mapToInt(String::length)
                  .max()
                  .getAsInt();
        // @formatter:on
        int maximumQuotedTestArgumentLength = maximumTestArgumentLength + 2;
        for (int i = 0; i < testArgumentsAndExpectedValues.length; i++) {
          String[] currentTestValues = testArgumentsAndExpectedValues[i];
          String string1 = currentTestValues[0];
          String string2 = currentTestValues[1];
          String expectedResult = currentTestValues[2];
          String actualResult = commonCharactersOf(string1, string2);
          boolean testSuccessful = expectedResult.equals(actualResult);
          if (testSuccessful) {
            // continue; // TODO: uncomment to filter out successful tests
          }
          Function<String, String> quoteString = s -> '"' + s + '"';
          // @formatter:off
          String outputFormat = "%2d) "
                              + "%5s: "
                              + "(%-" + maximumQuotedTestArgumentLength + "s"
                              + " , "
                              + "%-" + maximumQuotedTestArgumentLength + "s)"
                              + " -> "
                              + "%s "
                              + "(%s)"
                              + "%n";
          System.out.printf(outputFormat,
                            i,
                            testSuccessful ? "Success" : "Failure",
                            quoteString.apply(string1),
                            quoteString.apply(string2),
                            quoteString.apply(actualResult),
                            quoteString.apply(expectedResult));
          // @formatter:on
        }
      }
    }

