9

在 Java 方法中指定标志的最佳实践是什么?

我见过 SWT 使用 int 作为位域,例如:

(示例部分来自“Effective Java, 2nd Ed.” 第 159 页):

public class Text {
  public static final int STYLE_BOLD = 1 << 0; // 1
  public static final int STYLE_ITALIC = 1 << 1; // 2

  void printText(String text, int flags) {

  }
}

你的客户电话看起来像:

printText("hello", Text.STYLE_BOLD | Text.STYLE_ITALIC);

..但不鼓励这样做,因为您可以将来自不同类的标志(int 值)混合在一起而无需任何编译器检查。

在同一本书(“Effective Java”)中,我看到了 EnumSet 的使用,但随后您的用户调用变为:

printText("hello", EnumSet.of(Style.Bold, Style.ITALIC));

我觉得这有点冗长,我更喜欢 SWT 的优雅。

还有其他选择吗,或者这基本上是您必须选择的两种口味?

4

5 回答 5

7

猜你碰壁了。我没有看到任何其他选择。Java 很冗长,这是事实。在这种情况下,我通常会添加一个局部变量以使代码更具可读性。你可以这样做,

EnumSet<Style> styles = EnumSet.of(Style.Bold, Style.ITALIC);
printText("hello", styles);
于 2011-05-20T14:30:38.727 回答
2

如果您想要位样式标志,Java 将它们包装在BitSet. 它已经存在了很长时间,但很少有人愿意使用它(更喜欢在整数中嵌入 C 风格的位处理)。

BitSet 的 api 可以在这里找到

再加上一些精心挑选的静态整数,在您开始一次检查和设置多个位之前,它做得很好。

于 2011-05-20T14:40:07.387 回答
2

我建议你采用这种EnumSet方法。

EnumSet<Style> styles = EnumSet.of(Style.Bold, Style.Italic);

这种方法提供了更好的类型安全性,并且Style作为枚举将具有成熟的 OO 功能。

于 2011-05-20T15:22:00.590 回答
1

对于遇到此问题的任何人的迟到答案。这是一种减少内存并拥有像api这样的漂亮枚举的方法:

public static class MyFlag {

    public static final MyFlag A = new MyFlag(1<<0);
    public static final MyFlag B = new MyFlag(1<<1);
    public static final MyFlag C = new MyFlag(1<<2);
    public static final MyFlag ALL = A.and(B).and(C);

    private final int flag;

    private MyFlag(int flag){
        this.flag = flag;
    }

    public MyFlag and(MyFlag limit){
        return new MyFlag(flag & limit.flag);
    }

    public MyFlag not(MyFlag limit){
        return new MyFlag(flag | ~limit.flag);
    }

    public boolean isSet(MyFlag limit){
        if(limit ==null){
            return false;
        }
        return (this.flag & limit.flag) != 0;
    }
}

方法:

public void doFoo(MyFlag flag){
   if(MyFlag.A.isSet(flag)){
   ....
   }
   if(MyFlag.C.isSet(flag)){
   ....
   }
}

称呼:

x.doFoo(MyFlag.A.and(MyFlag.C));
于 2015-04-20T07:22:08.897 回答
0

如果您只有有限数量的方法将采用一组样式(例如printText,在您的示例中),您可以调整它们的签名以采用可变数量的样式参数:

void printText(String text, Style... flags) {
  EnumSet<Style> style = logicalOr(flags); // see comment below
  ...
 }

然后你的电话非常接近无类型(int)标志路线:

printText("hello", Style.BOLD, Style.ITALIC);

可悲的是,没有EnumSet.of(E... )工厂,只是EnumSet.of(E first, E... more),所以您需要一个通用logicalOr方法将您的数组拆分为 first + rest 块。 留给读者作为练习 =)

于 2017-10-18T19:08:34.420 回答