如何从Java中的颜色名称获取十六进制格式?

时间:2019-04-21 00:02:30

标签: java java-8

我有颜色名称,例如白色,红色,蓝色等,并且我正在尝试获取其对应的十六进制表示形式。例如,如果我传递诸如White之类的参数,则它应返回#FFFFFF

尽管看起来很简单,但是Java中没有直接的解决方案。以下是我尝试过的代码。

private String getHexaColor (String basicColor /* White */) {
    java.awt.Color color;
    try {
        java.lang.reflect.Field field = Class.forName("java.awt.Color").getField(basicColor.toLowerCase());
        color = (Color) field.get(null);
    } catch (Exception e) {
        color = null; 
    }

    return (color != null) ? String.format("%06x#", color.getRGB() & 0x00FFFFFF).toUpperCase() : ""; //return #FFFFFF
}

上面的代码运行良好,但是我正在寻找一种干净优雅的解决方案,至少在最新版本的Java中可用。

2 个答案:

答案 0 :(得分:2)

您应该使用格式字符串"#%06X"而不是"%06x#"。首先,它将#字符放置在您实际想要的位置,其次,它生成一个大写的十六进制字符串,从而无需在结果上调用toUpperCase()

但是,与显式的值映射相比,反射操作可以节省多少费用?当我在Color的所有字段上运行时,例如通过

final int constant = Modifier.STATIC | Modifier.FINAL;
Map<String,Integer> m = new HashMap<>();
for(Field f: Color.class.getFields()) try {
    if((f.getModifiers() & constant) == constant && f.getType() == Color.class)
        m.put(f.getName().toLowerCase().replace('_', ' '),
            ((Color)f.get(null)).getRGB() & 0x00FFFFFF);
} catch(ReflectiveOperationException ex) {
    throw new AssertionError(ex);
}
m.forEach((name,rgb) -> System.out.printf("map.put(\"%s\", \"#%06X\");%n", name, rgb));

我明白了

map.put("magenta", "#FF00FF");
map.put("pink", "#FFAFAF");
map.put("green", "#00FF00");
map.put("black", "#000000");
map.put("yellow", "#FFFF00");
map.put("cyan", "#00FFFF");
map.put("dark gray", "#404040");
map.put("red", "#FF0000");
map.put("orange", "#FFC800");
map.put("gray", "#808080");
map.put("white", "#FFFFFF");
map.put("blue", "#0000FF");
map.put("darkgray", "#404040");
map.put("light gray", "#C0C0C0");
map.put("lightgray", "#C0C0C0");

那不是那么大。因此,当我将输出复制到新代码中时,例如

static final Map<String,String> COLOR_CODES;
static {
    Map<String,String> map = new HashMap<>();
    map.put("magenta", "#FF00FF");
    map.put("pink", "#FFAFAF");
    map.put("green", "#00FF00");
    map.put("black", "#000000");
    map.put("yellow", "#FFFF00");
    map.put("cyan", "#00FFFF");
    map.put("dark gray", "#404040");
    map.put("red", "#FF0000");
    map.put("orange", "#FFC800");
    map.put("gray", "#808080");
    map.put("white", "#FFFFFF");
    map.put("blue", "#0000FF");
    map.put("darkgray", "#404040");
    map.put("light gray", "#C0C0C0");
    map.put("lightgray", "#C0C0C0");
    COLOR_CODES = Collections.unmodifiableMap(map);
}
static String getHexaColor(String colorName) {
    return COLOR_CODES.getOrDefault(colorName.toLowerCase(), "");
}

结果简单且无反射。维护将独立于产生此版本的反射代码,因为向映射添加新常量比向java.awt.Color类添加新常量(重新运行反射代码)更简单。 / p>

唯一更具灵活性的选择是使用properties文件,该文件允许更新地图而无需重新编译应用程序。改编上面的代码以生成属性文件很容易,但是另一方面,互联网上有足够的即用型文件可用,甚至还有更多颜色。例如,您可以直接从w3.org复制基于VGA的Web颜色,而this extended list则需要将高级搜索和替换功能带入属性格式。

答案 1 :(得分:0)

使用OpenFX中的Color,为所有Web颜色设置一种方法。

import javafx.scene.paint.Color;
Color color = Color.web("skyblue");

结果:

0x87cdebff

这些也作为常量字段存在,因此可以使用自动补全。但是我喜欢CSS / HTML编辑器中的自动补全功能。


这是一个单独的依赖库。但是从Java 9开始,可以期待模块化的基础结构,而以前的JavaFX的OpenFX毫无问题。