java - 获取 Java 的 Class<? 的行为扩展 .NET 中的 Map>

标签 java c# generics inheritance instantiation

我在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/

相关文章:

c# - ASP.NET 定时器刷新整个页面

java - 接收通用类类型

java - 摆脱 List<GenericClass<T>>.toArray() 中的警告

java - 服务于多个路径的 Jersey 资源类

java - 用户登录后如何重定向到目标页面

c# - Javascript 舍入 v C# 舍入

java - 在列表中添加假类型时没有异常

java - 无法在android中使用retrofit2发出@Post请求

java - 用户如何将参数键入 JPanel 的构造函数?

c# - ImageMagick.NET 异常