我在java中有一个泛型类定义为:
public static class KeyCountMap<T>
{
private Map<T, MutableInt> map = new LinkedHashMap<T, MutableInt>();
// ... rest of the properties...
public KeyCountMap()
{ }
@SuppressWarnings({ "unchecked", "rawtypes" })
public KeyCountMap(Class<? extends Map> mapType) throws InstantiationException, IllegalAccessException
{
map = mapType.newInstance();
}
//... rest of the methods...
}
我在 .NET 中定义了相同的类:
public static class KeyCountMap<T>
{
private Dictionary<T, MutableInt> map = new Dictionary<T, MutableInt>();
// ... rest of properties...
public KeyCountMap()
{ }
public void KeyCountMap<T>(T obj) where T : Dictionary<T, MutableInt>
{
obj = new T(); // Unable to define new instance of T
map = obj; // Unable to convert T to base class
}
}
然后定义一个方法对 KeyCountMap<T>
类型的 map 进行排序按值降序排列。该方法定义为:
public static KeyCountMap<T> SortMapByDescendValue<T>(KeyCountMap<T> _map)
{
List<KeyValuePair<T, MutableInt>> _list = new List<KeyValuePair<T, MutableInt>>(_map.EntrySet());
// whereas _map.EntrySet() return of type HashSet<KeyValuePair<T, MutableInt>>
_list = _list.OrderByDescending(_x => _x.Value).ToList();
KeyCountMap<T> _result = new KeyCountMap<T>();
foreach (KeyValuePair<T, MutableInt> _entry in _list)
{
_result.Put(_entry.Key, _entry.Value);
}
return _result;
}
如何更正 .NET 中定义的类?
最佳答案
我假设您知道 Java 在编译后会删除所有泛型类型信息(有变量的元数据,但实际对象没有泛型类型信息)。此外,您的代码不是类型安全的:
@SuppressWarnings({ "unchecked", "rawtypes" })
您使用它是因为您正在创建 Map
的非参数化实例.
在 .NET 中,您无法像这样绕过类型系统,因为通用类型信息在运行时保存并使用。
让我们看看您的 C# 代码:
public static class KeyCountMap<T>
C# 中的静态类是无法实例化的类,它仅用于其静态成员。我想你不想要这个。也许KeyCountMap
是 Java 中的静态嵌套类,而不是内部类。
在 C# 中,没有内部类。嵌套类不与包含类的实例共享数据,就好像包含类的名称是嵌套类的命名空间的一部分。因此,您不需要,实际上也不想要 static
此处为关键字。
{
private Dictionary<T, MutableInt> map = new Dictionary<T, MutableInt>();
在.NET中,Dictionary
是一个类。为了保持意图,您应该使用 IDictionary
,对应的接口(interface),类型为map
字段。
// ... rest of properties...
public KeyCountMap()
{ }
public void KeyCountMap<T>(T obj) where T : Dictionary<T, MutableInt>
为什么 void
返回类型,这不是构造函数吗?
在 C# 中,构造函数不能是泛型的。您可能想要 Type
.
您的 C# 代码没有意义,因此您可以执行以下操作:
public KeyCountMap(Type dictionaryType)
{
if (!typeof(IDictionary<T, MutableInt>).IsAssignableFrom(dictionaryType))
{
throw new ArgumentException("Type must be a IDictionary<T, MutableInt>", nameof(dictionaryType));
}
map = (IDictionary<T, MutableInt>)Activator.CreateInstance(dictionaryType);
}
}
我们在创建实例之前检查类型。如果不这样做,我们将创建一个实例,转换将失败,并且分配甚至不会发生,因此新实例将只是垃圾。
实际实例可能是一个代理;如果是这样,您可能不想在创建实例之前检查类型。
<小时/>对于有效的某些定义,您不能只是将 Java 复制粘贴为 C#(反之亦然),并期望只进行一些更改即可使其正常工作。它编译。这些语言并不那么相似,很可能有太多微妙的事情是错误的。
这种方法一开始可能会很有趣,但你会经常犯错,很快就会变得不再有任何乐趣。在开始逐行翻译代码之前,您应该学习基础知识并了解目标语言的完成方式。很多时候,您可能会发现在一种环境中必须做的事情在另一种环境中已经存在,反之亦然,或者某些事情在另一种环境中可能需要或多或少的步骤才能完成,等等。
在这种特殊情况下,Java 生成了 Class
是一个泛型类,而 .NET 保留 Type
非泛型类。在 .NET 中,只有接口(interface)和委托(delegate)可以声明泛型类型协变或逆变。无论如何,这都是相当严格的,如果 Type
是通用的,预期用途可以是协变的或逆变的。但请记住,在 Java 中,泛型 Class<T>
运行时与 Class
一样好,它只在编译时才有任何值(value),您可以告诉编译器您无论如何都知道,就像您所做的那样。
关于java - 获取 Java 的 Class<? 的行为扩展 .NET 中的 Map>,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37806107/