在每种情况下使用带有值范围的 switch 语句?

问题描述 投票:0回答:20

在Java中,是否可以编写一个switch语句,其中每个case包含多个值?例如(尽管显然以下代码不起作用):

switch (num) {
    case 1 .. 5:
        System.out.println("testing case 1 to 5");
        break;
    case 6 .. 10:
        System.out.println("testing case 6 to 10");
        break;
}

我认为这可以在 Objective C 中完成,Java 中有类似的事情吗?或者我应该只使用

if
,
else if
语句来代替?

java if-statement switch-statement
20个回答
117
投票

Java 没有这样的东西。为什么不直接执行以下操作呢?

public static boolean isBetween(int x, int lower, int upper) {
  return lower <= x && x <= upper;
}

if (isBetween(num, 1, 5)) {
  System.out.println("testing case 1 to 5");
} else if (isBetween(num, 6, 10)) {
  System.out.println("testing case 6 to 10");
}

82
投票

使用

switch
语句最接近这种行为的是

switch (num) {
case 1:
case 2:
case 3:
case 4:
case 5:
     System.out.println("1 through 5");
     break;
case 6:
case 7:
case 8:
case 9:
case 10:
     System.out.println("6 through 10");
     break;
}

使用

if
语句。


36
投票

另一种选择是通过除法进行数学运算,例如:

switch ((int) num/10) {
    case 1:
        System.out.println("10-19");
        break;
    case 2:
        System.out.println("20-29");
        break;
    case 3:
        System.out.println("30-39");
        break;
    case 4:
        System.out.println("40-49");
        break;
    default:
        break;
}

但是,正如您所看到的,只有在每种情况下范围固定时才能使用。


20
投票

我知道这篇文章很旧,但我相信这个答案值得一些认可。没有必要避免使用 switch 语句。这可以在java中完成,但通过switch语句,而不是cases。它涉及使用三元运算符。

public class Solution {

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int num = Integer.parseInt(sc.nextLine());

        switch ((1 <= num && num <= 5 ) ? 0 :
                (6 <= num && num <= 10) ? 1 : 2) {

            case 0:
                System.out.println("I'm between one and five inclusive.");
                break;
            case 1:
                System.out.println("I'm between 6 and 10 inclusive.");
                break;
            case 2:
                System.out.println("I'm not between one and five or 6 and 10 inclusive.");
                break;
        }
    }
}

15
投票

我认为你不能在 Java 中做到这一点。最好的办法是将代码放在范围的最后一个案例中。

switch (num) {
  case 1: case 2: case 3: case 4: case 5: 
     System.Out.Println("testing case 1 to 5");
     break;
  case 6: case 7: case 8: case 9: case 10:
     System.Out.Println("testing case 6 to 10");
     break;
  default:
     //
}

12
投票
  case 1: case 2: case 3: case 4: case 5: 
         System.out.println("testing case 1 to 5");
         break;
  case 6: case 7: case 8: case 9: case 10:
         System.out.println("testing case 6 to 10");
         break;
  default:
         System.out.println("default"); 

7
投票

读完所有评论后,我没有看到任何人提到增强开关其中 在一种情况下您可以有多个值,如下所示 ->

switch(value){
   case 1,2,3,4:
      //dosth
      break;
   case 7,9,10,23:
      //dosth
      break;
}

并且由于在您的情况下,每种情况下只有一个表达式,因此您可以执行以下操作,而无需

break
每种情况->

switch (value) {
    case 1, 2, 3, 4 -> System.out.println("one of 1,2,3,4 matched");
    case 7, 9, 10, 23 -> System.out.println("one of 7,9,10,23 matched");
}

这是 java 中增强开关的众多附加好处之一。


7
投票

Java 12+

从 Java 12 开始支持它。查看 JEP 354。这里没有“范围”的可能性,但也很有用。

switch (day) {
    case MONDAY, FRIDAY, SUNDAY -> System.out.println(6);//number of letters
    case TUESDAY                -> System.out.println(7);
    case THURSDAY, SATURDAY     -> System.out.println(8);
    case WEDNESDAY              -> System.out.println(9);
}

您也应该能够在整数上实现这一点。请注意,您的 switch 语句必须详尽无遗(使用

default
关键字,或在 case 语句中使用所有可能的值)。

Java 20+

更新: 从 Java 20(23 年 3 月)开始,这是可能的! 查看JEP 433。他们的例子是,如果您想测试特定大小的三角形。

switch (s) {
        case null -> 
            { break; }
        case Triangle t
        when t.calculateArea() > 100 ->
            System.out.println("Large triangle");
        default ->
            System.out.println("A shape, possibly a small triangle");
    }

6
投票

如果您必须使用开关,请尝试此操作。

public static int range(int num){ 
    if ( 10 < num && num < 20)
        return 1;
    if ( 20 <= num && num < 30)
        return 2;
    return 3;
}

public static final int TEN_TWENTY = 1;
public static final int TWENTY_THIRTY = 2;

public static void main(String[]args){
    int a = 110;
    switch (range(a)){
        case TEN_TWENTY: 
            System.out.println("10-20"); 
            break;
        case TWENTY_THIRTY: 
            System.out.println("20-30"); 
            break;
        default: break;
    }
}

6
投票

或者您可以按预期使用单独的情况,并使用默认情况将范围指令指定为:

switch(n) {
    case 1 : System.out.println("case 1"); break;
    case 4 : System.out.println("case 4"); break;
    case 99 : System.out.println("case 99"); break;
    default :
        if (n >= 10 && n <= 15)
            System.out.println("10-15 range"); 
        else if (n >= 100 && n <= 200)
            System.out.println("100-200 range");
        else
            System.out.println("Your default case");
        break;   
}

4
投票

可以使用 switch 语句允许的

fall through
机制将多个条件分组在同一个 case 语句中,它在 Java 教程 中提到,并在第 §14.11 节中详细说明。 Java 语言规范的 switch 语句

以下代码片段取自教程中的示例,它计算每个月的天数(从 1 月到 12 月编号):

switch (month) {
    case 1: case 3: case 5:
    case 7: case 8: case 10:
    case 12:
        numDays = 31;
        break;
    case 4: case 6:
    case 9: case 11:
        numDays = 30;
        break;
    case 2:
        if (((year % 4 == 0) && 
             !(year % 100 == 0))
             || (year % 400 == 0))
            numDays = 29;
        else
            numDays = 28;
        break;
    default:
        System.out.println("Invalid month.");
        break;
}

如您所见,为了在单个

case
语句中涵盖一系列值,唯一的选择是逐一列出每个可能的值。作为另一个示例,以下是如何实现问题中的伪代码:

switch(num) {
    case 1: case 2: case 3: case 4: case 5:
        System.out.println("testing case 1 to 5");
        break;
    case 6: case 7: case 8: case 9: case 10:
        System.out.println("testing case 6 to 10");
        break;
}

4
投票

不,你不能那样做。你能做的最好的就是

case 1:
case 2:
case 3:
case 4:
case 5: 
  System.Out.Println("testing case 1 to 5");
break;

4
投票

使用

NavigableMap
实现,例如
TreeMap

/* Setup */
NavigableMap<Integer, Optional<String>> messages = new TreeMap<>();
messages.put(Integer.MIN_VALUE, Optional.empty());
messages.put(1, Optional.of("testing case 1 to 5"));
messages.put(6, Optional.of("testing case 6 to 10"));
messages.put(11, Optional.empty());

/* Use */
messages.floorEntry(3).getValue().ifPresent(System.out::println);

4
投票

这是一种美丽而简约的方式

(num > 1 && num < 5) ? first_case_method() 
                     : System.out.println("testing case 1 to 5")
                     : (num > 5 && num < 7)  ? System.out.println("testing case 5 to 7") 
                     : (num > 7 && num < 8)  ? System.out.println("testing case 7 to 8") 
                     : (num > 8 && num < 9)  ? System.out.println("testing case 8 to 9") 
                     : ... 
                     : System.out.println("default");

3
投票

您可以使用

enum
来表示您的范围,

public static enum IntRange {
  ONE_TO_FIVE, SIX_TO_TEN;
  public boolean isInRange(int v) {
    switch (this) {
    case ONE_TO_FIVE:
      return (v >= 1 && v <= 5);
    case SIX_TO_TEN:
      return (v >= 6 && v <= 10);
    }
    return false;
  }

  public static IntRange getValue(int v) {
    if (v >= 1 && v <= 5) {
      return ONE_TO_FIVE;
    } else if (v >= 6 && v <= 10) {
      return SIX_TO_TEN;
    }
    return null;
  }
}

2
投票

@missingfaktor 的答案确实是正确的,但有点过于复杂。 代码比它可能的更冗长(至少对于连续间隔),并且需要重载/强制转换和/或长整型、浮点型、整数等的参数化

if (num < 1)
    System.Out.Println("invalid case: " + num); // you should verify that anyway
else if (num <= 5)
    System.Out.Println("1 to 5");
else if (num <= 10)
    System.Out.Println("6 to 10");
else if (num <= 42)
    System.Out.Println("11 to 42");
else    
    System.Out.Println("43 to +infinity");

1
投票

对于输入0..100范围内的数字

int n1 = 75; // input value
String res; int n=0; 
int[] a ={0,20,35,55,70,85,101};

for(; n1>=a[n]; n++);
switch(6-n+1) {
  case 1: res="A"; break;
  case 2: res="B"; break;
  case 3: res="C"; break;
  case 4: res="D"; break;
  case 5: res="E"; break;
  default:res="F";
}
System.out.println(res);

1
投票

除了 switch 语句之外,从 Java 14 SE 开始,还有 switch 表达式,当每种情况产生一些结果时,您可以使用它作为替代方案。例如,如果处理非整数,可以使用

Math.round()

String result = switch (Math.round(value)) {
    case 1, 2, 3, 4, 5:
        // handle case 1
        yield "result of case 1";
    case 6, 7, 8, 9, 10:
        // handle case 2
        yield "result of case 2";
    default:
        // handle the default case
        yield "result of the default case";
}

System.out.println("result = " + result);

与 switch 语句的区别在于必须覆盖所有情况(通常使用

default
情况)。


0
投票

Java 不支持这种类型的行为。但是,如果您有一个需要此功能的大型项目,请考虑在您的项目中混合 Groovy 代码。 Groovy 代码被编译为字节代码并可以在 JVM 中运行。我工作的公司使用 Groovy 编写服务类,使用 Java 编写其他所有内容。


0
投票

做出了杰出的贡献。如果它仍然有帮助,我找到了一个选项来分析字符串中是否存在不同于字母和空格的字符:

string name = scEntry.nextLine();
int len = name.length(); 
int s = 0;
do { 
     int asc = name.codePointAt(s);
     switch ((asc == 32) ? 0 : (64 < asc && asc < 91) ? 1 : 
             (96 < asc && asc < 123) ? 2 : 99) {
         case 0, 1, 2 -> {
                  s++; 
                  break;
         }
         case 99 -> {
           s = 0;
           System.out.println("Type in only letters [A-Za-z]
                             or spaces");
           System.out.print("Type the user name again: ");
           name = scEntry.nextLine();
           len = name.length();
           break;
         }
} while (s < len);
s = 0;

在此示例中,字母的 ASCII 代码与“空格”ASCII 代码一起分组并用数字 0、1 或 2 表示。 与上述不同的 ASCII 代码被分组在值“99”中,因此我可以向用户告知有关它们的任何信息。

© www.soinside.com 2019 - 2024. All rights reserved.