6

我目前在.Net 中维护一个内部应用程序。应用程序使用 IPC 一次维护一个正在运行的会话;如果另一个会话尝试打开(有人再次单击图标,有人打开保存的结果文件),第二个会话将其传达给第一个会话,然后终止,第一个会话然后启动请求的操作。

我目前使用 System.ServiceModel 命名空间来执行此操作,如下所示:

namespace EventLogViewer.IPCServer { //提供进程间通信的通用框架。

[ServiceContract(Namespace = "http://ELV.domain")]
interface IELVLauncher
{
    [OperationContract]
    bool LaunchTabs(String[] fileNames);
}
public delegate void TabLauncher(String[] fileNames);
class ELVLauncherServer : IDisposable
{
    ServiceHost ipcService;
    public ELVLauncherServer()
    {
        Uri baseAddress = new Uri("Http://localhost:45600/elvlauncher/service");
        String address = "net.pipe://localhost/elvlauncher/tablauncher";

        ipcService = new ServiceHost(typeof(ELVLauncher), baseAddress);

        NetNamedPipeBinding binding = new NetNamedPipeBinding(NetNamedPipeSecurityMode.None);
        ipcService.AddServiceEndpoint(typeof(IELVLauncher), binding, address);

        ServiceMetadataBehavior behavior = new ServiceMetadataBehavior();
        behavior.HttpGetEnabled = true;
        behavior.HttpGetUrl = new Uri("http://localhost:45601/elvlauncher");
        ipcService.Description.Behaviors.Add(behavior);
        Utilities.WriteMemoryDebugStatement(new DebugStatement(DebugStatement.StatementType.INFO, "Registering IPC Service"));
        ipcService.Open();

    }

    #region IDisposable Members

    public void Dispose()
    {
        ipcService.Close();
    }

    #endregion
}
public class ELVLauncher : IELVLauncher
{
    #region IELVLauncher Members

    public bool LaunchTabs(string[] fileNames)
    {
        try
        {
            Utilities.WriteMemoryDebugStatement(new DebugStatement(DebugStatement.StatementType.INFO, String.Format("IPC request received, files: {0}", String.Join(", ", fileNames))));
            Program.mainWindow.backgroundListener_OnLaunchRequestReceived(fileNames);
        }
        catch (Exception exception)
        {
            Utilities.WriteMemoryDebugStatement(new DebugStatement(exception));
        }
        return (true);
    }

    #endregion

}

这段代码最初是在 Windows XP 下开发的,就像大多数公司一样,我们没有迁移到 Vista。由于我们都以本地管理员身份运行,因此此代码运行没有问题。

但是,我们现在正在过渡到 Win 7,并且此代码在启动时会产生异常,因为它无法注册 IPC 端点。更多信息在这里: http: //mark.michaelis.net/Blog/WindowsCommunicationFoundationWithWindowsVistaAndUAC.aspx

给出的解决方法是将应用程序升级为需要管理员权限。这不起作用有两个原因:

  1. 此应用程序不需要管理员权限,否则
  2. 我们使用 ClickOnce 来部署我们的内部工具,并且 ClickOnce 不支持除了作为进程启动器运行之外的任何内容。

所以在这一点上,我需要找到一个不需要管理员权限的 IPC 解决方案,让我实现最初的目标:检测已经运行的代码实例并告诉它要做什么,否则会自行启动。任何关于做什么的建议(在.Net框架内,请不要第三方解决方案)将不胜感激。

4

2 回答 2

3

确保应用程序的单个会话非常困难(并且正如您所发现的那样,有问题的方式)。您应该重构/重做该功能以使用 mutex

这是另一个例子。 以及关于 mutexes的必要wiki 文章

编辑

为了绕过通信位,您可以在操作系统提供的临时文件夹中生成一个具有您自己的特殊前缀和/或扩展名的临时文件,单个应用程序将轮询(过滤您的特殊前缀和/或扩展名)以查看是否发出了更多请求. 看:

System.IO.Path.GetTempFileName 

System.IO.Path.GetTempPath
于 2010-06-18T17:58:16.430 回答
2

绝对使用命名互斥体将应用程序限制为单个实例,然后您可以使用NamedPipeServerStreamNamedPipeClientStream来执行不需要管理员权限的 IPC。

于 2010-06-18T18:51:29.310 回答