All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.io7m.jequality.AlmostEqualFloat Maven / Gradle / Ivy

/*
 * Copyright © 2014 Mark Raynsford  https://www.io7m.com
 *
 * Permission to use, copy, modify, and/or distribute this software for any
 * purpose with or without fee is hereby granted, provided that the above
 * copyright notice and this permission notice appear in all copies.
 * 
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
 * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
 * IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 */

package com.io7m.jequality;

import com.io7m.jequality.annotations.EqualityReference;
import com.io7m.junreachable.UnreachableCodeException;

/**
 * 

Better floating point comparisons.

See Comparing floating point numbers 2012.

*/ @EqualityReference public final class AlmostEqualFloat { private AlmostEqualFloat() { throw new UnreachableCodeException(); } /** *

Compare the floating point numbers {@code x} and {@code y} using the * context {@code context}.

* *

Essentially, given {@code diff = abs(x - y)}, if {@code diff <= * context.getMaxRelativeDifference() * max(abs(x), abs(y))} , then {@code x} * and {@code y} can be considered to be close together.

* *

See the article mentioned at the top of this article for suggested * absolute and relative epsilon values: There are no values that work well * for all possible inputs.

* * @param context The equality context * @param x The left parameter * @param y The right parameter * * @return {@code true} if {@code x} and {@code y} are almost equal */ public static boolean almostEqual( final ContextRelative context, final float x, final float y) { if (x == Float.POSITIVE_INFINITY) { if (y == Float.POSITIVE_INFINITY) { return true; } } if (x == Float.NEGATIVE_INFINITY) { if (y == Float.NEGATIVE_INFINITY) { return true; } } final float diff = Math.abs(x - y); if (diff <= context.getMaxAbsoluteDifference()) { return true; } final float ax = Math.abs(x); final float ay = Math.abs(y); final float m = Math.max(ax, ay); final float k = m * context.getMaxRelativeDifference(); if (diff <= k) { return true; } return false; } /** * The necessary context for floating point comparisons. */ public static final class ContextRelative { private float max_absolute_diff; private float max_relative_diff; /** * Construct a new equality context. */ public ContextRelative() { this.max_absolute_diff = 0.0f; this.max_relative_diff = 0.0f; } /** * @return The current maximum absolute difference */ public float getMaxAbsoluteDifference() { return this.max_absolute_diff; } /** * Set the current maximum absolute difference * * @param max The required difference */ public void setMaxAbsoluteDifference( final float max) { this.max_absolute_diff = max; } /** * @return The current maximum relative difference */ public float getMaxRelativeDifference() { return this.max_relative_diff; } /** * Set the current maximum relative difference * * @param max The required difference */ public void setMaxRelativeDifference( final float max) { this.max_relative_diff = max; } @Override public String toString() { final StringBuilder builder = new StringBuilder(); builder.append("[ContextRelative [Absolute "); builder.append(this.max_absolute_diff); builder.append("] [Relative "); builder.append(this.max_relative_diff); builder.append("]]"); final String text = builder.toString(); assert text != null; return text; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy