C# XDocument / linq to object

How can i import xml elements to object? My code below doesn't work, it fails at the SetValue and i can't figure out why.

But even then, i suspect that linq has a much cleaner way of doing this but i can't find any examples.

class Printers {
    public List<Printer> list = new List<Printer>();

    public Printers()
    {
        var xDoc = XDocument.Load(Properties.Settings.Default.XmlSetupPath).Root;
        var xPrinters = xDoc.Element("printers").Elements();
        foreach (var xPrinter in xPrinters)
        {
            var printer = new Printer();
            foreach (var xEl in xPrinter.Elements())
            {
                printer.GetType().GetProperty(xEl.Name.ToString()).SetValue(printer, xEl.Value);
            }
        }

    }

}

class Printer
{
    public string name;
    public string ip;
    public string model;
    public string infx86;
    public string infx64;
    public string location;
    public string comment;
}

my XML:

<printers>
 <printer>
  <name>my Printer</name>
  <ip>192.168.100.100</ip>
  <model>Brother</model>
  <driver>ab</driver>
  <infx86>ab\cd.INF</infx86>
  <comment>Copycenter</comment>
 </printer>
 <printer>
  <name>my Printer</name>
  <foobar>oh no!</foobar>
 </printer>
</printers>

I want to

Jon Skeet
people
quotationmark

You're asking for properties - but your type only has fields. Either make them properties, like this:

 public string name { get; set; }

... or use Type.GetField instead.

In terms of making it prettier, I'd personally add a static FromXElement method to your Printer class, at which point you could have:

list = xDoc.Element("printers")
           .Elements()
           .Select(Printer.FromXElement)
           .ToList();

Or you could write a generic method to create a new instance and populate it via reflection, e.g.

public static T FromXElement<T>(XElement element) where T : class, new()
{
    T value = new T();
    foreach (var subElement in element.Elements())
    {
        var field = typeof(T).GetField(subElement.Name.LocalName);
        field.SetValue(value, (string) subElement);
    }
    return value;
}

Then:

list = xDoc.Element("printers")
           .Elements()
           .Select(XmlReflection.FromXElement<Printer>)
           .ToList();

people

See more on this question at Stackoverflow