Как сохранить коллекцию пользовательских объектов в файле user.config? - PullRequest
33 голосов
/ 09 апреля 2009

Я хотел бы сохранить коллекцию пользовательских объектов в файле user.config и хотел бы программно добавлять и удалять элементы из коллекции, а затем сохранять измененный список обратно в файл конфигурации.

Мои вещи имеют следующую простую форму:

class UserInfo
{
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public string Email { get; set; }        
}

В моем app.config я уже создал пользовательский раздел:

<configuration>
  <configSections>
    <section name="userInfo" type="UserInfoConfigurationHandler, MyProgram"/>

  </configSections>
  <userInfo>
    <User firstName="John" lastName="Doe" email="john@example.com" />
    <User firstName="Jane" lastName="Doe" email="jane@example.com" />
  </userInfo>

</configuration>

Я также могу читать в настройках, введя IConfigurationSectionHandler:

class UserInfoConfigurationHandler : IConfigurationSectionHandler
{
    public UserInfoConfigurationHandler() { }

    public object Create(object parent, object configContext, System.Xml.XmlNode section)
    {
        List<UserInfo> items = new List<UserInfo>();
        System.Xml.XmlNodeList processesNodes = section.SelectNodes("User");

        foreach (XmlNode processNode in processesNodes)
        {
            UserInfo item = new UserInfo();
            item.FirstName = processNode.Attributes["firstName"].InnerText;
            item.LastName = processNode.Attributes["lastName"].InnerText;
            item.Email = processNode.Attributes["email"].InnerText;
            items.Add(item);
        }
        return items;
    }
}

Я сделал все это после этой статьи . Однако, используя этот подход, я могу только прочитать настройки из app.config в коллекцию List<UserInfo>, но мне также нужно записать измененный список обратно.

Я безуспешно искал документацию, а теперь застрял. Чего мне не хватает?

Ответы [ 5 ]

55 голосов
/ 16 апреля 2009

Способ добавления настраиваемой конфигурации (если вам требуется больше, чем просто простые типы) - это использовать ConfigurationSection, в рамках которого для определенной вами схемы вам понадобится ConfigurationElementCollection (установленный как коллекция по умолчанию без имени), который содержит ConfigurationElement , следующим образом:

public class UserElement : ConfigurationElement
{
    [ConfigurationProperty( "firstName", IsRequired = true )]
    public string FirstName
    {
        get { return (string) base[ "firstName" ]; }
        set { base[ "firstName" ] = value;}
    }

    [ConfigurationProperty( "lastName", IsRequired = true )]
    public string LastName
    {
        get { return (string) base[ "lastName" ]; }
        set { base[ "lastName" ] = value; }
    }

    [ConfigurationProperty( "email", IsRequired = true )]
    public string Email
    {
        get { return (string) base[ "email" ]; }
        set { base[ "email" ] = value; }
    }

    internal string Key
    {
        get { return string.Format( "{0}|{1}|{2}", FirstName, LastName, Email ); }
    }
}

[ConfigurationCollection( typeof(UserElement), AddItemName = "user", CollectionType = ConfigurationElementCollectionType.BasicMap )]
public class UserElementCollection : ConfigurationElementCollection
{
    protected override ConfigurationElement CreateNewElement()
    {
        return new UserElement();
    }

    protected override object GetElementKey( ConfigurationElement element )
    {
        return ( (UserElement) element ).Key;
    }

    public void Add( UserElement element )
    {
        BaseAdd( element );
    }

    public void Clear()
    {
        BaseClear();
    }

    public int IndexOf( UserElement element )
    {
        return BaseIndexOf( element );
    }

    public void Remove( UserElement element )
    {
        if( BaseIndexOf( element ) >= 0 )
        {
            BaseRemove( element.Key );
        }
    }

    public void RemoveAt( int index )
    {
        BaseRemoveAt( index );
    }

    public UserElement this[ int index ]
    {
        get { return (UserElement) BaseGet( index ); }
        set
        {
            if( BaseGet( index ) != null )
            {
                BaseRemoveAt( index );
            }
            BaseAdd( index, value );
        }
    }
}

public class UserInfoSection : ConfigurationSection
{
    private static readonly ConfigurationProperty _propUserInfo = new ConfigurationProperty(
            null,
            typeof(UserElementCollection),
            null,
            ConfigurationPropertyOptions.IsDefaultCollection
    );

    private static ConfigurationPropertyCollection _properties = new ConfigurationPropertyCollection();

    static UserInfoSection()
    {
        _properties.Add( _propUserInfo );
    }

    [ConfigurationProperty( "", Options = ConfigurationPropertyOptions.IsDefaultCollection )]
    public UserElementCollection Users
    {
        get { return (UserElementCollection) base[ _propUserInfo ]; }
    }
}

Я сохранил класс UserElement простым, хотя он действительно должен следовать шаблону объявления каждого свойства полностью, как описано в этой превосходной статье CodeProject . Как видите, он представляет элементы user в вашей конфигурации, которую вы указали.

Класс UserElementCollection просто поддерживает наличие более одного «пользовательского» элемента, включая возможность добавлять / удалять / удалять элементы из коллекции, если вы хотите изменить ее во время выполнения.

Наконец, существует UserInfoSection, который просто сообщает, что у него есть набор по умолчанию "пользовательских" элементов.

Далее приведен пример файла App.config:

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <configSections>
    <sectionGroup>
      <section
        name="userInfo"
        type="ConsoleApplication1.UserInfoSection, ConsoleApplication1"
        allowDefinition="Everywhere"
        allowExeDefinition="MachineToLocalUser"
      />
    </sectionGroup>
  </configSections>

  <userInfo>
    <user firstName="John" lastName="Doe" email="John.Doe@company.com" />
    <user firstName="Jane" lastName="Doe" email="Jane.Doe@company.com" />
  </userInfo>
</configuration>

Как видите, в этом примере я включил некоторые элементы userInfo / user в App.config. Я также добавил настройки, чтобы они могли быть определены на уровне компьютера / приложения / пользователя / пользователя роуминга.

Далее нам нужно знать, как их обновлять во время выполнения, следующий код показывает пример:

Configuration userConfig = ConfigurationManager.OpenExeConfiguration( ConfigurationUserLevel.PerUserRoamingAndLocal );

var userInfoSection = userConfig.GetSection( "userInfo" ) as UserInfoSection;

var userElement = new UserElement();

userElement.FirstName = "Sample";
userElement.LastName = "User";
userElement.Email = "Sample.User@company.com";

userInfoSection.Users.Add( userElement );

userConfig.Save();

Приведенный выше код создаст новый файл user.config, если это необходимо, и находится глубоко внутри папки «Local Settings \ Application Data» для пользователя.

Если вместо этого вы хотите, чтобы новый пользователь был добавлен в файл app.config, просто измените параметр метода OpenExeConfiguration () на ConfigurationUserLevel.None.

Как видите, это достаточно просто, хотя для поиска этой информации нужно немного покопаться.

7 голосов
/ 09 апреля 2009

Я бы не стал хранить такие данные в app.config, по крайней мере, если бы они предназначались для программного обновления. Концептуально, это для параметров конфигурации, а не для данных приложения, поэтому, возможно, вы хотите сохранить информацию об имени пользователя и пароле в отдельном файле XML (при условии, что вы не можете или не хотите использовать базу данных)?

Сказав это, я думаю, что вам лучше всего прочитать в app.config стандартный файл XML, проанализировать его, добавить нужные вам узлы и записать его обратно. Встроенный в ConfigurationManager API не позволяет записывать новые настройки (что, как я полагаю, дает подсказку о предполагаемом использовании Microsoft).

1 голос
/ 09 апреля 2009

Используйте новый System.Configuration API из .Net Framework 2. (Сборка: System.Configuration) IConfigurationSectionHandler устарел.

Вы можете найти множество очень хороших образцов и описаний на http://www.codeproject.com/KB/dotnet/mysteriesofconfiguration.aspx

Существует также пример кода о том, как вы можете изменять и сохранять значения.


РЕДАКТИРОВАТЬ: Соответствующая часть документации в codeproject

Сохраняет только измененные значения, если существуют какие-либо изменения

Configuration.Save() 

Сохраняет указанный уровень изменений, если они существуют

Configuration.Save(ConfigurationSaveMode) 

Сохраняет указанный уровень изменений, вызывая сохранение, если второй параметр имеет значение true

Configuration.Save(ConfigurationSaveMode, bool)

Перечисление ConfigurationSaveMode имеет следующие значения:

  • Полный - Сохраняет все свойства конфигурации, независимо от того, изменились они или нет
  • Изменено - Сохраняет свойства, которые были изменены, даже если текущее значение совпадает с исходным
  • Минимальный - Сохраняет только свойства, которые были изменены и имеют значения, отличные от исходных
0 голосов
/ 24 марта 2011
 private void frmLocalFileTransfer_Load(object sender, EventArgs e)
    {
        try
        {
            dt.Columns.Add("Provider");
            dt.Columns.Add("DestinationPath");
            string[] folders = null;
            dt.Columns.Add("SourcePath");

            for (int i = 1; i < System.Configuration.ConfigurationManager.ConnectionStrings.Count; i++)
            {
                string key = System.Configuration.ConfigurationManager.ConnectionStrings[i].Name;
                string constr = System.Configuration.ConfigurationManager.ConnectionStrings[i].ConnectionString;
                DataGridViewTextBoxColumn column = new DataGridViewTextBoxColumn();
                folders = constr.Split('\\');
                string newstr = (folders[folders.Length - 2]);
                if (!newstr.Contains("BackUp"))
                {
                    DataRow row = dt.NewRow();
                    row[dt.Columns[0].ToString()] = key;
                    row[dt.Columns[1].ToString()] = constr;
                    row[dt.Columns[2].ToString()] = constr;
                    dt.Rows.InsertAt(row, i - 1);
                }
            }

            foreach (DataColumn dc in dt.Columns)
            {
                DataGridViewTextBoxColumn column = new DataGridViewTextBoxColumn();
                column.DataPropertyName = dc.ColumnName;
                column.HeaderText = dc.ColumnName;
                column.Name = dc.ColumnName;
                column.SortMode = DataGridViewColumnSortMode.Automatic;
                column.ValueType = dc.DataType;
                GVCheckbox();

                gevsearch.Columns.Add(column);
            }
            if (gevsearch.ColumnCount == 4)
            {
                DataGridViewButtonColumn btnColoumn = new DataGridViewButtonColumn();
                btnColoumn.Width = 150;
                btnColoumn.HeaderText = "Change SourcePath";
                btnColoumn.DefaultCellStyle.Alignment = DataGridViewContentAlignment.MiddleCenter;
                gevsearch.Columns.Insert(4, btnColoumn);


            }
             gevsearch.DataSource = dt;
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.ToString());
        }
    }

private void btnAddProvider_Click(object sender, EventArgs e)
    {
        try
        {
            System.Xml.XmlDocument xDoc = new System.Xml.XmlDocument();
            path = "D:\\Pandurang\\Jitendra\\LocalFileTransfer\\LocalFileTransfer
            xDoc.Load(path);
            System.Xml.XmlElement element = xDoc.CreateElement("add");
            element.SetAttribute("name", txtProviderName.Text.Trim());
            element.SetAttribute("connectionString", txtNewPath.Text.Trim());
            System.Xml.XmlElement elementBackup = xDoc.CreateElement("add");
            elementBackup.SetAttribute("name", BackUpName);
            elementBackup.SetAttribute("connectionString", txtBackupPath.Text.Trim());
            System.Xml.XmlNode node= xDoc.ChildNodes[1].ChildNodes[0];
            node.AppendChild(element);
            node.AppendChild(elementBackup);
            xDoc.Save(path);
          }
        catch (Exception ex)
        {
            MessageBox.Show(ex.ToString());
        }
    }
0 голосов
/ 09 апреля 2009

вы должны создать класс как:

public class MySettings : ConfigurationSection 
{
    public MySettings Settings = (MySettings)WebConfigurationManager.GetSection("MySettings");

    [ConfigurationProperty("MyConfigSetting1")]
    public string DefaultConnectionStringName
    {
        get { return (string)base["MyConfigSetting1"]; }
        set { base["MyConfigSetting1"] = value; }
    }
}

после этого в вашем web.config используйте:

<section name="MySettings" type="MyNamespace.MySettings"/>
<MySettings MyConfigSetting1="myValue">

Вот так) Если вы хотите использовать не атрибуты, а свойства, просто создайте класс, производный от ConfigurationElement, и включите его в свой класс, определенный в ConfigurationSettings.

...