C# 將NamedPipeClientStream封裝爲事件驅動的組件

 本封裝的組件,能夠方便鏈接本機或遠程 主機的命名管道。數組

鏈接遠程主機時,須要提供賬號和密碼。安全

同時解決了委託事件中修改界面元素時引發的安全錯誤。服務器

代碼爲【調試】版,有很多調試信息傳遞到調用者。async

using System;
using System.ComponentModel;
using System.IO.Pipes;
using System.Security.Principal;
using System.Threading;
using System.Drawing;

namespace LeesNamedPipeClient
{
    [DefaultProperty("PipeName")]
    [DefaultEvent("OnPipeReadData")]
    [ToolboxBitmap("fifo.ico")]
    public partial class LeesNamedPipeClient : Component
    {

        NamedPipeClientStream _PipeClient = null;

        public LeesNamedPipeClient()
        {
            InitializeComponent();
        }

        public LeesNamedPipeClient(IContainer container)
        {
            container.Add(this);            
            InitializeComponent();
        }

        #region 屬性
        [Browsable(true), Category("鏈接設置"), Description("設置鏈接的主機"), DefaultValue(".")]
        public string Host { get; set; }

        private bool _isLocalHost;
        [Browsable(true), Category("鏈接設置"), Description("是否本機"), DefaultValue("true")]
        public bool IsLocalHost
        {
            set
            {                
                _isLocalHost = value;
                if (value)
                {
                    Host = ".";
                }
            }
            get
            {
                return _isLocalHost;
            }
        }


        [Browsable(true), Category("鏈接設置"), Description("鏈接到遠程主機的賬號,本機鏈接忽略此項。"), DefaultValue("")]
        public string UserName { get; set; }


        [Browsable(true), Category("鏈接設置"), Description("鏈接到遠程主機的密碼,本機鏈接忽略此項。"), DefaultValue("")]
        public string PassWord { get; set; }


        [Browsable(true), Category("鏈接設置"), Description("管道名字"), DefaultValue("PipeName")]
        public string PipeName { get; set; }

        [Browsable(true), Category("鏈接設置"), Description("超時時間(毫秒)"), DefaultValue(1000)]
        public uint TimeOut { get; set; }


        [Browsable(false)]
        /// <summary>
        /// 獲取一個值,該值指示當前流是否支持讀操做。
        /// 若是流支持讀操做,則爲 true;不然爲 false。
        /// </summary>
        public bool CanRead { get { return _PipeClient.CanRead; } }

        [Browsable(false)]
        /// <summary>
        /// 獲取一個值,該值指示當前管道是否支持寫操做。
        /// 若是管道支持寫操做,則爲 true;不然爲 false。
        /// </summary>
        public bool CanWrite { get { return _PipeClient.CanWrite; } }


        [Browsable(false)]
        /// <summary>
        /// 獲取一個值,該值指示當前管道是否支持查找操做。
        /// 在全部狀況下均爲 false。
        /// </summary>
        public bool CanSeek { get { return _PipeClient.CanSeek; } }


        [Browsable(false)]
        /// <summary>
        /// 獲取一個值,該值指示當前管道是否已經鏈接
        /// </summary>
        public bool IsConnected
        {
            get
            {
                if (_PipeClient == null)
                    return false;
                return _PipeClient.IsConnected;
            }
        }

        #endregion

        #region 事件
        /// <summary>
        /// 從管道中讀取數據完成
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        public delegate void PipeReadDataHandle(object sender, PipeReadDataEventArgs e);
        /// <summary>
        /// 管道寫完成/能夠進行下一次寫
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        public delegate void PipeWriteOverHandle(object sender, EventArgs e);
        public delegate void PipeConnecteHandle(object sender, EventArgs e);
        public delegate void PipeColsedHandle(object sender, EventArgs e);
        public delegate void PipeDisconnectHandle(object sender, EventArgs e);

        public delegate void PipeErrorHandle(object sender, PipeErrorEventArgs e);
        
        [Description("從管道中讀取數據完成")]
        /// <summary>
        /// 從管道中讀取數據完成
        /// </summary>
        public event PipeReadDataHandle OnPipeReadData=null;




        [Description("管道寫完成/能夠進行下一次寫")]
        /// <summary>
        /// 管道寫完成/能夠進行下一次寫
        /// </summary>
        public event PipeWriteOverHandle OnPipeWriteOver = null;


        [Description("管道已經鏈接成功")]
        /// <summary>
        /// 管道已經鏈接成功
        /// </summary>
        public event PipeConnecteHandle OnPipeConnect = null;


        [Description("管道已經鏈接成功")]
        /// <summary>
        /// 主動關閉管道鏈接
        /// </summary>
        public event PipeColsedHandle OnPipeClosed = null;


        [Description("管道被動關閉【管道服務器主動關閉鏈接】")]
        /// <summary>
        /// 管道被動關閉【管道服務器主動關閉鏈接】
        /// </summary>
        public event PipeDisconnectHandle OnPipeDisconnect = null;


        [Description("有錯誤發生")]
        public event PipeErrorHandle OnPipeError = null;
        #endregion

        #region 方法
        #region 私有方法

        private IntPtr token = IntPtr.Zero;
        private WindowsIdentity newIdentity;
        private WindowsImpersonationContext impersonatedUser;
        private bool _isLogonSuccessed = false;
        private bool LogonToRemote()
        {
            _isLogonSuccessed = false;
            try
            {
                _isLogonSuccessed = Win32Helper.LogonUser(UserName, Host, PassWord,
                    Win32Helper.LOGON32_LOGON_NEW_CREDENTIALS,
                    Win32Helper.LOGON32_PROVIDER_DEFAULT, ref token);

                newIdentity = new WindowsIdentity(token);
                impersonatedUser = newIdentity.Impersonate();


#if DEBUG

                string sMsg= string.Format("UserName:{0}   Password:{1}    Host:{2}    newIdentity={3},impersonatedUser={4}",
                    UserName, PassWord, Host, newIdentity == null ? "NULL" : "Not Null", impersonatedUser == null ? "NULL" : "Not NULL");
                MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.REMOTELOGON, sMsg));
#endif
            }
            catch (Exception e)
            {
                //OnPipeError?.Invoke(this, new PipeErrorEventArgs(PipeErrorReson.REMOTELOGON, e.Message));
                MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.REMOTELOGON, e.Message));
                return false;
            }
            return _isLogonSuccessed;
        }
        private void LogonOut()
        {
            if (_isLogonSuccessed)
            {
                if (impersonatedUser != null)
                    impersonatedUser.Undo();
                if (token != IntPtr.Zero)
                    Win32Helper.CloseHandle(token);
            }
        }

        private void PipeWriteCallback(IAsyncResult ar)
        {
            var pipe = (NamedPipeClientStream)ar.AsyncState;

            pipe.EndWrite(ar);
            pipe.Flush();
            pipe.WaitForPipeDrain();

            MyInvoke(OnPipeWriteOver, this, new EventArgs());
            //OnPipeWriteOver?.Invoke(this, new EventArgs());
            
        }
        private void ReleasePipe()
        {
            if(_PipeClient!=null)
            {
                if(_PipeClient.IsConnected)
                {
                    _PipeClient.Close();
                    _PipeClient = null;
                    MyInvoke(OnPipeClosed, this, new EventArgs());
                }
                else
                {
                    _PipeClient = null;
                    MyInvoke(OnPipeDisconnect, this, new EventArgs());
                }                
            }
            LogonOut();
        }

        private class AsyncReadState
        {
            public NamedPipeClientStream Pipe { get; set; }
            public byte[] Buffer { get; set; }

            public ManualResetEvent EventHandle { get; set; }
        }
        private void ReadThread()
        {
            while (true)
            {
                if (_PipeClient != null)
                {
                    try
                    {
                        if (_PipeClient.IsConnected)
                        {
                            AsyncReadState asyncState = new AsyncReadState()
                            {
                                Pipe = _PipeClient,
                                Buffer = new byte[4096],
                                EventHandle = new ManualResetEvent(false)
                            };

                            MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.READ, "beginread"));
                            _PipeClient.BeginRead(asyncState.Buffer, 0, 4096, ReadCallBack, asyncState);
                            asyncState.EventHandle.WaitOne();
                        }
                        else
                        {
                            break;                         
                        }
                    }
                    catch(ArgumentException e)
                    {
                        //OnPipeError?.Invoke(this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        break;
                    }
                    catch (ObjectDisposedException e)//管道已關閉。
                    {
                        //OnPipeDisconnect?.Invoke(this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        //OnPipeError?.Invoke(this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        break;
                    }
                    catch(InvalidOperationException e)//管道已斷開鏈接,正在等待鏈接,或還沒有設置句柄。
                    {
                        //OnPipeDisconnect?.Invoke(this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        MyInvoke(OnPipeDisconnect, this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        break;
                    }
                    catch (Exception e)
                    {
                        //OnPipeError?.Invoke(this, new PipeErrorEventArgs(PipeErrorReson.READ, ex.Message));
                        MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.READ, e.Message));
                        break;
                    }
                }
                else
                    break;
            }

            MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.READ, "PipeReadThreadOver"));
            ReleasePipe();

        }

        private void ReadCallBack(IAsyncResult ar)
        {
            var asyncState = (AsyncReadState)ar.AsyncState;

            int nReadLength = asyncState.Pipe.EndRead(ar);
            if (nReadLength == 0)
            {                
                MyInvoke(OnPipeDisconnect, this, new EventArgs());
                ReleasePipe();
            }
            else
            {
                MyInvoke(OnPipeReadData, this, new PipeReadDataEventArgs(asyncState.Buffer, nReadLength));
            }
            asyncState.EventHandle.Set();
        }
        #endregion

        /// <summary>
        /// 根據設置,鏈接到管道服務器
        /// </summary>
        public void Connect()
        {
            //bool bRet = false;
            if(!IsLocalHost)
            {
                if(string.IsNullOrEmpty(UserName)
                    || string.IsNullOrEmpty(Host)
                    || string.IsNullOrEmpty(PassWord))
                {
                    throw new Exception("遠程鏈接必須設置主機地址、登陸賬號、登陸密碼!");                    
                }                

                if(!LogonToRemote())
                {
                    return;
                }
            }
            else
            {
                if (string.IsNullOrEmpty(Host))
                {
                    Host = ".";
                }
            }
            try
            {
                if (string.IsNullOrEmpty(PipeName))
                {
                    throw new Exception("必須設置管道名稱!");
                }
                _PipeClient = new NamedPipeClientStream(Host,
                    PipeName,
                    PipeDirection.InOut,
                    PipeOptions.Asynchronous,
                    System.Security.Principal.TokenImpersonationLevel.Impersonation);

                _PipeClient.Connect((int)TimeOut);
                Thread t = new Thread(new ThreadStart(ReadThread));
                t.Name = "PipeRreadThread";
                t.IsBackground = true;
                t.Start();
                try
                {
                    if (_PipeClient.CanTimeout)
                    {
                        _PipeClient.WriteTimeout = (int)TimeOut;
                        _PipeClient.ReadTimeout = (int)TimeOut;
                    }
                }
                catch(InvalidOperationException)
                { }
                //OnPipeConnect?.Invoke(this, new EventArgs());
                MyInvoke(OnPipeConnect, this, new EventArgs());
            }
            catch(TimeoutException ex)
            {
                throw ex;
            }
            catch(Exception e)
            {
                
                //OnPipeError?.Invoke(this, new PipeErrorEventArgs(PipeErrorReson.CONNECT, e.Message));
                MyInvoke(OnPipeError, this, new PipeErrorEventArgs(PipeErrorReson.CONNECT, e.Message));
                return;
            }

            //return bRet;
        }

        public void Close()
        {
            ReleasePipe();
        }
        /// <summary>
        /// 將字節數組寫入管道
        /// </summary>
        /// <param name="data">要寫入的字節數組</param>
        public void Write(byte[] data)
        {
            Write(data, 0, data.Length);
        }
        /// <summary>
        /// 將字節數組寫入管道
        /// </summary>
        /// <param name="data">要寫入的字節數組</param>
        /// <param name="offset">數組起始位置</param>
        /// <param name="count">寫入的字節數</param>
        public void Write(byte[] data,int offset,int count)
        {
            if(_PipeClient==null)
            {
                throw new Exception("請在寫數據以前鏈接管道");
            }
            if(!_PipeClient.IsConnected)
            {
                throw new Exception("請在寫數據以前鏈接管道");
            }
            if(!_PipeClient.CanWrite)
            {
                throw new Exception("該管道不支持寫操做");
            }
            if(count>4096)
            {
                throw new Exception("要求寫入數據總長度不大於4096");
            }
            if(offset<0||offset>data.Length-1||count<0)
            {
                throw new Exception("參數錯誤");
            }
            if (offset +count > data.Length)
            {
                throw new Exception("參數錯誤");
            }
            if (count == 0)
            {
                //OnPipeWriteOver?.Invoke(this, new EventArgs());
                MyInvoke(OnPipeWriteOver, this, new EventArgs());
                return;
            }
            _PipeClient.BeginWrite(data, offset, count, PipeWriteCallback, _PipeClient);
        }
        #endregion

        protected virtual void MyInvoke(Delegate del, object sender, object args)
        {
            if (del != null)
            {
                if (del.Target is System.ComponentModel.ISynchronizeInvoke)
                {
                    //當前委託的實例。如一個Form實例
                    System.ComponentModel.ISynchronizeInvoke aSynch = del.Target as System.ComponentModel.ISynchronizeInvoke;
                    if (aSynch.InvokeRequired)
                    {
                        //此類對象綁定到特定線程並非線程安全。 若是正在從另外一個線程調用的方法,則必須使用Invoke封送到正確的線程調用的方法。
                        //如在非UI線程中修改UI控件
                        object[] objs = new object[2] { sender, args };
                        aSynch.BeginInvoke(del, objs);
                    }
                    else
                    {   
                        //不涉及線程安全問題時,直接調用
                        del.DynamicInvoke(sender, args);
                    }
                }
            }
        }
    }

    public enum PipeErrorReson
    {
        /// <summary>
        /// 登陸遠程主機
        /// </summary>
        REMOTELOGON,
        /// <summary>
        /// 鏈接到管道服務器
        /// </summary>
        CONNECT,
        /// <summary>
        /// 寫數據
        /// </summary>
        WRITE,
        /// <summary>
        /// 讀數據
        /// </summary>
        READ
    }

    public class PipeReadDataEventArgs: EventArgs
    {
        byte[] _data;
        int _nDataLength=0;
        public PipeReadDataEventArgs(byte[] buf,int nLen)
        {
            _data = new byte[nLen];
            Array.Copy(buf, _data, nLen);
            _nDataLength = nLen;
        }

        /// <summary>
        /// 返回從管道中讀取的字節數組
        /// </summary>
        public byte[] Data { get { return _data; } }

        /// <summary>
        /// 返回管道中讀取的字節數組長度
        /// </summary>
        public int DataLength { get { return _nDataLength; } }
    }

    public class PipeErrorEventArgs:EventArgs
    {
        PipeErrorReson _pipeErrorReson ;
        string _msg;
        public PipeErrorEventArgs(PipeErrorReson pipeErrorReson,string msg)
        {
            _pipeErrorReson = pipeErrorReson;
            _msg = msg;
        }
        
        /// <summary>
        /// 返回錯誤引發的緣由【由哪一個動做致使的錯誤】
        /// </summary>
        public PipeErrorReson ErrorReson { get { return _pipeErrorReson; } }

        /// <summary>
        /// 返回錯誤信息描述
        /// </summary>
        public string Message { get { return _msg; } }
    }

}

 測試界面:測試

相關文章
相關標籤/搜索