如何将文件扩展名关联到 C# 中的当前可执行文件

Posted

技术标签:

【中文标题】如何将文件扩展名关联到 C# 中的当前可执行文件【英文标题】:How to associate a file extension to the current executable in C# 【发布时间】:2010-09-09 08:16:29 【问题描述】:

我想将文件扩展名与 C# 中的当前可执行文件相关联。 这样,当用户随后在资源管理器中单击文件时,它将以给定文件作为第一个参数运行我的可执行文件。 理想情况下,它还将给定文件扩展名的图标设置为我的可执行文件的图标。 谢谢大家。

【问题讨论】:

codeproject.com/KB/dotnet/System_File_Association.aspx 那么csharpfriends.com/Forums/ShowPost.aspx?PostID=32627 可能会有所帮助。如果需要,您可以随时通过 API 直接为关联进行注册表编辑...但由您决定 编辑:还可以阅读那里的 cmets,一些优点,例如使用带空格的路径的引号:@"C:\ SomePath\MyApp.exe ""%1""" 还有注册表访问权限问题,尤其是 Vista...祝你好运:) 这已在以下stack overflow thread 中得到彻底解答。我们一直在使用这个实现并且效果很好。它也是开源的,并集成到 MSBuild 中。 这已在以下线程中得到彻底回答:***.com/questions/2993118/… @BlakeNiemyjski 您关于将图标嵌入到 .NET 程序集中的链接如何与这个关于文件扩展名关联的问题相关? 我正在链接到堆栈溢出帖子,其中包含有关通过 Wix 安装程序关联文件扩展名的有用 wix 信息。 【参考方案1】:

文件关联在注册表中的 HKEY_CLASSES_ROOT 下定义。

有一个 VB.NET 示例 here,您可以轻松地将其移植到 C#。

【讨论】:

【参考方案2】:

似乎没有用于直接管理文件关联的 .Net API,但您可以使用 Registry 类来读取和写入所需的密钥。

您需要在 HKEY_CLASSES_ROOT 下创建一个密钥,并将名称设置为您的文件扩展名(例如:“.txt”)。将此键的默认值设置为文件类型的唯一名称,例如“Acme.TextFile”。然后在 HKEY_CLASSES_ROOT 下创建另一个键,名称设置为“Acme.TextFile”。添加一个名为“DefaultIcon”的子项,并将该项的默认值设置为包含您希望用于此文件类型的图标的文件。添加另一个名为“shell”的兄弟。在“shell”键下,为您希望通过资源管理器上下文菜单提供的每个操作添加一个键,将每个键的默认值设置为可执行文件的路径,后跟空格和“%1”表示路径到选定的文件。

例如,这是一个示例注册表文件,用于在 .txt 文件和 EmEditor 之间创建关联:

Windows 注册表编辑器版本 5.00 [HKEY_CLASSES_ROOT\.txt] @="emeditor.txt" [HKEY_CLASSES_ROOT\emeditor.txt] @="文本文档" [HKEY_CLASSES_ROOT\emeditor.txt\DefaultIcon] @="%SystemRoot%\\SysWow64\\imageres.dll,-102" [HKEY_CLASSES_ROOT\emeditor.txt\shell] [HKEY_CLASSES_ROOT\emeditor.txt\shell\open] [HKEY_CLASSES_ROOT\emeditor.txt\shell\open\command] @="\"C:\\Program Files\\EmEditor\\EMEDITOR.EXE\" \"%1\"" [HKEY_CLASSES_ROOT\emeditor.txt\shell\print] [HKEY_CLASSES_ROOT\emeditor.txt\shell\print\command] @="\"C:\\Program Files\\EmEditor\\EMEDITOR.EXE\" /p \"%1\""

【讨论】:

有没有办法将一些参数传递给 exe(在本例中是 C:\Program Files\EmEditor\EMEDITOR.EXE)? @Apparao 是的,以打印命令为例。它在文件名之前传递 /p 参数。 我知道这个答案已有十多年的历史了,但仍然 - 我认为这不是将纯文本 (*.txt) 的公共文件格式与随机程序——这基本上是“接管”文件关联,如果至少之前已经有文本文件与某些程序的关联,则覆盖它。微软文档,尽管它是不令人满意的,但至少指出将程序与公共文件类型相关联(使用“打开方式”上下文菜单)的方法是使用文件类型键下的OpenWithProgIds 注册表键。【参考方案3】:

此外,如果您决定采用注册表方式,请记住当前用户关联位于 HKEY_CURRENT_USER\Software\Classes 下。将您的应用程序添加到那里而不是本地机器类可能会更好。

如果您的程序将由有限的用户运行,您将无法修改 CLASSES_ROOT。

【讨论】:

【参考方案4】:

您可能出于特定原因选择不为您的项目使用安装包,但安装包是轻松执行应用程序配置任务(例如注册文件扩展名、添加桌面快捷方式等)的好地方。

以下是使用内置 Visual Studio 安装工具创建文件扩展名关联的方法:

    在您现有的 C# 解决方案中,添加一个新项目并选择项目类型为 Other Project Types -> Setup and Deployment -> Setup Project(或尝试安装向导)

    配置您的安装程序(如果您需要帮助,可以使用大量现有文档)

    右键单击解决方案资源管理器中的安装项目,选择View -> File Types,然后添加您要注册的扩展程序以及运行它的程序。

如果用户为您的应用程序运行卸载,此方法还有一个额外的好处,即自行清理。

【讨论】:

【参考方案5】:

如果您使用 ClickOnce 部署,这一切都会为您处理(至少,在 VS2008 SP1 中);简单地说:

项目属性 发布 选项 文件关联 (添加您需要的任何内容)

(请注意,它必须是完全信任的,面向 .NET 3.5,并设置为离线使用)

另见 MSDN:How to: Create File Associations For a ClickOnce Application

【讨论】:

我有 VS2008 Professional,SP1,但我没有看到这个选项。我一定是错过了什么。【参考方案6】:

具体说一下“Windows 注册表”的方式:

我在 HKEY_CURRENT_USER\Software\Classes 下创建密钥(就像 Ishmaeel 所说)

并按照 X-Cubed 回答的说明进行操作。

示例代码如下:

private void Create_abc_FileAssociation()

    /***********************************/
    /**** Key1: Create ".abc" entry ****/
    /***********************************/
    Microsoft.Win32.RegistryKey key1 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("Software", true);

    key1.CreateSubKey("Classes");
    key1 = key1.OpenSubKey("Classes", true);

    key1.CreateSubKey(".abc");
    key1 = key1.OpenSubKey(".abc", true);
    key1.SetValue("", "DemoKeyValue"); // Set default key value

    key1.Close();

    /*******************************************************/
    /**** Key2: Create "DemoKeyValue\DefaultIcon" entry ****/
    /*******************************************************/
    Microsoft.Win32.RegistryKey key2 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("Software", true);

    key2.CreateSubKey("Classes");
    key2 = key2.OpenSubKey("Classes", true);

    key2.CreateSubKey("DemoKeyValue");
    key2 = key2.OpenSubKey("DemoKeyValue", true);

    key2.CreateSubKey("DefaultIcon");
    key2 = key2.OpenSubKey("DefaultIcon", true);
    key2.SetValue("", "\"" + "(The icon path you desire)" + "\""); // Set default key value

    key2.Close();

    /**************************************************************/
    /**** Key3: Create "DemoKeyValue\shell\open\command" entry ****/
    /**************************************************************/
    Microsoft.Win32.RegistryKey key3 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("Software", true);

    key3.CreateSubKey("Classes");
    key3 = key3.OpenSubKey("Classes", true);

    key3.CreateSubKey("DemoKeyValue");
    key3 = key3.OpenSubKey("DemoKeyValue", true);

    key3.CreateSubKey("shell");
    key3 = key3.OpenSubKey("shell", true);

    key3.CreateSubKey("open");
    key3 = key3.OpenSubKey("open", true);

    key3.CreateSubKey("command");
    key3 = key3.OpenSubKey("command", true);
    key3.SetValue("", "\"" + "(The application path you desire)" + "\"" + " \"%1\""); // Set default key value

    key3.Close();

只是给你们看一个简单的演示,非常容易理解。您可以修改这些键值,一切顺利。

【讨论】:

不要忘记添加必要的代码来通知外壳使用图标。 SHChangeNotify(0x8000000, 0, 0, 0);【参考方案7】:

这是一个完整的例子:

public class FileAssociation

    public string Extension  get; set; 
    public string ProgId  get; set; 
    public string FileTypeDescription  get; set; 
    public string ExecutableFilePath  get; set; 


public class FileAssociations

    // needed so that Explorer windows get refreshed after the registry is updated
    [System.Runtime.InteropServices.DllImport("Shell32.dll")]
    private static extern int SHChangeNotify(int eventId, int flags, IntPtr item1, IntPtr item2);

    private const int SHCNE_ASSOCCHANGED = 0x8000000;
    private const int SHCNF_FLUSH = 0x1000;

    public static void EnsureAssociationsSet()
    
        var filePath = Process.GetCurrentProcess().MainModule.FileName;
        EnsureAssociationsSet(
            new FileAssociation
            
                Extension = ".binlog",
                ProgId = "MSBuildBinaryLog",
                FileTypeDescription = "MSBuild Binary Log",
                ExecutableFilePath = filePath
            ,
            new FileAssociation
            
                Extension = ".buildlog",
                ProgId = "MSBuildStructuredLog",
                FileTypeDescription = "MSBuild Structured Log",
                ExecutableFilePath = filePath
            );
    

    public static void EnsureAssociationsSet(params FileAssociation[] associations)
    
        bool madeChanges = false;
        foreach (var association in associations)
        
            madeChanges |= SetAssociation(
                association.Extension,
                association.ProgId,
                association.FileTypeDescription,
                association.ExecutableFilePath);
        

        if (madeChanges)
        
            SHChangeNotify(SHCNE_ASSOCCHANGED, SHCNF_FLUSH, IntPtr.Zero, IntPtr.Zero);
        
    

    public static bool SetAssociation(string extension, string progId, string fileTypeDescription, string applicationFilePath)
    
        bool madeChanges = false;
        madeChanges |= SetKeyDefaultValue(@"Software\Classes\" + extension, progId);
        madeChanges |= SetKeyDefaultValue(@"Software\Classes\" + progId, fileTypeDescription);
        madeChanges |= SetKeyDefaultValue($@"Software\Classes\progId\shell\open\command", "\"" + applicationFilePath + "\" \"%1\"");
        return madeChanges;
    

    private static bool SetKeyDefaultValue(string keyPath, string value)
    
        using (var key = Registry.CurrentUser.CreateSubKey(keyPath))
        
            if (key.GetValue(null) as string != value)
            
                key.SetValue(null, value);
                return true;
            
        

        return false;
    

【讨论】:

【参考方案8】:

自 Windows 7 以来就有两个 cmd 工具,可以很容易地创建简单的文件关联。他们是assoc 和ftype。这是每个命令的基本解释。

Assoc - 将文件扩展名(如“.txt”)与“文件类型”相关联。 FType - 定义在用户打开给定“文件类型”时运行的可执行文件。

请注意,这些是 cmd 工具,而不是可执行文件 (exe)。这意味着它们只能在 cmd 窗口中运行,或者通过使用带有“cmd /c assoc”的 ShellExecute 来运行。您可以通过链接或键入“assoc /?”了解更多关于它们的信息。和“ftype /?”在 cmd 提示符下。

因此,要将应用程序与 .bob 扩展名关联,您可以打开一个 cmd 窗口(WindowKey+R,键入 cmd,按 Enter)并运行以下命令:

assoc .bob=BobFile
ftype BobFile=c:\temp\BobView.exe "%1"

这比弄乱注册表要简单得多,而且更有可能在未来的 Windows 版本中工作。

总结一下,这是一个创建文件关联的 C# 函数:

public static int setFileAssociation(string[] extensions, string fileType, string openCommandString) 
    int v = execute("cmd", "/c ftype " + fileType + "=" + openCommandString);
    foreach (string ext in extensions) 
        v = execute("cmd", "/c assoc " + ext + "=" + fileType);
        if (v != 0) return v;
    
    return v;

public static int execute(string exeFilename, string arguments) 
    ProcessStartInfo startInfo = new ProcessStartInfo();
    startInfo.CreateNoWindow = false;
    startInfo.UseShellExecute = true;
    startInfo.FileName = exeFilename;
    startInfo.WindowStyle = ProcessWindowStyle.Hidden;
    startInfo.Arguments = arguments;
    try 
        using (Process exeProcess = Process.Start(startInfo)) 
            exeProcess.WaitForExit();
            return exeProcess.ExitCode;
        
     catch 
        return 1;
    

【讨论】:

【参考方案9】:

下面的代码是一个应该可以工作的函数,它在 Windows 注册表中添加了所需的值。通常我在我的可执行文件中运行 SelfCreateAssociation(".abc") 。 (表单构造函数或 onload 或 onshown)每次执行可执行文件时,它将更新当前用户的注册表项。 (适合调试,如果您有一些更改)。 如果您需要有关所涉及的注册表项的详细信息,请查看此 MSDN 链接。

https://msdn.microsoft.com/en-us/library/windows/desktop/dd758090(v=vs.85).aspx

获取有关常规 ClassesRoot 注册表项的更多信息。请参阅这篇 MSDN 文章。

https://msdn.microsoft.com/en-us/library/windows/desktop/ms724475(v=vs.85).aspx

public enum KeyHiveSmall

    ClassesRoot,
    CurrentUser,
    LocalMachine,


/// <summary>
/// Create an associaten for a file extension in the windows registry
/// CreateAssociation(@"vendor.application",".tmf","Tool file",@"C:\Windows\SYSWOW64\notepad.exe",@"%SystemRoot%\SYSWOW64\notepad.exe,0");
/// </summary>
/// <param name="ProgID">e.g. vendor.application</param>
/// <param name="extension">e.g. .tmf</param>
/// <param name="description">e.g. Tool file</param>
/// <param name="application">e.g.  @"C:\Windows\SYSWOW64\notepad.exe"</param>
/// <param name="icon">@"%SystemRoot%\SYSWOW64\notepad.exe,0"</param>
/// <param name="hive">e.g. The user-specific settings have priority over the computer settings. KeyHive.LocalMachine  need admin rights</param>
public static void CreateAssociation(string ProgID, string extension, string description, string application, string icon, KeyHiveSmall hive = KeyHiveSmall.CurrentUser)

    RegistryKey selectedKey = null;

    switch (hive)
    
        case KeyHiveSmall.ClassesRoot:
            Microsoft.Win32.Registry.ClassesRoot.CreateSubKey(extension).SetValue("", ProgID);
            selectedKey = Microsoft.Win32.Registry.ClassesRoot.CreateSubKey(ProgID);
            break;

        case KeyHiveSmall.CurrentUser:
            Microsoft.Win32.Registry.CurrentUser.CreateSubKey(@"Software\Classes\" + extension).SetValue("", ProgID);
            selectedKey = Microsoft.Win32.Registry.CurrentUser.CreateSubKey(@"Software\Classes\" + ProgID);
            break;

        case KeyHiveSmall.LocalMachine:
            Microsoft.Win32.Registry.LocalMachine.CreateSubKey(@"Software\Classes\" + extension).SetValue("", ProgID);
            selectedKey = Microsoft.Win32.Registry.LocalMachine.CreateSubKey(@"Software\Classes\" + ProgID);
            break;
    

    if (selectedKey != null)
    
        if (description != null)
        
            selectedKey.SetValue("", description);
        
        if (icon != null)
        
            selectedKey.CreateSubKey("DefaultIcon").SetValue("", icon, RegistryValueKind.ExpandString);
            selectedKey.CreateSubKey(@"Shell\Open").SetValue("icon", icon, RegistryValueKind.ExpandString);
        
        if (application != null)
        
            selectedKey.CreateSubKey(@"Shell\Open\command").SetValue("", "\"" + application + "\"" + " \"%1\"", RegistryValueKind.ExpandString);
        
    
    selectedKey.Flush();
    selectedKey.Close();


 /// <summary>
    /// Creates a association for current running executable
    /// </summary>
    /// <param name="extension">e.g. .tmf</param>
    /// <param name="hive">e.g. KeyHive.LocalMachine need admin rights</param>
    /// <param name="description">e.g. Tool file. Displayed in explorer</param>
    public static void SelfCreateAssociation(string extension, KeyHiveSmall hive = KeyHiveSmall.CurrentUser, string description = "")
    
        string ProgID = System.Reflection.Assembly.GetExecutingAssembly().EntryPoint.DeclaringType.FullName;
        string FileLocation = System.Reflection.Assembly.GetExecutingAssembly().Location;
        CreateAssociation(ProgID, extension, description, FileLocation, FileLocation + ",0", hive);
    

【讨论】:

留下一大块没有解释的代码对未来的读者几乎没有帮助。请尝试添加一些关于此代码的作用的解释。 代码正在运行。源代码中的解释很好,对我来说很完美。

以上是关于如何将文件扩展名关联到 C# 中的当前可执行文件的主要内容,如果未能解决你的问题,请参考以下文章

移动当前可执行文件 c#

如何在 C# 中获取当前可执行文件的名称?

将 C# 控制台应用程序发布为独立的可执行文件

如何为文件类型创建所有文件关联(友好的应用程序名称和可执行文件)的 C# 列表

如何将提交哈希添加到可执行文件的详细信息

如何让shell脚本变成可执行文件