- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
协变性(大致)是在使用“简单”类型的复杂类型中镜像继承的能力。
例如。我们总是可以处理 Cat
的实例作为 Animal
的实例. ComplexType<Cat>
可能被视为 ComplexType<Animal>
,如果 ComplexType 是协变的。
我想知道:协方差的“类型”是什么,它们与 C# 有什么关系(是否支持它们?)
代码示例会有所帮助。
例如,一种类型是返回类型协方差,Java 支持,但 C# 不支持。
我希望有函数式编程能力的人也能插话!
最佳答案
这是我能想到的:
更新
在阅读了 Eric Lippert 提出(和撰写)的建设性评论和大量文章后,我改进了答案:
返回类型协方差:
可用于 Java (>= 5) [1]
和 C++ [2]
,在 C# 中不受支持(Eric Lippert 解释 why not 和 what you can do about it ):
class B {
B Clone();
}
class D: B {
D Clone();
}
接口(interface)协方差 [3]
- C# 支持
BCL 定义了通用的 IEnumerable
接口(interface)是协变的:
IEnumerable<out T> {...}
因此下面的例子是有效的:
class Animal {}
class Cat : Animal {}
IEnumerable<Cat> cats = ...
IEnumerable<Animal> animals = cats;
请注意 IEnumerable
根据定义,它是“只读的”——您不能向其中添加元素。
将其与 IList<T>
的定义进行对比可以修改,例如使用 .Add()
:
public interface IEnumerable<out T> : ... //covariant - notice the 'out' keyword
public interface IList<T> : ... //invariant
通过方法组委托(delegate)协变 [4]
- C# 支持
class Animal {}
class Cat : Animal {}
class Prog {
public delegate Animal AnimalHandler();
public static Animal GetAnimal(){...}
public static Cat GetCat(){...}
AnimalHandler animalHandler = GetAnimal;
AnimalHandler catHandler = GetCat; //covariance
}
“纯”委托(delegate)协方差 [5 - pre-variance-release article]
- C# 支持
委托(delegate)的 BCL 定义不接受任何参数并返回一些东西是协变的:
public delegate TResult Func<out TResult>()
这允许:
Func<Cat> getCat = () => new Cat();
Func<Animal> getAnimal = getCat;
数组协变 - 在 C# 中得到支持,以一种中断的方式 [6]
[7]
string[] strArray = new[] {"aa", "bb"};
object[] objArray = strArray; //covariance: so far, so good
//objArray really is an "alias" for strArray (or a pointer, if you wish)
//i can haz cat?
object cat == new Cat(); //a real cat would object to being... objectified.
//now assign it
objArray[1] = cat //crash, boom, bang
//throws ArrayTypeMismatchException
最后 - 令人惊讶且有点令人费解的事情
委托(delegate)参数协方差(是的,这就是co-方差)- 用于高阶函数。 [8]
采用一个参数且不返回任何内容的委托(delegate)的 BCL 定义是逆变:
public delegate void Action<in T>(T obj)
请耐心等待。让我们定义一个马戏团驯兽师 - 他可以被告知如何训练动物(通过给他一个与该动物一起工作的 Action
)。
delegate void Trainer<out T>(Action<T> trainingAction);
我们有培训师定义,让我们找一个培训师并让他工作。
Trainer<Cat> catTrainer = (catAction) => catAction(new Cat());
Trainer<Animal> animalTrainer = catTrainer;
// covariant: Animal > Cat => Trainer<Animal> > Trainer<Cat>
//define a default training method
Action<Animal> trainAnimal = (animal) =>
{
Console.WriteLine("Training " + animal.GetType().Name + " to ignore you... done!");
};
//work it!
animalTrainer(trainAnimal);
输出证明这是可行的:
Training Cat to ignore you... done!
为了理解这一点,开个玩笑吧。
A linguistics professor was lecturing to his class one day.
"In English," he said, "a double negative forms a positive.
However," he pointed out, "there is no language wherein a double positive can form a negative."A voice from the back of the room piped up, "Yeah, right."
与协方差有什么关系?!
让我尝试一个餐巾纸背面的演示。
安Action<T>
是逆变的,即它“翻转”了类型的关系:
A < B => Action<A> > Action<B> (1)
更改 A
和 B
上面有 Action<A>
和 Action<B>
并得到:
Action<A> < Action<B> => Action<Action<A>> > Action<Action<B>>
or (flip both relationships)
Action<A> > Action<B> => Action<Action<A>> < Action<Action<B>> (2)
将 (1) 和 (2) 放在一起,我们有:
,-------------(1)--------------.
A < B => Action<A> > Action<B> => Action<Action<A>> < Action<Action<B>> (4)
`-------------------------------(2)----------------------------'
但是我们的Trainer<T>
delegate 实际上是一个 Action<Action<T>>
:
Trainer<T> == Action<Action<T>> (3)
所以我们可以将 (4) 重写为:
A < B => ... => Trainer<A> < Trainer<B>
- 根据定义,这意味着 Trainer 是协变的。
简而言之,申请 Action
两次我们得到了反反方差,即类型之间的关系翻转了两次(见 (4)),所以我们回到了协方差。
关于c# - C# 中的协变有哪些类型? (或者,协方差 : by example),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17231577/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!