Search code examples
javadesign-patternsoptimizationrefactoringcommand-pattern

Long list of if comparisons in java


I need to compare two Objects. If there is a difference I need to log it corresponding to particular difference and return the true.

For example:

private boolean compTwoObjects(Object objA, Object ObjB) {
   if(objA.getType() != objB.getType()) {
      logTheDifference("getType is differing");
      return true;
   }
                   .
                   .
                   .
   // Now this could invoke other composite methods
   if(checkFont(objA.getFont(), objB.getFont()) {
      logTheDifference("Font is differing");
      return true;
   }
}


private boolean checkFont(Font fontObjA, Font fontObjB) {
   if(fontObjA.getBold() != fontObjB.getBold()) {
      logTheDifference("font bold formatting differs");
      return true;
   }
                   .
                   .
                   .
   if(fontObjA.getAllCaps() != fontObjB.getAllCaps()) {
      logTheDifference("font all caps formatting differs");
      return true;
   }
                   .
                   .
                   .
   if(checkBorderDiff(fontObjA.getBorder(), fontObjB.getBorder())) {
      logTheDifference("border diff");
      return true;
   }
}

private boolean checkBorderDiff(Border borderObjA, Border borderObjB) {
    if (borderObjA.getColor() != null || borderObjB.getColor() != null) {
       if (!borderObjA.getColor().equals(borderObjB.getColor())) {
            logIt("border color differing");
            return true;
        }
    }

    if (borderObjA.getDistanceFromText() != borderObjB.getDistanceFromText()) {
        logIt("distance of the border from text or from the page edge in points differing");
        return true;
    }

    if (borderObjA.isVisible() != borderObjB.isVisible()) {
        logIt("border visibility differing");
        return true;
    }

    if (borderObjA.getLineStyle() != borderObjB.getLineStyle()) {
        logIt("line style differing for border");
        return true;
    }

    if (borderObjA.getLineWidth() != borderObjB.getLineWidth()) {
        logIt("border width in points differing");
        return true;
    }

    if (borderObjA.getShadow() != borderObjB.getShadow()) {
        logIt("border shadow differing");
        return true;
    }
}

//And it is going like this.

My problem is I want to avoid multiple if statements in the methods. Also I want to log the messages corresponding to particular difference.

I have read few similar type of problems on stackoverflow solved either by command pattern or HashMap. But they don't include comparisons in that.

I want to refactor my code to get rid of series of if's.


Solution

  • Have a system of comparators, backed by generics. Every comparer will also know what is next in line. For example:

    interface IComparer<T> {
        boolean areDifferent (T first, T second);
    }
    
    class FontComparer implements IComparer<Font> {
        @Override
        public boolean areDifferent(Font first, Font second) {
            // Compare fonts start
            // ..
            // Compare fonts end
            return new BorderComparer().areDifferent(first.getBorder(), second.getBorder());
        }
    }
    
    
    class BorderComparer implements IComparer<Border> {
    
        @Override
        public boolean areDifferent(Border first, Border second) {
            //Do border comparison alone
            return false;
        }
    }
    

    You could setup a comparer chain now, and bail out when comparison fails. Otherwise, comparison goes to the comparer next in the chain.

    The client code will finally look like:

        Object one = new Object();
        Object two = new Object();
        new ObjectComparer().areDifferent(one, two);