Java中&&和?表达式结合时会出现的坑

  首先是背景,刚放假回家比较闲,就把以前写了一些算法题的一个项目拿出来继续写,想把其中的插入排序修改成支持升序和降序的,然后就出现了这个坑,具体是这样的:

  先把插入排序的代码摆出来吧。

    /**
     * 插入排序
     * @param arr 输入数组
     * @param order 顺序 1为升序 0为降序
     */
    static void insertionSort(int arr[],int order){

        for (int i = 1; i < arr.length; i++)
        {
            int get = arr[i];
            int j = i - 1;
            while (j >= 0 && (order == 1) ? (arr[j] > get):(arr[j] < get))
            {
                arr[j + 1] = arr[j];
                j--;
            }
            arr[j + 1] = get;
        }
    }

  main函数是这样调用的:

    public static void main(String[] args){

        int[] arr = {8,96,23,5,6,43};
        for(int a :arr){
            System.out.print(a + ",");
        }
        System.out.println();
        insertionSort(arr,1);

        for(int a :arr){
            System.out.print(a + ",");
        }
        System.out.println();

        insertionSort(arr,0);

        for(int a :arr){
            System.out.print(a + ",");
        }

    }

  运行后日志是这样的:

8,96,23,5,6,43,
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: -1

  异常的意思是说数组越界了,且问题出在这一行

while (j >= 0 && (order == 1) ? (arr[j] > get):(arr[j] < get))

  代码中j每次循环都会减1直到这两个条件都不满足为止,debug后发现是j=-1的时候出现的异常,但问题是j=-1的时候,不会去使用数组啊,因为众所周知&&属于短路操作,即如果第一个操作数能够决定结果,那么就不会再对第二个操作数求值,也就是说j=-1的时候后面的表达式是不会计算的啊,但这里进行计算了,从数组中取值了,所以出现了这个异常。

  我也随便写了一些代码测试了一下这种情况:

    /**
     * 对比两个输入参数的大小
     * @param a 输入参数1
     * @param b 输入参数2
     * @return boolean 如果a > b 返回true,反之返回false
     */
    static boolean compare(int a,int b){
            System.out.println(a + ">" + b + "?");
            System.out.println(a > b);
            return a > b;

    }
    public static void main(String[] args){
        int a = 1;
        int b = 2;
        int c = 3;
        boolean result = compare(a ,a) && (a == 1) ? (compare(b,b)):(compare(c,c));

        System.out.println();

        result = compare(b ,b) && compare(c ,c);

    }

  这里有一个对比大小的函数,这个函数会打出日志来让我们清晰的看到&&前后的表达式运行了没有,main函数中有两个&&表达式,

  第一个&&表达式中B是一个?表达式,第二的个&&表达式的B就是一个compare函数,日志结果是:

1>1?
false
3>3?
false

2>2?
false

  从日志结果我们可以清晰的看到,当B是?表达式的时候,A不成立的时候下B依旧运行了,而B不是?表达式的时候,A是false的情况下B是不会执行的。

原文地址:https://www.cnblogs.com/xxbbtt/p/10346564.html

时间: 2024-10-15 18:30:02

Java中&&和?表达式结合时会出现的坑的相关文章

Java中的表达式运算

1.问题背景    以下代码运行的结果为:    A.a的值:8   b的值:7           B.a的值:8  b的值:8    C.a的值:9   b的值:7           D.a的值:9  b的值:8 /** * */ package com.you.model; /** * @author YouHaidong * */ public class AB { /** * @param args */ public static void main(String[] args)

Java中Lambda表达式的使用

简介(译者注:虽然看着很先进,其实Lambda表达式的本质只是一个"语法糖",由编译器推断并帮你转换包装为常规的代码,因此你可以使用更少的代码来实现同样的功能.本人建议不要乱用,因为这就和某些很高级的黑客写的代码一样,简洁,难懂,难以调试,维护人员想骂娘.)Lambda表达式是Java SE 8中一个重要的新特性.lambda表达式允许你通过表达式来代替功能接口. lambda表达式就和方法一样,它提供了一个正常的参数列表和一个使用这些参数的主体(body,可以是一个表达式或一个代码块

Java中lambda(λ)表达式的语法

举一个排序的例子,我们传入代码来检查一个字符串是否比另一个字符串短.这里要计算: first.length() - second.length() first和second是什么?他们都是字符串.Java是一个强类型语言,所以我们还要指定他们的类型: (String first, String second) -> first.length() - second.length() lambda表达式就是一个代码块,以及必须传入代码的变量规范. 你已经见过Java中的一种lambda表达式形式:参

Java中Lambda表达式的使用(转)

https://www.cnblogs.com/franson-2016/p/5593080.html 简介(译者注:虽然看着很先进,其实Lambda表达式的本质只是一个"语法糖",由编译器推断并帮你转换包装为常规的代码,因此你可以使用更少的代码来实现同样的功能.本人建议不要乱用,因为这就和某些很高级的黑客写的代码一样,简洁,难懂,难以调试,维护人员想骂娘.)Lambda表达式是Java SE 8中一个重要的新特性.lambda表达式允许你通过表达式来代替功能接口. lambda表达式

java中lamda表达式的应用

lamda表达式主要是为了解决匿名内部类的繁琐过程 范例:简单的lamda表达式 此处使用匿名内部类 package com.java.demo; interface IMessage{ public void print();//抽象方法 } public class TestDemo{ public static void main(String args[]){ //使用给匿名内部类实现print()方法 fun(new IMessage(){ public void print(){ S

java 8 中lambda表达式学习

转自 http://blog.csdn.net/renfufei/article/details/24600507 http://www.jdon.com/idea/java/10-example-of-lambda-expressions-in-java8.html Lambda表达式的语法基本语法:(parameters) -> expression或(parameters) ->{ statements; } 下面是Java lambda表达式的简单例子: // 1. 不需要参数,返回值

Java中的switch语句后面的控制表达式的数据类型

Java中的switch语句后面的控制表达式的数据类型 1.byte 2.char 3.short 4.int 5.枚举类型 6.Java 7允许java.lang.String类型

java、el表达式中保留小数的方法

Java中: import java.math.BigDecimal; import java.text.DecimalFormat; import java.text.NumberFormat; public class format { double f = 111231.5585; public void m1() { BigDecimal bg = new BigDecimal(f); double f1 = bg.setScale(2, BigDecimal.ROUND_HALF_UP

利用 Lambda 表达式实现 Java 中的惰性求值

Java 中惰性求值的潜能,完全被忽视了(在语言层面上,它仅被用来实现 短路求值 ).更先进的语言,如 Scala,区分了传值调用与传名调用,或者引入了 lazy 这样的关键字. 尽管 Java 8 通过延迟队列的实现(java.util.stream.Stream)在惰性求值的方面有些改进,但是我们会先跳过 Stream,而把重点放在如何使用 lambda 表达式实现一个轻量级的惰性求值. 基于 lambda 的惰性求值 Scala 当我们想对 Scala 中的方法参数进行惰性求值时,我们用"