我目前在.Net中维护一个内部应用程序。该应用程序使用IPC一次维护一个正在运行的会话;如果另一个会话尝试打开(某人再次点击该图标,一些会打开一个保存的结果文件),第二个会话将此信息传达给第一个会话,然后终止,第一个会话然后启动请求的操作。
我目前使用System.ServiceModel命名空间执行此操作,如下所示:
命名空间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
给出的解决方法是将应用程序设置为需要管理员权限。这不起作用有两个原因:
所以在这一点上,我需要找到一个不需要管理员权限的IPC解决方案,让我实现最初的目标:检测已经运行的代码实例并告诉它该做什么,否则启动自己。任何关于该做什么的建议(在.Net框架内,请不要提供第三方解决方案)将不胜感激。
答案 0 :(得分:3)
确保您的应用的单个会话非常困难(并且您已经发现,有问题的方式)。您应该将该功能重构/重做为use a mutex。
这是另一个example. 必要的wiki article about mutexes。
修改强>
要绕过通信位,你可以生成一个带有你自己的特殊前缀的临时文件,或者在OS提供的临时文件夹中扩展,单个应用程序将轮询(过滤你的特殊前缀和/或扩展名)以查看是否有更多请求已经提出。参见:
System.IO.Path.GetTempFileName
和
System.IO.Path.GetTempPath
答案 1 :(得分:2)
明确使用named mutex将应用程序限制为单个实例,然后您可以使用NamedPipeServerStream和NamedPipeClientStream来执行不需要管理员权限的IPC。