RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 686430
Accepted
tretetex
tretetex
Asked:2020-07-03 05:09:43 +0000 UTC2020-07-03 05:09:43 +0000 UTC 2020-07-03 05:09:43 +0000 UTC

制作通用方法

  • 772

有一个通用类:

public struct Range<T>
{
    public T From { get; set; }
    public T To { get; set; }

    public bool HasValue(T value)
    {
        // Нужно написать проверку входа в диапазон
    }
}

T- 只能是数字 ( int, double) 或日期 ( DateTime),也可以是Nullable.

有必要实现一个方法HasValue来检查它是否在从到value的范围内。FromTo

如果不使用强制转换和重载,这可能吗?

c#
  • 6 6 个回答
  • 10 Views

6 个回答

  • Voted
  1. Best Answer
    Qwertiy
    2020-07-03T21:19:36Z2020-07-03T21:19:36Z

    http://ideone.com/WBbU8d

    using System;
    using System.Collections.Generic;
    
    public struct Range<T>
    {
        public T From { get; set; }
        public T To { get; set; }
    
        public bool HasValue(T value)
        {
            return value != null
                && (From == null || Comparer<T>.Default.Compare(From, value) <= 0)
                && (To == null || Comparer<T>.Default.Compare(To, value) >= 0);
        }
    }
    

    并检查:

    public class Test
    {
        public static void Main()
        {
            var a = new Range <int> { From = 10, To = 16 };
            Console.WriteLine("var a = new Range <int> { From = 10, To = 16 };");
            Console.WriteLine(a.HasValue(0));
            Console.WriteLine(a.HasValue(15));
            Console.WriteLine(a.HasValue(17));
            Console.WriteLine();
    
            var b = new Range <int?> { From = 10, To = 16 };
            Console.WriteLine("var b = new Range <int?> { From = 10, To = 16 };");
            Console.WriteLine(b.HasValue(null));
            Console.WriteLine(b.HasValue(0));
            Console.WriteLine(b.HasValue(15));
            Console.WriteLine(b.HasValue(17));
            Console.WriteLine();
    
            var c = new Range <int?> { From = null, To = 16 };
            Console.WriteLine("var c = new Range <int?> { From = null, To = 16 };");
            Console.WriteLine(c.HasValue(null));
            Console.WriteLine(c.HasValue(0));
            Console.WriteLine(c.HasValue(15));
            Console.WriteLine(c.HasValue(17));
            Console.WriteLine();
    
            var d = new Range <int?> { From = 10, To = null };
            Console.WriteLine("var d = new Range <int?> { From = 10, To = null };");
            Console.WriteLine(d.HasValue(null));
            Console.WriteLine(d.HasValue(0));
            Console.WriteLine(d.HasValue(15));
            Console.WriteLine(d.HasValue(17));
            Console.WriteLine();
    
            var e = new Range <int?> { From = null, To = null };
            Console.WriteLine("var e = new Range <int?> { From = null, To = null };");
            Console.WriteLine(e.HasValue(null));
            Console.WriteLine(e.HasValue(0));
            Console.WriteLine(e.HasValue(15));
            Console.WriteLine(e.HasValue(17));
            Console.WriteLine();
        }
    }
    
    • 12
  2. Alex Krass
    2020-07-03T05:27:50Z2020-07-03T05:27:50Z

    接口负责比较IComparable,你可以限制它们的类型。是的,那么您将无法将 Nullable 类型作为 T 传递。只能通过重载将它们与 From 和 To 进行比较HasValue(T? value)。我没有想到最好的解决方案。

    public struct Range<T> where T : IComparable
    {
        public T From
        {
            get;
            set;
        }
        public T To
        {
            get;
            set;
        }
    
        public bool HasValue(T value)
        {
            return value.CompareTo(From) >= 0 && value.CompareTo(To) <= 0;
        }
    
        public bool HasValue(T? value)
        {
            return value.HasValue && value.Value.CompareTo(From) >= 0 && value.Value.CompareTo(To) <= 0;
        }
    }
    

    UPD 带有 Nullable 检查的类(未测试,我不保证性能)

    public struct Range<T> 
    {
        public T From
        {
            get;
            set;
        }
        public T To
        {
            get;
            set;
        }
    
        public bool HasValue(T value) 
        {
            IComparable _from = null;
            IComparable _to = null;
            IComparable _value = null;
    
            if (From != null) 
            {
                _from = GetValue(From) as IComparable;
            }
            if (To != null)
            {
                _to = GetValue(To) as IComparable;
            }
            if (value != null)
            {
                _value = GetValue(value) as IComparable;
            }
    
            if (_value == null) throw new ArgumentNullException();
    
            bool checkFrom = false;
            if (_from == null || _from != null && _value.CompareTo(_from) >= 0) checkFrom = true;
    
            bool checkTo = false;
            if (_to == null || _to != null && _value.CompareTo(_to) <= 0) checkTo = true;
    
            return checkFrom && checkTo;
        }
    
        private T GetValue<T>(T t) { return t; }
        private T GetValue<T>(T? t) where T : struct { return t.Value; }
    
    }
    
    class Program
    {
        static void Main(string[] args)
        {
    
            Console.WriteLine(new Range<int>() { From = -20, To = 20 }.HasValue(10)); //true
            Console.WriteLine(new Range<int>() { From = -20, To = 20 }.HasValue(-100)); //false
            Console.WriteLine(new Range<int>() { From = -20, To = 20 }.HasValue(100)); //false
    
            Console.WriteLine(new Range<int?>() { From = -20, To = 20 }.HasValue(10)); //true
            Console.WriteLine(new Range<int?>() { From = -20, To = 20 }.HasValue(-100)); //false
            Console.WriteLine(new Range<int?>() { From = -20, To = 20 }.HasValue(100)); //false
            Console.WriteLine(new Range<int?>() { From = null, To = 20 }.HasValue(10)); //true
            Console.WriteLine(new Range<int?>() { From = -20, To = null }.HasValue(10)); //true
            Console.WriteLine(new Range<int?>() { From = -20, To = null }.HasValue(-100)); //false
    
            Console.WriteLine(new Range<DateTime>() { From = DateTime.Now.AddDays(-5), To = DateTime.Now.AddDays(5) }.HasValue(DateTime.Now.AddDays(-2))); //true
            Console.WriteLine(new Range<DateTime>() { From = DateTime.Now.AddDays(-5), To = DateTime.Now.AddDays(5) }.HasValue(DateTime.Now.AddDays(-10))); //false
            Console.WriteLine(new Range<DateTime>() { From = DateTime.Now.AddDays(-5), To = DateTime.Now.AddDays(5) }.HasValue(DateTime.Now.AddDays(10))); //false
            Console.WriteLine(new Range<DateTime?>() { From = null, To = DateTime.Now.AddDays(5) }.HasValue(DateTime.Now.AddDays(-2))); //true
            Console.WriteLine(new Range<DateTime?>() { From = DateTime.Now.AddDays(-5), To = null }.HasValue(DateTime.Now.AddDays(-2))); //true
            Console.WriteLine(new Range<DateTime?>() { From = DateTime.Now.AddDays(-5), To = null }.HasValue(DateTime.Now.AddDays(-100))); //false
    
            Console.ReadLine();
        }
    }
    
    • 8
  3. Alexander Petrov
    2020-07-03T06:51:51Z2020-07-03T06:51:51Z

    对于此类类型,您通常会创建一堆重载。例如,您可以看到方法Console.WriteLine。您可以这样做:创建许多结构/类来处理特定类型。为了不全部手动编写,可以使用T4代码生成。

    在文件tt中我们写:

    <#@ template debug="false" hostspecific="false" language="C#" #>
    <#@ output extension=".cs" #>
    
    using System;
    
    namespace TestApp
    {
        <# 
        var types = new string[] { "SByte", "Byte", "Char", "Int16", "UInt16", "Int32", "UInt32",
            "Int64", "UInt64", "Single", "Double", "Decimal", "DateTime", "DateTimeOffset" };
    
        foreach(var T in types) {
        #>
        public struct Range<#=T#>
        {
            public <#=T#> From { get; set; }
            public <#=T#> To { get; set; }
    
            public bool HasValue(<#=T#> value)
            {
                return value >= From && value <= To;
            }
        }
        public struct RangeNullable<#=T#>
        {
            public <#=T#>? From { get; set; }
            public <#=T#>? To { get; set; }
    
            public bool HasValue(<#=T#>? value)
            {
                return value >= From && value <= To;
            }
        }
        <#
            }
        #>
    }
    

    根据需要添加/减去类型。
    编译后cs会生成一个包含很多结构的文件。

    接下来,您将需要创建和使用明确指定类型的结构实例:

    var range = new RangeInt32();
    var range2 = new RangeNullableDateTime();
    

    在原来它会是:

    var range = new Range<int>();
    var range2 = new Range<DateTime?>();
    
    • 6
  4. VladD
    2020-07-03T20:28:20Z2020-07-03T20:28:20Z

    另一个实现选项。

    条件:要么该类型T不是Nullable,并且实现IComparable<Т>,要么它是一个类型T1?,并且它T1实现IComparable<Т1>。不幸的是在编译时没有检查。

    class Range<T>
    {
        // компаратор для не-nullable-случая
        static int CompareNonNullable<V>(V l, V r) where V : IComparable<V> => l.CompareTo(r);
    
        // компаратор для nullable-случая
        static int CompareNullable<V>(V? l, V? r) where V : struct, IComparable<V>
        {
            if (l == null)
                return r == null ? 0 : 1;
            else if (r == null)
                return -1;
            else return l.Value.CompareTo(r.Value);
        }
    
        static Range()
        {
            Type parameter = typeof(T);
            Type nullableBase = Nullable.GetUnderlyingType(typeof(T));
    
            Type underlying = nullableBase ?? parameter; // T или T1
    
            // проверка на IComparable
            var icomparable = typeof(IComparable<>).MakeGenericType(underlying);
            var hasInterface = underlying.GetInterfaces().Contains(icomparable);
            if (!hasInterface)
                throw new ArgumentException(
                    "Type should implement IComparable generic interface");
    
            var methodName = (nullableBase == null) ? nameof(CompareNonNullable) :
                                                      nameof(CompareNullable);
            var rawMI = typeof(Range<T>).GetMethod(
                            methodName,
                            BindingFlags.Static | BindingFlags.NonPublic);
            var genericMI = rawMI.MakeGenericMethod(underlying);
    
            comparer = (Func<T, T, int>)Delegate.CreateDelegate(
                                            typeof(Func<T, T, int>), genericMI);
        }
    
        static Func<T, T, int> comparer;
    
        public Range(T from, T to)
        {
            From = from;
            To = to;
        }
    
        public bool Contains(T t) => comparer(From, t) <= 0 && comparer(t, To) <= 0;
    
        public T From { get; }
        public T To { get; }
    }
    

    像这样使用:

    var intRange = new Range<int>(0, 5);
    Console.WriteLine(intRange.Contains(5));
    
    var dtRange = new Range<DateTime?>(DateTime.Now.AddDays(-1), null);
    Console.WriteLine(dtRange.Contains(DateTime.Now));
    
    • 4
  5. Andrey Rygin
    2020-07-11T01:25:23Z2020-07-11T01:25:23Z

    可以通过扩展来完成:

    public static class RangeExtensions
    {
        public static bool HasValue<T>(this Range<T> range, T value, IComparer<T> comparer = null)
            where T : struct
        {
            comparer = comparer ?? Comparer<T>.Default;
            return comparer.Compare(value, range.From) >= 0 && comparer.Compare(value, range.To) <= 0;
        }
    
        public static bool HasValue<T>(this Range<T?> range, T? value, IComparer<T> comparer = null)
            where T : struct
        {
            comparer = comparer ?? Comparer<T>.Default;
            if (value == null || range.From == null || range.To == null)
            {
                return false;
            }
    
            return comparer.Compare(value.Value, range.From.Value) >= 0 && comparer.Compare(value.Value, range.To.Value) <= 0;
        }
    }
    

    如果 Range 是结构,则此方法存在潜在的性能问题 - 该结构将按值传递。如果 Range 将是一个类或者操作不频繁,那么试试这个。

    • 1
  6. Chloroform
    2020-07-03T05:17:33Z2020-07-03T05:17:33Z

    实现是这样的:

    public bool HasValue(T value)
    {
        if (value == null)
            return false;
        return (value >= From) && (To >= value);
    }
    

    附言。第一次,我写错了。原来什么都不用带,因为。数字,DataTime我们可以比较。第一步是检查null,然后以通常的方式进行比较。

    • 0

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    Python 3.6 - 安装 MySQL (Windows)

    • 1 个回答
  • Marko Smith

    C++ 编写程序“计算单个岛屿”。填充一个二维数组 12x12 0 和 1

    • 2 个回答
  • Marko Smith

    返回指针的函数

    • 1 个回答
  • Marko Smith

    我使用 django 管理面板添加图像,但它没有显示

    • 1 个回答
  • Marko Smith

    这些条目是什么意思,它们的完整等效项是什么样的

    • 2 个回答
  • Marko Smith

    浏览器仍然缓存文件数据

    • 1 个回答
  • Marko Smith

    在 Excel VBA 中激活工作表的问题

    • 3 个回答
  • Marko Smith

    为什么内置类型中包含复数而小数不包含?

    • 2 个回答
  • Marko Smith

    获得唯一途径

    • 3 个回答
  • Marko Smith

    告诉我一个像幻灯片一样创建滚动的库

    • 1 个回答
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Алексей Шиманский 如何以及通过什么方式来查找 Javascript 代码中的错误? 2020-08-03 00:21:37 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    user207618 Codegolf——组合选择算法的实现 2020-10-23 18:46:29 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5