2

在 C# 中,我正在定义特定类的静态字段。在类中,我希望能够显示静态字段的名称,就像这样:

public class Unit {
  public string NameOfField { get { return ...; } }
}

public static Unit Hectare = new Unit();

如果我现在访问:

Hectare.NameOfField

我希望它返回:

Hectare

我知道有一个静态函数 System.Reflection.MethodBase.GetCurrentMethod(),但据我所知,没有办法获取包含此当前方法的实例的名称?

还有 System.RuntimeFieldHandle 结构,但我无法识别任何 GetCurrentFieldHandle() 方法。

我不确定我是否遗漏了一些明显的东西?

非常感谢您对此的任何帮助。

4

4 回答 4

1

你不应该在你的开发中指望变量名,因为它们不会在运行时退出。

最好直接用名字初始化 Unit:

public class Unit {

  public Unit(string name)
  {
     NameOfField = name;
  }
  public string NameOfField { get; private set;} }
}

public static Unit Hectare = new Unit("Hectare");
于 2011-04-05T09:20:18.097 回答
0

解决这个问题的唯一方法是将这些信息存储在类中:

public static Unit Hectare = new Unit("Hectare");

编译代码时,所有变量名都会丢失并被内部引用替换。没有办法再次获得这个名字。

于 2011-04-05T09:22:10.383 回答
0

您可以使用反射来获取类字段和属性。如下所示:

假设您有一个具有一个属性的类:

class Test
{
    public static string MySupperField
    {
        get
        {
            return "Some symbols here";
        }
    }
}

......

您可以通过以下方式读取属性名称:

public string[] GetClassStaticNames(Type T)
{
    string[] names;

    System.Reflection.PropertyInfo[] props = T.GetProperties(); // This will return only properties not fields! For fields obtaining use T.GetFields();

    names = new string[props.Count()];
    for (int i = 0; i < props.Count(); i++)
    {
        names[i] = props[i].Name;
    }

    return names;
}

希望这会有所帮助。

[编辑]
回到您的问题 - 不,您无法获取当前变量的名称。
由于类的性质,您所问的问题无法完成,它们是内存中的对象,并且对一个对象的引用可以保存在许多变量中,并且当您请求实例字段或属性的值时,它将实际执行与对象中的操作没有变量的内存持有对该对象的引用。因此,获取持有对当前实例的引用的变量的名称是没有意义的

于 2011-04-05T09:37:07.173 回答
0

感谢所有花时间回答和讨论我的问题的人。

只是为了让您知道,我已经实施了一个足以满足我需求的解决方案。该解决方案不是通用的,并且存在一些缺陷,但我想无论如何我都会分享它,以防它对其他人有所帮助。

原则上,定义字段时使用的类如下所示:

public class Unit : IUnit {
  public NameOfField { get; set; }
  ...
}

如您所见,该类实现了 IUnit 接口,并且我在 NameOfField 属性中提供了一个公共设置器。

静态字段通常在某些包含类中定义如下:

public static Unit Hectare = new Unit();

我的解决方案是在实现中使用该字段之前通过反射设置 NameOfField 属性。我通过一个静态构造函数来做到这一点(当然需要在访问 Unit 字段之前调用它。我使用 Linq 遍历相关字段的执行程序集,并且当我检测到这些字段(类型实现 IUnit 接口的字段),我使用 Any 扩展方法为它们中的每一个设置 NameOfField 属性:

Assembly.GetExecutingAssembly().GetTypes().
  SelectMany(type => type.GetFields(BindingFlags.Public | BindingFlags.Static)).
  Where(fieldInfo => fieldInfo.FieldType.GetInterfaces().Contains(typeof(IUnit))).
  Any(fieldInfo =>
  {
    ((IUnit)fieldInfo.GetValue(null)).NameOfField= fieldInfo.Name;
    return false;
  });

这种方法有一些缺点:

  • 在访问任何 Unit 字段之前,必须通过手动干预调用静态构造函数
  • NameOfField 设置器是公开的。就我而言,这没有问题,但在其他场景中应用时可能会出现这种情况。(我假设 setter 可以设为私有并通过进一步的反射调用,但我没有花时间进一步探索这条路径。)
  • ... ?

无论哪种方式,也许这个解决方案对我以外的其他人都有帮助。

于 2011-04-05T20:17:23.267 回答