如何使应用程序设置独立于.exe路径?

时间:2012-08-18 07:56:20

标签: c# .net application-settings

假设我想将我的应用程序从c:\ myapp.exe移动到d:\ myapp.exe。它会松开所有设置。如何防止?

5 个答案:

答案 0 :(得分:4)

我个人使用注册表来保存和加载我的设置,因此我的应用程序的位置没有受到影响,但如果您使用User.Config等并想要修复该位置,这可能会有所帮助: Can I control the location of .NET user settings to avoid losing settings on application upgrade?

答案 1 :(得分:4)

这是LocalFileSettingsProvider类的实现细节。其中存在令人遗憾的工作,即将用户范围的设置存储在保证唯一的文件中,以便不同的应用程序不会意外地覆盖彼此的设置。

通过将文件存储在具有散列名称的AppData目录中来实现。哈希值是从应用程序的几个属性计算出来的。它从AssemblyInfo.cs文件中的属性开始,尽可能多地抓取它。特别是[AssemblyVersion]很重要,它可以检测到新版本的应用程序可能与旧版本的user.config文件不兼容。

但是属性不足以使其唯一,它还使用散列中.exe的完整路径名。对于相应的.config文件,这是一个非常强大的选择器。

所以,不可避免的是,如果你将.exe移动到其他地方,那将会改变哈希值,这会让你获得一个空的user.config文件,并将所有设置恢复为默认设置。

修补此问题有点可疑,应用程序应该只有一个安装目录。 c:\ program files \ companyname \ appname是标准。但您可以通过实现自己的SettingsProvider类。这并不容易,System.Configuration是一个非常讨厌的命名空间。但是一个不错的起点是RegistrySettingsProvider sample,它可能原样可用。

      -

答案 2 :(得分:1)

这取决于应用程序的100%。

应用程序本身只需要找到它的依赖项,或运行它所需的DLL列表。它大部分时间都会在当前目录中查找,所以这通常不是问题。

最大的问题是在注册表中。如果应用程序已将其安装到注册表的位置,它可能会在运行时查找旧目录中的某些文件。

如果您安装了应用程序,它也会存储在注册表中,从“添加/删除”程序中卸载将不再有效。

如果应用程序不使用注册表,则可以移动它而不会产生任何后果。许多使用闪存驱动器的便携式应用都采用这种方法,因此可以根据需要移动或删除...

希望它有助于你的事业.. :)

答案 3 :(得分:1)

您可以创建自己的“设置”类。哪个像原版一样有效。下面我发布了我的Settings类实现。任何改进都将不胜感激。

using System;
using System.ComponentModel;
using System.Drawing;
using System.Reflection;
using System.Windows.Forms;
using Microsoft.Win32;

namespace MyNamespace
{
    static class Settings
    {
        private static string _connectionString = @"data source=C:\\database.s3db";

        public static string ConnectionString
        {
            get { return GetSetting("_connectionString"); }
            set { _connectionString = value; }
        }

        private static string _archives = "";

        public static string Archives
        {
            get { return GetSetting("_archives"); }
            set { _archives = value; }
        }

        public static bool CheckDuplicates
        {
            get { return bool.Parse(GetSetting("_checkDuplicates")); }
            set { _checkDuplicates = value; }
        }

        private static bool _saveDocks = true;

        public static bool SaveDocks
        {
            get { return bool.Parse(GetSetting("_saveDocks")); }
            set { _saveDocks = value; }
        }

        private static Font _font = new Font("Tahoma", 12, GraphicsUnit.Pixel);

        public static Font Font
        {
            get
            {
                var convert = new FontConverter();
                var value = convert.ConvertFromString(GetSetting("_font"));
                return (Font) value;
            }
            set { _font = value; }
        }

        public static void Save()
        {
            Type type = typeof(Settings);

            var registryKey = Registry.CurrentUser.CreateSubKey(string.Format(@"Software\{0}\{1}\Settings", Application.CompanyName, Application.ProductName));
            if (registryKey != null)
            {
                foreach (var field in type.GetFields(BindingFlags.NonPublic | BindingFlags.Static))
                {
                    var converter = TypeDescriptor.GetConverter(field.FieldType);
                    var value = converter.ConvertToString(field.GetValue(null));
                    registryKey.SetValue(field.Name, value ?? field.GetValue(null));
                }
                registryKey.Close();
            }
        }

        public static void SetDefaults()
        {
            var registryKey = Registry.CurrentUser.OpenSubKey(string.Format(@"Software\{0}\{1}\Settings", Application.CompanyName, Application.ProductName));
            if (registryKey == null)
            {
                Save();
            }
            else
            {
                registryKey = Registry.CurrentUser.CreateSubKey(string.Format(@"Software\{0}\{1}\Settings", Application.CompanyName, Application.ProductName));
                if(registryKey == null) return;
                Type type = typeof(Settings);
                foreach (var field in type.GetFields(BindingFlags.NonPublic | BindingFlags.Static))
                {
                    if (registryKey.GetValue(field.Name) != null)
                    {
                        var converter = TypeDescriptor.GetConverter(field.FieldType);
                        var value = converter.ConvertFrom(registryKey.GetValue(field.Name, field.GetValue(null)));
                        field.SetValue(null, value);
                    }
                }
                registryKey.Close();
            }
        }

        private static string GetSetting(string name)
        {
            var registryKey = Registry.CurrentUser.OpenSubKey(string.Format(@"Software\{0}\{1}\Settings", Application.CompanyName, Application.ProductName));
            if (registryKey != null)
            {
                if (registryKey.GetValue(name) != null)
                {
                    return registryKey.GetValue(name).ToString();
                }
                registryKey.Close();
            }
            return "";
        }
    }
}

要在您的应用程序中使用它,只需像上面一样为您的设置添加属性和支持字段。确保在属性的get访问器中的GetSetting方法中使用支持字段的名称作为字符串参数。确保将默认值分配给设置字段。

有关保存设置,请参阅以下代码。

    Settings.Archives = ".7z,.rar,.zip";
    Settings.CheckDuplicates = true;
    Settings.SaveDocks = false;
    Settings.Font = fontDialog.Font;
    Settings.Save();

您必须在主窗体的构造函数中调用SetDefaults方法。见下面的代码。

namespace MyNamespace
{
    public partial class FormMain : Form
    {
        public FormMain()
        {
            InitializeComponent();
            Settings.SetDefaults();
        }
    }
}

如果您有改进此课程的建议。然后评论。

答案 4 :(得分:0)

创建一个单独的dll,通过您喜欢的任何方法读取设置 - 注册表或xml读取,...并将其放入系统路径。然后你可以在exe的任何地方调用dll。

但在什么情况下你需要这个要求?我只是想知道。