在C# WPF開發中,內存泄漏是一個常見且棘手的問題。內存泄漏會導致應用程序隨著時間的推移消耗越來越多的內存,最終可能導致應用程序崩潰或者系統性能下降。以下是一些常見的內存泄漏原因以及相應的排查和解決方法。
1. 事件訂閱
在 C# 中,實現兩個獨立執行程序(EXE)之間的通信是一個常見的需求。這種通信可以通過多種方式實現,包括但不限于命名管道、匿名管道、消息隊列、內存映射文件、套接字(Sockets)以及使用 Windows API 如 SendMessage 等。本文將介紹幾種常用的方法,并提供示例代碼。
1. 使用命名管道進行通信
命名管道是一種常用的進程間通信(IPC)機制,允許不同進程之間進行雙向通信。
服務端示例代碼
using System;
using System.IO.Pipes;
using System.Text;
public class NamedPipeServer
{
public static void Main()
{
using (NamedPipeServerStream pipeServer = new NamedPipeServerStream("testpipe", PipeDirection.InOut))
{
Console.WriteLine("Waiting for client connection...");
pipeServer.WaitForConnection();
using (var reader = new StreamReader(pipeServer))
using (var writer = new StreamWriter(pipeServer))
{
writer.WriteLine("Hello from server");
writer.Flush();
string message = reader.ReadLine();
Console.WriteLine("Received from client: " + message);
}
}
}
}
客戶端示例代碼
using System;
using System.IO.Pipes;
using System.Text;
public class NamedPipeClient
{
public static void Main()
{
using (NamedPipeClientStream pipeClient = new NamedPipeClientStream(".", "testpipe", PipeDirection.InOut))
{
Console.WriteLine("Connecting to server...");
pipeClient.Connect();
using (var reader = new StreamReader(pipeClient))
using (var writer = new StreamWriter(pipeClient))
{
string message = reader.ReadLine();
Console.WriteLine("Received from server: " + message);
writer.WriteLine("Hello from client");
writer.Flush();
}
}
}
}
2. 使用匿名管道進行通信
匿名管道是另一種 IPC 機制,通常用于父子進程之間的通信。
服務端示例代碼
using System;
using System.IO.Pipes;
public class AnonymousPipeServer
{
public static void Main()
{
using (AnonymousPipeServerStream pipeServer = new AnonymousPipeServerStream(PipeDirection.Out))
{
Console.WriteLine("Writing to client...");
pipeServer.Write(new byte[] { 0x01, 0x02, 0x03, 0x04 }, 0, 4);
}
}
}
客戶端示例代碼
using System;
using System.IO.Pipes;
public class AnonymousPipeClient
{
public static void Main()
{
using (AnonymousPipeClientStream pipeClient = new AnonymousPipeClientStream(PipeDirection.In))
{
AnonymousPipeServerStream pipeServer = new AnonymousPipeServerStream(PipeDirection.Out, pipeClient);
pipeServer.DisposeLocalCopyOfClientHandle();
byte[] buffer = new byte[4];
int bytesRead = pipeClient.Read(buffer, 0, 4);
Console.WriteLine("Read {0} bytes from server.", bytesRead);
}
}
}
3. 使用 SendMessage 實現通信
SendMessage 是一個 Windows API 函數,可以用來在兩個獨立的 EXE 程序之間發送消息。
發送端示例代碼
using System;
using System.Runtime.InteropServices;
public class SendMessageSender
{
public const int WM_COPYDATA = 0x004A;
[DllImport("User32.dll", EntryPoint = "FindWindow")]
private static extern int FindWindow(string lpClassName, string lpWindowName);
[DllImport("User32.dll", EntryPoint = "SendMessage")]
private static extern int SendMessage(int hWnd, int Msg, int wParam, ref COPYDATASTRUCT lParam);
public struct COPYDATASTRUCT
{
public IntPtr dwData;
public int cbData;
[MarshalAs(UnmanagedType.LPStr)]
public string lpData;
}
public static void Main()
{
string message = "Hello from Sender";
COPYDATASTRUCT cds;
cds.dwData = IntPtr.Zero;
cds.cbData = message.Length + 1;
cds.lpData = message;
int hwnd = FindWindow(null, "Receiver Window Title");
SendMessage(hwnd, WM_COPYDATA, 0, ref cds);
}
}
接收端示例代碼
using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
public class SendMessageReceiver : Form
{
public const int WM_COPYDATA = 0x004A;
public SendMessageReceiver()
{
// Set up the main window...
}
protected override void WndProc(ref Message m)
{
if (m.Msg == WM_COPYDATA)
{
COPYDATASTRUCT cds = (COPYDATASTRUCT)Marshal.PtrToStructure(m.LParam, typeof(COPYDATASTRUCT));
string text = Marshal.PtrToStringAnsi(cds.lpData, cds.cbData - 1);
MessageBox.Show(text);
}
base.WndProc(ref m);
}
public static void Main()
{
Application.Run(new SendMessageReceiver());
}
}
結論
以上介紹了幾種在 C# 中實現兩個 EXE 程序之間通信的方法。每種方法都有其適用場景和優缺點。命名管道適用于需要全雙工通信的場景,匿名管道適用于父子進程間的通信,而 SendMessage 則適用于簡單的數據傳遞。開發者可以根據具體需求選擇合適的通信方式。
該文章在 2024/10/28 16:33:17 編輯過