繁体   English   中英

在c#中打印变量的名称

[英]print name of the variable in c#

我有一个声明

int A = 10,B=6,C=5;

并且我想编写一个打印函数,以便将 int 变量传递给它,它会向我打印变量名称和值。

例如,如果我调用 print(A) 它必须返回“A: 10”,而 print (B) 那么它必须返回“B:6”

简而言之,我想知道如何访问变量的名称并将其打印为 C# 中的字符串。 我必须使用反射吗?

阅读答案后

大家好,感谢提供的建议。 我会尝试一下,但是我想知道在 .NET 2.0 中是否有可能? 没有类似的

#define prt(x) std::cout << #x " = '" << x << "'" << std::endl;

C/C++ 中有哪些宏?

唯一明智的方法是使用Expression API; 但这进一步改变了代码......

static void Main() {
    int A = 10, B = 6, C = 5;
    Print(() => A);
}
static void Print<T>(Expression<Func<T>> expression) {
    Console.WriteLine("{0}={1}",
        ((MemberExpression)expression.Body).Member.Name,
        expression.Compile()());
}

注意:如果这是出于调试目的,请务必将[Conditional("DEBUG")]到方法中,因为以这种方式使用变量会以微妙的方式改变代码的性质。

您可以使用 lambda 表达式:

static void Main( string[] args ) {
    int A = 50, B = 30, C = 17;
    Print( () => A );
    Print( () => B );
    Print( () => C );
}

static void Print<T>( System.Linq.Expressions.Expression<Func<T>> input ) {
    System.Linq.Expressions.LambdaExpression lambda = (System.Linq.Expressions.LambdaExpression)input;
    System.Linq.Expressions.MemberExpression member = (System.Linq.Expressions.MemberExpression)lambda.Body;

    var result = input.Compile()();
    Console.WriteLine( "{0}: {1}", member.Member.Name, result );
}

如果没有呼叫站点的一些“帮助”,这是不可能的; 甚至反射也不知道局部变量的名称。

另一个解决方案(来自封闭的帖子):

受 Jon Skeet 关于 Null Reference 异常处理的帖子的启发,突然想起投影,有一种方法可以做到这一点。

这是完整的工作代码:

public static class ObjectExtensions {
    public static string GetVariableName<T>(this T obj) {
        System.Reflection.PropertyInfo[] objGetTypeGetProperties = obj.GetType().GetProperties();

        if(objGetTypeGetProperties.Length == 1)
            return objGetTypeGetProperties[0].Name;
        else
            throw new ArgumentException("object must contain one property");
    }
}

class Program {
    static void Main(string[] args) {
        string strName = "sdsd";
        Console.WriteLine(new {strName}.GetVariableName());

        int intName = 2343;
        Console.WriteLine(new { intName }.GetVariableName());
    }
}

这不可能与反射有关(参见 Brian 和 Joel)。 一般来说,这是不可能的,因为您不能保证将命名值传递给您的打印函数。 例如,我可以轻松地执行以下操作

print(42);
print(A + 42);

这些表达式实际上都没有名称。 您希望在这里打印什么?

在未来,我们只使用nameof运算符。

如果您需要支持比int更多的类型,请使用Expression API 但避免使用泛型并优雅地处理不同的表达式:

private static string ToDebugOutput(params Expression<Func<object>>[] variables)
{
    var sb = new StringBuilder();
    foreach (var input in variables)
    {
        string name;
        if (input.Body is UnaryExpression unary && unary.Operand is MemberExpression operand)
        {
            name = operand.Member.Name;
        }
        else if (input.Body is MemberExpression member)
        {
            name = member.Member.Name;
        }
        else
        {
            throw new NotSupportedException($"typeof lambda: {input.Body.GetType()}");
        }

        var result = input.Compile()();
        sb.Append($"{name}={result}, ");
    }

    return sb.ToString();
}

用法:

string s = "123";
double d = 1.23;
int i2 = 123;
var out2 = ToDebugOutput(() => s, () => d, () => i2);
// out2 = "s=123, d=1.23, i2=123, "

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM