Цепочка DebuggerDisplay для сложных типов - PullRequest
15 голосов
/ 30 декабря 2011

У меня есть несколько классов, определяющих атрибут DebuggerDisplay.Я хочу знать, есть ли способ определить один атрибут DebuggerDisplay на основе другого.Если у меня есть следующие классы:

[DebuggerDisplay ("Text = {Text}")]
class A
{
    public string Text {get;set;}
}

[DebuggerDisplay ("Property = {Property}")]
class B
{
    public A Property {get; set;}
}

Я хотел бы видеть на экземплярах класса B класс A, как он определен в атрибуте DebuggerDisplay класса A.Вместо этого я получаю метод класса A ToString () на отладчик при просмотре объектов класса B.

Ответы [ 3 ]

6 голосов
/ 30 декабря 2011

Не уверен, правильно ли я понял вашу проблему, но попробуйте:

[DebuggerDisplay("Property = {Property.Text}")]
public class B
{
    public A Property { get; set; }
}

Это отобразит свойство текста А.

Если вам нужен более сложный элемент управления, вы можете использовать DebuggerTypeProxyAttribute

1 голос
/ 02 марта 2016

С https://blogs.msdn.microsoft.com/jaredpar/2011/03/18/debuggerdisplay-attribute-best-practices/ (я добавил директивы условной компиляции)

#if DEBUG
    [DebuggerDisplay("{DebuggerDisplay}")]
    public sealed class Student {
    public string FirstName { get; set; }
    public string LastName { get; set; }
    private string DebuggerDisplay {
        get { return string.Format("Student: {0} {1}", FirstName, LastName);}
    }
}
#endif

Это похоже на ответ Микки Перлштейна (clear Свойство, которое форматирует строку отладчика) безнеобходимость переопределить ToString () (что может понадобиться для других целей.)

* source также имеет ряд других полезных советов для DebuggerDisplay, включая некоторые соображения производительности.

Редактировать


Так как в любом случае это отладочный код, нарушать ООП (доступ к частной собственности извне) не так уж и плохо ... но мы довольно сильно его нарушаемздесь трудно.

private string DebuggerString {
    get {
        StringBuilder sb = new StringBuilder();
        sb.Append("Whatever you want your Parent class' Debugger Text To Say");

        var properties = typeof(GroupQuote).GetProperties()
            //get the properties with the DebuggerDisplay attribute and our property
            .Where(x =  > x.PropertyType.IsDefined(typeof(DebuggerDisplayAttribute))
                     && x.PropertyType.GetProperties(BindingFlags.NonPublic | BindingFlags.Instance).Any(y =  > y.Name == "DebuggerString"));

        foreach(PropertyInfo property in properties) {
            object itemWithProperty = property.GetValue(this);
        //we have to check our property for null, otherwise trying to get its DebuggerString property will throw an exception
        if (itemWithProperty != null) {
                PropertyInfo privateDebuggerProperty = property.PropertyType.GetProperty("DebuggerString", BindingFlags.NonPublic | BindingFlags.Instance);
                sb.Append(privateDebuggerProperty.GetValue(itemWithProperty)as string);
            }
        }
        return sb.ToString();
    }
}

Example

В коде, который я написал и протестировал, у меня были некоторые свойства моего класса Parent, показывающие, что DebuggerDisplay был определенкогда это не было (возможно, вещь наследства?).Я добавил дополнительную проверку, чтобы мы искали DebuggerString только в тех свойствах, у которых она есть.

1 голос
/ 27 июня 2012

Я знаю, что это не «правильное кодирование», но, поскольку я не могу связать свои энты, я решил вернуться к старым способам.Просто переопределите метод ToString ().тогда цепочка - это кусок пирога.

    public partial class Tld
{
    public override string ToString()
    {
        return this.Name;
    }    
}

public partial class Domain
{
    public override string ToString()
    {
        return this.DomainName + "." +this.Tld.ToString();
    } 

    public  Domain (string domain, string tld):this( domain, new Tld(tld))
    {

    }
    public Domain(string domain, Tld tld):this()
    {
        this.DomainName = domain;
        this.Tld = tld;

    }
}


public partial class Url
{
    public override string ToString()
    {
        return this.Scheme + "://" + this.Subdomain + this.Domain.ToString() + ((string.IsNullOrWhiteSpace(this.Path)) ? "" :  this.Path);
    }
    public Url (string scheme, string subdomain, string domain, string tld, string path):this(new Tld(tld),domain, subdomain,scheme,path){}

    public Url(Tld tld, string domainName, string subdomain, string scheme, string path): this(new Domain(domainName, tld),subdomain,scheme,path){}

     public Url(Domain domain, string subdomain, string scheme, string path):this()
    {
        this.Domain = domain;
        this.Path = path;
        this.Scheme = scheme;
        this.Subdomain = subdomain;

    }

}


public void Domain_Create_GOOD()
    {
     Domain expected = new Domain("google","co.nz");

    }
...