单个MSI安装正确的32或64位c#应用程序

时间:2010-09-16 08:30:13

标签: c# windows windows-installer 32bit-64bit

我有一个C#应用程序,它是为x86(32位)和x64(64位)平台构建的。我的构建系统目前输出两个MSI安装程序,每个平台一个。如果它有所不同,我的C#应用​​程序包含一个Windows任务栏工具栏,这意味着必须由explorer.exe进程加载已安装的DLL。

是否可以生成单个MSI安装程序,它将根据当前操作系统是否为64位操作系统安装正确版本的应用程序?

目前已通过使用http://dotnetinstaller.codeplex.com/生成执行体系结构检查然后启动正确MSI的EXE来实现。但是,我更喜欢纯粹基于MSI的方法。

2 个答案:

答案 0 :(得分:7)

不,这是不可能的。请参阅Heath Stewart的Different Packages are Required for Different Processor Architectures帖子。使用MSI处理此问题的唯一方法是使用您描述的引导程序。如果您只需要将文件或一两个密钥放在64位位置,则可以(但不建议)在自定义操作中执行此操作,但更改目标安装位置并使用内置MSI文件支持赢得了'工作。

答案 1 :(得分:7)

你可以解决这个问题。在第三个部署项目下打包2个安装程序。创建一个自定义操作,检查正在运行的操作系统版本,然后让安装程序调用正确的安装程序。

这样的事情:

[RunInstaller(true)]
public partial class MyInstaller: Installer
{
    String installerPath;

    public MyInstaller()
    {
        InitializeComponent();       
        if (Is64Bit())//running as 64-bit
        {
            installerPath= @"installfolder\my64bitsetup.exe";
        }
        else
        {
            installerPath= @"installfolder\my32bitsetup.exe";
        }
    }

    [SecurityPermission(SecurityAction.Demand)]
    public override void Install(IDictionary stateSaver)
    {
        base.Install(stateSaver);
    }

    [SecurityPermission(SecurityAction.Demand)]
    public override void Commit(IDictionary savedState)
    {
        base.Commit(savedState);
        MyInstall();
    }

    [SecurityPermission(SecurityAction.Demand)]
    public override void Rollback(IDictionary savedState)
    {
        base.Rollback(savedState);
    }

    [SecurityPermission(SecurityAction.Demand)]
    public override void Uninstall(IDictionary savedState)
    {
        base.Uninstall(savedState);
        base.Commit(savedState);
    }

    private void MyInstall()
    {
         ProcessStartInfo procStartInfo = new ProcessStartInfo("cmd.exe", "/c " + installerPath);
        RunProcess(procStartInfo);
    }

    private void RunProcess(ProcessStartInfo procStartInfo)
    {
        Process proc = new Process();
        proc.StartInfo = procStartInfo;
        proc.Start();
        proc.WaitForExit();
    }

[DllImport("kernel32.dll", SetLastError = true, CallingConvention = CallingConvention.Winapi)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool IsWow64Process([In] IntPtr hProcess, [Out] out bool lpSystemInfo);

private bool Is64Bit()
{
    return (IntPtr.Size == 8 || (IntPtr.Size == 4 && Is32BitProcessOn64BitProcessor()));
}

private bool Is32BitProcessOn64BitProcessor()
{
    bool retVal;
    IsWow64Process(Process.GetCurrentProcess().Handle, out retVal);
    return retVal;
}

好的,那很长......

无论如何,在提交中你可以确定安装程序已经解压缩,只需确保你有正确的路径。 (您可以将cmd命令从/ c更改为/ k进行测试,这将使命令提示符窗口保持活动状态,以便您查看消息)

您可以阅读有关自定义操作的更多信息,可以通过参数传递安装路径。