//---------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//
// @owner [....]
// @backupOwner [....]
//---------------------------------------------------------------------
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Diagnostics;
namespace System.Data.Common.Utils.Boolean
{
///
/// Represents a variable with finite domain, e.g., c in {1, 2, 3}
///
/// Type of domain variables (int in the above example).
/// Type of the identifier (c above -- it need not be int).
internal class DomainVariable
{
private readonly T_Variable _identifier;
private readonly Set _domain;
private readonly int _hashCode;
private readonly IEqualityComparer _identifierComparer;
///
/// Constructs a new domain variable.
///
/// Identifier
/// Domain of variable.
/// Comparer of identifier
internal DomainVariable(T_Variable identifier, Set domain, IEqualityComparer identifierComparer)
{
Debug.Assert(null != identifier && null != domain);
_identifier = identifier;
_domain = domain.AsReadOnly();
_identifierComparer = identifierComparer ?? EqualityComparer.Default;
int domainHashCode = _domain.GetElementsHashCode();
int identifierHashCode = _identifierComparer.GetHashCode(_identifier);
_hashCode = domainHashCode ^ identifierHashCode;
}
internal DomainVariable(T_Variable identifier, Set domain) : this(identifier, domain, null) { }
///
/// Gets the variable.
///
internal T_Variable Identifier { get { return _identifier; } }
///
/// Gets the domain of this variable.
///
internal Set Domain { get { return _domain; } }
public override int GetHashCode()
{
return _hashCode;
}
public override bool Equals(object obj)
{
if (Object.ReferenceEquals(this, obj)) { return true; }
DomainVariable other = obj as DomainVariable;
if (null == other) { return false; }
if (_hashCode != other._hashCode) { return false; }
return (_identifierComparer.Equals(_identifier, other._identifier) && _domain.SetEquals(other._domain));
}
public override string ToString()
{
return StringUtil.FormatInvariant("{0}{{{1}}}",
_identifier.ToString(), _domain);
}
}
///
/// Represents a constraint of the form:
///
/// Var1 in Range
///
/// Type of range elements.
/// Type of the variable.
internal class DomainConstraint
{
private readonly DomainVariable _variable;
private readonly Set _range;
private readonly int _hashCode;
///
/// Constructs a new constraint for the given variable and range.
///
/// Variable in constraint.
/// Range of constraint.
internal DomainConstraint(DomainVariable variable, Set range)
{
Debug.Assert(null != variable && null != range);
_variable = variable;
_range = range.AsReadOnly();
_hashCode = _variable.GetHashCode() ^ _range.GetElementsHashCode();
}
///
/// Constructor supporting a singleton range domain constraint
///
internal DomainConstraint(DomainVariable variable, T_Element element)
: this(variable, new Set(new T_Element[] { element }).MakeReadOnly())
{
}
///
/// Gets the variable for this constraint.
///
internal DomainVariable Variable { get { return _variable; } }
///
/// Get the range for this constraint.
///
internal Set Range { get { return _range; } }
///
/// Inverts this constraint (this iff. !result)
/// !(Var in Range) iff. Var in (Var.Domain - Range)
///
///
internal DomainConstraint InvertDomainConstraint()
{
return new DomainConstraint(_variable,
_variable.Domain.Difference(_range).AsReadOnly());
}
public override bool Equals(object obj)
{
if (Object.ReferenceEquals(this, obj)) { return true; }
DomainConstraint other = obj as DomainConstraint;
if (null == other) { return false; }
if (_hashCode != other._hashCode) { return false; }
return (_range.SetEquals(other._range) && _variable.Equals(other._variable));
}
public override int GetHashCode()
{
return _hashCode;
}
public override string ToString()
{
return StringUtil.FormatInvariant("{0} in [{1}]",
_variable, _range);
}
}
}