4 votes

Distance (pieds, pouces) classe ou structure

Est-ce que quelqu'un connaît des exemples ou du code qui montre une classe ou une structure qui peut être utilisée pour la distance comme la structure DateTime ? Je dois être en mesure d'ajouter, de soustraire et d'afficher les données en pieds et en pouces et l'utilisation de méthodes de conversion devient désordonnée. Une classe ou une structure serait parfaite, mais je n'ai rien trouvé dans mes recherches.

13voto

280Z28 Points 49515

Utilisez une structure, mais rendez-la immuable (toutes les propriétés ne peuvent être obtenues qu'à l'aide de la fonction get).

Les propriétés doivent comprendre au moins :

  • TotalFeet
  • TotalInches

Les méthodes doivent inclure au moins :

  • FromFeet (statique)
  • FromInches (statique)

Déclarez le champ de sauvegarde privé comme :

private readonly double _meters;

Edit : Peut-être quelque chose comme ça.

public struct Distance : IEquatable<Distance>, IComparable<Distance>
{
    private static readonly double MetersPerKilometer = 1000.0;
    private static readonly double CentimetersPerMeter = 100.0;
    private static readonly double CentimetersPerInch = 2.54;
    private static readonly double InchesPerFoot = 12.0;
    private static readonly double FeetPerYard = 3.0;
    private static readonly double FeetPerMeter = CentimetersPerMeter / (CentimetersPerInch * InchesPerFoot);
    private static readonly double InchesPerMeter = CentimetersPerMeter / CentimetersPerInch;

    private readonly double _meters;

    public Distance(double meters)
    {
        this._meters = meters;
    }

    public double TotalKilometers
    {
        get
        {
            return _meters / MetersPerKilometer;
        }
    }

    public double TotalMeters
    {
        get
        {
            return _meters;
        }
    }

    public double TotalCentimeters
    {
        get
        {
            return _meters * CentimetersPerMeter;
        }
    }

    public double TotalYards
    {
        get
        {
            return _meters * FeetPerMeter / FeetPerYard;
        }
    }

    public double TotalFeet
    {
        get
        {
            return _meters * FeetPerMeter;
        }
    }

    public double TotalInches
    {
        get
        {
            return _meters * InchesPerMeter;
        }
    }

    public static Distance FromKilometers(double value)
    {
        return new Distance(value * MetersPerKilometer);
    }

    public static Distance FromMeters(double value)
    {
        return new Distance(value);
    }

    public static Distance FromCentimeters(double value)
    {
        return new Distance(value / CentimetersPerMeter);
    }

    public static Distance FromYards(double value)
    {
        return new Distance(value * FeetPerYard / FeetPerMeter);
    }

    public static Distance FromFeet(double value)
    {
        return new Distance(value / FeetPerMeter);
    }

    public static Distance FromInches(double value)
    {
        return new Distance(value / InchesPerMeter);
    }

    public static Distance operator +(Distance a, Distance b)
    {
        return new Distance(a._meters + b._meters);
    }

    public static Distance operator -(Distance a, Distance b)
    {
        return new Distance(a._meters - b._meters);
    }

    public static Distance operator -(Distance a)
    {
        return new Distance(-a._meters);
    }

    public override bool Equals(object obj)
    {
        if (!(obj is Distance))
            return false;

        return Equals((Distance)obj);
    }

    public bool Equals(Distance other)
    {
        return this._meters == other._meters;
    }

    public int CompareTo(Distance other)
    {
        return this._meters.CompareTo(other._meters);
    }

    public override int GetHashCode()
    {
        return _meters.GetHashCode();
    }

    public override string ToString()
    {
        return string.Format("{0}[m]", TotalMeters);
    }
}

8voto

Reed Copsey Points 315315

J'ai déjà écrit des classes de convertisseurs d'unités autonomes, mais je n'en connais pas de bonnes pour .NET qui soient publiques.

Ceci étant dit, il est assez facile à écrire - Il suffit de créer une structure qui peut être construite en pouces ou en pieds, et convertible dans les deux cas.

public struct Distance
{
     private Distance(int inches)
     {
         this.totalInches = inches;
     }

     private int totalInches;

     public int Inches { get { return this.totalInches % 12; }  }

     public int Feet { get { return this.totalInches / 12; } }

     public static Distance FromInches(int inches)
     {
          return new Distance(inches);
     }

     public static Distance FromFeet(int feet)
     {
          return new Distance(feet * 12);
     }

     public static Distance FromFeetAndInches(int feet, int inches)
     {
          return new Distance(feet * 12 + inches);
     }

}

0voto

Ed Power Points 3179
public class Length
{
    public double Inches { get; set; }

    public double Feet
    {
        get { return Inches / 12.0; }
        set { Inches = value * 12.0; }
    }

    public double Meters
    {
        get { return Inches / 39.3700787; }
        set { Inches = value * 39.3700787; }
    }

    public double Furlongs
    {
        get { return Feet / 660.0; }
        set { Feet = value * 660.0; }
    }

    public double Miles
    {
        get { return Furlongs / 8.0; }
        set { Furlongs = value * 8.0; }
    }
}

0voto

mattmc3 Points 6768

Je recommande de faire deux classes pour la distance. L'une avec les mesures impériales et l'autre avec les mesures métriques. Ainsi, vous pourrez facilement faire des conversions entre les deux, avec l'inconvénient évident de perdre en précision.

Voici un exemple de classe de distance impériale, avec les pouces comme unité de mesure de base.

    public class ImperialDistance {

    public static readonly ImperialDistance Inch = new ImperialDistance(1.0);
    public static readonly ImperialDistance Foot = new ImperialDistance(12.0);
    public static readonly ImperialDistance Yard = new ImperialDistance(36.0);
    public static readonly ImperialDistance Mile = new ImperialDistance(63360.0);

    private double _inches;

    public ImperialDistance(double inches) {
        _inches = inches;
    }

    public double ToInches() {
        return _inches;
    }

    public double ToFeet() {
        return _inches / Foot._inches;
    }

    public double ToYards() {
        return _inches / Yard._inches;
    }

    public double ToMiles() {
        return _inches / Mile._inches;
    }

    public MetricDistance ToMetricDistance() {
        return new MetricDistance(_inches * 0.0254);
    }

    public override int GetHashCode() {
        return _inches.GetHashCode();
    }

    public override bool Equals(object obj) {
        var o = obj as ImperialDistance;
        if (o == null) return false;
        return _inches.Equals(o._inches);
    }

    public static bool operator ==(ImperialDistance a, ImperialDistance b) {
        // If both are null, or both are same instance, return true
        if (ReferenceEquals(a, b)) return true;

        // if either one or the other are null, return false
        if (ReferenceEquals(a, null) || ReferenceEquals(b, null)) return false;

        // compare
        return a._inches == b._inches;
    }

    public static bool operator !=(ImperialDistance a, ImperialDistance b) {
        return !(a == b);
    }

    public static ImperialDistance operator +(ImperialDistance a, ImperialDistance b) {
        if (a == null) throw new ArgumentNullException();
        if (b == null) throw new ArgumentNullException();
        return new ImperialDistance(a._inches + b._inches);
    }

    public static ImperialDistance operator -(ImperialDistance a, ImperialDistance b) {
        if (a == null) throw new ArgumentNullException();
        if (b == null) throw new ArgumentNullException();
        return new ImperialDistance(a._inches - b._inches);
    }

    public static ImperialDistance operator *(ImperialDistance a, ImperialDistance b) {
        if (a == null) throw new ArgumentNullException();
        if (b == null) throw new ArgumentNullException();
        return new ImperialDistance(a._inches * b._inches);
    }

    public static ImperialDistance operator /(ImperialDistance a, ImperialDistance b) {
        if (a == null) throw new ArgumentNullException();
        if (b == null) throw new ArgumentNullException();
        return new ImperialDistance(a._inches / b._inches);
    }
}

Et voici une classe métrique dont l'unité de base est le mètre :

public class MetricDistance {

    public static readonly MetricDistance Milimeter = new MetricDistance(0.001);
    public static readonly MetricDistance Centimeter = new MetricDistance(0.01);
    public static readonly MetricDistance Decimeter = new MetricDistance(0.1);
    public static readonly MetricDistance Meter = new MetricDistance(1.0);
    public static readonly MetricDistance Decameter = new MetricDistance(10.0);
    public static readonly MetricDistance Hectometer = new MetricDistance(100.0);
    public static readonly MetricDistance Kilometer = new MetricDistance(1000.0);

    private double _meters;

    public MetricDistance(double meters) {
        _meters = meters;
    }

    public double ToMilimeters() {
        return _meters / Milimeter._meters;
    }

    public double ToCentimeters() {
        return _meters / Centimeter._meters;
    }

    public double ToDecimeters() {
        return _meters / Decimeter._meters;
    }

    public double ToMeters() {
        return _meters;
    }

    public double ToDecameters() {
        return _meters / Decameter._meters;
    }

    public double ToHectometers() {
        return _meters / Hectometer._meters;
    }

    public double ToKilometers() {
        return _meters / Kilometer._meters;
    }

    public ImperialDistance ToImperialDistance() {
        return new ImperialDistance(_meters * 39.3701);
    }

    public override int GetHashCode() {
        return _meters.GetHashCode();
    }

    public override bool Equals(object obj) {
        var o = obj as MetricDistance;
        if (o == null) return false;
        return _meters.Equals(o._meters);
    }

    public static bool operator ==(MetricDistance a, MetricDistance b) {
        // If both are null, or both are same instance, return true
        if (ReferenceEquals(a, b)) return true;

        // if either one or the other are null, return false
        if (ReferenceEquals(a, null) || ReferenceEquals(b, null)) return false;

        return a._meters == b._meters;
    }

    public static bool operator !=(MetricDistance a, MetricDistance b) {
        return !(a == b);
    }

    public static MetricDistance operator +(MetricDistance a, MetricDistance b) {
        if (a == null) throw new ArgumentNullException("a");
        if (b == null) throw new ArgumentNullException("b");
        return new MetricDistance(a._meters + b._meters);
    }

    public static MetricDistance operator -(MetricDistance a, MetricDistance b) {
        if (a == null) throw new ArgumentNullException("a");
        if (b == null) throw new ArgumentNullException("b");
        return new MetricDistance(a._meters - b._meters);
    }

    public static MetricDistance operator *(MetricDistance a, MetricDistance b) {
        if (a == null) throw new ArgumentNullException("a");
        if (b == null) throw new ArgumentNullException("b");
        return new MetricDistance(a._meters * b._meters);
    }

    public static MetricDistance operator /(MetricDistance a, MetricDistance b) {
        if (a == null) throw new ArgumentNullException("a");
        if (b == null) throw new ArgumentNullException("b");
        return new MetricDistance(a._meters / b._meters);
    }
}

Et voici une méthode de test qui illustre l'utilisation.

[TestMethod]
public void _5in_Equals_12_7cm() {
    var inches = new ImperialDistance(5);
    var cms = new MetricDistance(MetricDistance.Centimeter.ToMeters() * 12.7);
    var calcCentimeters = Math.Round(inches.ToMetricDistance().ToCentimeters(), 2, MidpointRounding.AwayFromZero);
    var calcInches = Math.Round(cms.ToImperialDistance().ToInches(), 2, MidpointRounding.AwayFromZero);

    Assert.AreEqual(cms.ToCentimeters(), 12.7);
    Assert.AreEqual(calcCentimeters, 12.7);
    Assert.AreEqual(inches.ToInches(), 5);
    Assert.AreEqual(calcInches, 5);
}

Vous pouvez également ajouter des méthodes d'extension

public static MetricDistance Centimeters(this Int32 that) {
    return new MetricDistance(MetricDistance.Centimeter.ToMeters() * that);
}

[TestMethod]
public void _100cm_plus_300cm_equals_400cm() {
    Assert.AreEqual(100.Centimeters() + 300.Centimeters(), 400.Centimeters());
}

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X