- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
编辑: C# 3.0,net 3.5。
我是 C++ 程序员,所以我可能会错过 C# 中的一些简单解决方案。
简化示例:我有几个类继承自 MyBase 类(使用方法 Inc)。继承的类可能会覆盖 Inc。我有几个用于继承的类的重载“函数”(静态方法):
void Print(MyInherited1 i1) ....
void Print(MyInherited2 i2) ....
等等。 编辑:这些方法是 MyBase 和 MyInherited 的“外部”方法,它们不属于任何这些类。
我有修改其参数并调用 Print 函数的泛型函数:
void Transform<T>(T t)
{
t.Inc();
Print(t);
}
编辑:这是一个简化的示例,在实际代码中我无法仅使用多态性将 Transform 方法转换为非泛型方法。
现在,在 C++ 中它会像那样工作。但是在 C# 方法中 Inc 是未知的。所以我指定 T 是 MyBase 类型。
void Transform<T>(T t) where T : MyBase
{
t.Inc();
Print(t);
}
但我仍然对 Print 有疑问——基类没有这样的方法。
作为解决方法,我使用了丑陋的 (!) 解决方案 -- PrintProxy,我在其中放置了几个
if (t is Inherited1)
Print(t as Inherited1);
else if ...
如何混合这两个概念——重载和泛型?不错,C# 方式?
预先感谢您的帮助。
最佳答案
C# 4 中的一个选项是使用动态类型:
void Transform<T>(T t)
{
t.Inc();
dynamic d = t;
Print(d);
}
这将在执行时为您执行重载决议 - 这基本上是您可以做的最好的,除非您可以提供真正通用的 Print
方法,因为只有一个版本的 Transform
生成。
请注意,此时没有真正需要通用 - 您可以将其替换为:
void Transform(MyBase t)
{
...
}
我通常发现基于接口(interface)的约束比基于类的约束更有用,除非我也在做其他通用的事情(例如创建一个应该是正确的实际类型的 List<T>
)。
显然,这种动态方法有缺点:
object
作为一种回退以提供自定义错误处理,但您仍然必须等到执行时间)基本上这只是 .NET 泛型和 C++ 模板之间的区别 - 它们是非常不同的生物,即使它们解决了许多相似的问题。
而不是静态 Print
方法,你有什么理由不能写摘要 Print
MyBase
中的方法,并在每个类(class)覆盖它?无论如何,这感觉更像是一个面向对象的解决方案,老实说——尽管如果打印实际上在逻辑上与类本身有些距离,显然它没有意义。即使您不想要实际 Print
原始类型层次结构中的方法,您是否能够公开足够的功能来让您编写虚拟 Print
方法?毕竟,我认为所有这些方法都应该得出某种类似的结果。
编辑:几个可供选择的想法...
传入打印机
您可以传入委托(delegate)来进行打印。如果您从知道实际类型的非泛型上下文中调用它,则可以利用方法组转换来简化此过程。这是一个简短但完整的示例:
using System;
class Test
{
static void SampleMethod<T>(T item, Action<T> printer)
{
// You'd do all your normal stuff here
printer(item);
}
static void Print(string x)
{
Console.WriteLine("Here's a string: {0}", x);
}
static void Print(int x)
{
Console.WriteLine("Here's an integer: {0}", x);
}
static void Main()
{
SampleMethod(5, Print);
SampleMethod("hello", Print);
}
}
使用类型/委托(delegate)字典
另一种选择是使用 Dictionary<Type, Delegate>
包含打印方法。这可以是内联的(如果打印方法很简单)或类似这样的东西:
static readonly Dictionary<Type, Delegate> Printers =
new Dictionary<Type, Delegate>
{
{ typeof(MyClass1), (Action<MyClass1>) Print },
{ typeof(MyClass2), (Action<MyClass2>) Print },
{ typeof(MyClass3), (Action<MyClass3>) Print },
};
然后在你的方法中:
Delegate printer;
if (Printers.TryGetValue(typeof(T), out printer))
{
((Action<T>) printer)(t);
}
else
{
// Error handling
}
不过,这是另一个执行时间解决方案,如果您希望它处理进一步的派生(例如,如果它找不到相关的打印机,则遍历基类),您需要做更多的工作。
关于c# - 如何通过重载使用泛型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1997999/
假设我有一个类,我在其中重载了运算符 == : Class A { ... public: bool operator== (const A &rhs) const; ... };
我知道你不应该使用 std::find(some_map.begin(), some_map.end()) 或 std::lower_bound,因为它会采用线性时间而不是 some_map.lowe
我正在尝试在 Haskell 中定义 Vector3 数据类型,并允许在其上使用 (+) 运算符。我尝试了以下方法: data Vector3 = Vector3 Double Double Doub
我已经为我的类图将运算符重载为“-”。它的用途并不完全直观(糟糕的编码 - 我知道)但是如果我做 graph3 = graph2-graph1 那么图 3 是应该只接收图 2 和图 1 中的那些顶点。
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: Operator overloading 我想重载 以按字母顺序排列字符串,但我不确定该怎么做。 如何再次
下面的代码给我一个编译错误。谁能告诉我为什么? class mytype { public: int value; mytype(int a) { value = a;
这有什么问题吗? class Vec2 attr_accessor :x, :y # ... def += (v) @x += v.x @y += v.y retu
是否可以重载 [] 运算符两次?允许这样的事情:function[3][3](就像在二维数组中一样)。 如果可能的话,我想看看一些示例代码。 最佳答案 您可以重载 operator[] 以返回一个对象
我的团队目前正在与 Lua 合作,创建一个 android 游戏。我们遇到的一件事是表面上无法创建重载构造函数。 我习惯于使用默认值设置一个对象,然后在需要时使其过载。 前任: apples() {
我有一个网页,在某个时候显示一个导航栏,它只不过是一个 a 元素的列表 (ul)。所述 a 元素的大多数样式规则都是通用的。唯一应该改变的部分是要显示的图像,可以从列表中每个 li 元素的 id 标签
我对使用/重载“范围步长”运算符(.. ..)很感兴趣,但我终其一生都无法了解如何使用它。 在文档中它说 // Usage: start .. step .. finish 但是在 F# shell
Java 11(可能无关紧要): public static String toString(Object obj) { return ReflectionToStringBuilder.to
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
我无法理解以下代码(针对行号进行注释) class Base { void m1(Object o) { } void m2(String o) { } } publi
我有以下代码片段: #include using namespace std; struct Integer{ int x; Integer(const int val) : x(v
class myclass{ //definitions here }; myclass e; int myarray[10]; /* Do something... */ e = myarray;
为什么不能将下标运算符(operator [])作为 friend 函数重载? 最佳答案 正如Bjarne Stroustrup在D&E book中所说: However, even in the o
我有以下代码片段: #include using namespace std; struct Integer{ int x; Integer(const int val) : x(v
因此,我有一个问题是我最近尝试重载 namespace Eng { /** * A structure to represent pixels */ typedef
如何重载onResume()以正确的方式工作?我想从 activity 返回到 MainActivity ,我希望在其中具有与应用程序启动后相同的状态。我想使用 recreate() 但它循环了或者类
我是一名优秀的程序员,十分优秀!