【转】C#下带进度条的FTP上传类

该类转自:http://blog.csdn.net/byshome/article/details/5801991,稍作修改

using System;
using System.Net;
using System.Net.Sockets;
using System.Collections.Generic;
using System.Text;
using System.IO;
using System.Globalization;
using System.Text.RegularExpressions;

namespace FTP
{
    public class FTPHelper : IDisposable
    {
        #region 声明事件
        /// <summary>
        /// 正在下载文件
        /// </summary>
        public event FTPSendEventHandler FileDownloading;
        private delegate void OnFileDownloadingDelegate(FTPConnect ftpConnect, long lTotalBytes, long lBytesTransfered);
        /// <summary>
        /// 正在下载文件封装模式
        /// </summary>
        private void OnFileDownloading(FTPConnect ftpConnect, long lTotalBytes, long lBytesTransfered)
        {
            if (this.FileDownloading != null)
            {
                if (lBytesTransfered > lTotalBytes)
                    lBytesTransfered = lTotalBytes;
                System.ComponentModel.ISynchronizeInvoke aSynch = this.FileDownloading.Target as System.ComponentModel.ISynchronizeInvoke;
                if (aSynch != null && aSynch.InvokeRequired)
                    aSynch.Invoke(new OnFileDownloadingDelegate(OnFileDownloading), new object[] { ftpConnect, lTotalBytes, lBytesTransfered });
                else
                    this.FileDownloading(ftpConnect, new FTPSendEventArgs(lTotalBytes, lBytesTransfered));
            }
        }
        /// <summary>
        /// 文件下载完成
        /// </summary>
        public event EventHandler FileDownloadCompleted;
        private delegate void OnFileDownloadCompletedDelegate(FTPConnect ftpConnect);
        /// <summary>
        /// 文件下载完成封装模式
        /// </summary>
        private void OnFileDownloadCompleted(FTPConnect ftpConnect)
        {
            if (this.FileDownloadCompleted != null)
            {
                System.ComponentModel.ISynchronizeInvoke aSynch = this.FileDownloadCompleted.Target as System.ComponentModel.ISynchronizeInvoke;
                if (aSynch != null && aSynch.InvokeRequired)
                    aSynch.Invoke(new OnFileDownloadCompletedDelegate(OnFileDownloadCompleted), new object[] { ftpConnect });
                else
                    this.FileDownloadCompleted(ftpConnect, new EventArgs());
            }
        }
        /// <summary>
        /// 取消正在下载的文件
        /// </summary>
        public event EventHandler FileDownloadCanceled;
        private delegate void OnFileDownloadCanceledDelegate(FTPConnect ftpConnect);
        /// <summary>
        /// 取消正在下载的文件封装模式
        /// </summary>
        private void OnFileDownloadCanceled(FTPConnect ftpConnect)
        {
            if (this.FileDownloadCanceled != null)
            {
                System.ComponentModel.ISynchronizeInvoke aSynch = this.FileDownloadCanceled.Target as System.ComponentModel.ISynchronizeInvoke;
                if (aSynch != null && aSynch.InvokeRequired)
                    aSynch.Invoke(new OnFileDownloadCanceledDelegate(OnFileDownloadCanceled), new object[] { ftpConnect });
                else
                    this.FileDownloadCanceled(ftpConnect, new EventArgs());
            }
        }
        /// <summary>
        /// 正在上传文件
        /// </summary>
        public event FTPSendEventHandler FileUploading;
        private delegate void OnFileUploadingDelegate(FTPConnect ftpConnect, long lTotalBytes, long lBytesTransfered);
        /// <summary>
        /// 正在下载事件封装模式
        /// </summary>
        /// <param name="lTotalBytes"></param>
        /// <param name="lBytesTransfered"></param>
        private void OnFileUploading(FTPConnect ftpConnect, long lTotalBytes, long lBytesTransfered)
        {
            if (this.FileUploading != null)
            {
                if (lBytesTransfered > lTotalBytes)
                    lBytesTransfered = lTotalBytes;
                System.ComponentModel.ISynchronizeInvoke aSynch = this.FileUploading.Target as System.ComponentModel.ISynchronizeInvoke;
                if (aSynch != null && aSynch.InvokeRequired)
                    aSynch.Invoke(new OnFileUploadingDelegate(OnFileUploading), new object[] { ftpConnect, lTotalBytes, lBytesTransfered });
                else
                    this.FileUploading(ftpConnect, new FTPSendEventArgs(lTotalBytes, lBytesTransfered));
            }
        }
        /// <summary>
        /// 文件上传完成
        /// </summary>
        public event EventHandler FileUploadCompleted;
        private delegate void OnFileUploadCompletedDelegate(FTPConnect ftpConnect);
        private void OnFileUploadCompleted(FTPConnect ftpConnect)
        {
            if (this.FileUploadCompleted != null)
            {
                System.ComponentModel.ISynchronizeInvoke aSynch = this.FileUploadCompleted.Target as System.ComponentModel.ISynchronizeInvoke;
                if (aSynch != null && aSynch.InvokeRequired)
                    aSynch.Invoke(new OnFileUploadCompletedDelegate(OnFileUploadCompleted), new object[] { ftpConnect });
                else
                    this.FileUploadCompleted(ftpConnect, new EventArgs());
            }
        }
        /// <summary>
        /// 取消了上传文件
        /// </summary>
        public event EventHandler FileUploadCanceled;
        private delegate void OnFileUploadCanceledDelegate(FTPConnect ftpConnect);
        private void OnFileUploadCanceled(FTPConnect ftpConnect)
        {
            if (this.FileUploadCanceled != null)
            {
                System.ComponentModel.ISynchronizeInvoke aSynch = this.FileUploadCanceled.Target as System.ComponentModel.ISynchronizeInvoke;
                if (aSynch != null && aSynch.InvokeRequired)
                    aSynch.Invoke(new OnFileUploadCanceledDelegate(OnFileUploadCanceled), new object[] { ftpConnect });
                else
                    this.FileUploadCanceled(ftpConnect, new EventArgs());
            }
        }
        /// <summary>
        /// 传输过程发生错误事件
        /// </summary>
        public event FTPErrorEventHandler FtpError;
        private delegate void OnFtpErrorDelegate(FTPConnect ftpConnect, Exception error);
        public void OnFtpError(FTPConnect ftpConnect, Exception error)
        {
            if (this.FtpError != null)
            {
                System.ComponentModel.ISynchronizeInvoke aSynch = this.FtpError.Target as System.ComponentModel.ISynchronizeInvoke;
                if (aSynch != null && aSynch.InvokeRequired)
                    aSynch.Invoke(new OnFtpErrorDelegate(OnFtpError), new object[] { ftpConnect, error });
                else
                    this.FtpError(ftpConnect, new FTPErrorEventArgs(error));
            }
        }
        #endregion
        #region FTPUrl结构
        public class FTPUrl
        {
            private string m_Url = String.Empty;
            private string m_RemoteHost = String.Empty;
            private IPAddress m_RemoteHostIP = IPAddress.None;
            private int m_RemotePort = 21;
            private string m_UserName = String.Empty;
            private string m_Password = String.Empty;
            private string m_SubUrl = String.Empty;
            public FTPUrl()
            { }
            public FTPUrl(string url)
            {
                this.Url = url;
            }
            /// <summary>
            /// FTP的全地址
            /// </summary>
            public string Url
            {
                get { return this.m_Url; }
                set
                {
                    if (value.IndexOf("@") < 0)
                        throw (new Exception("FTP地址路径不合法!格式应为ftp://用户名:密码@地址[:端口]/"));
                    string strSubUrl = "";
                    string strRemoteHostAndPort = value.Substring(value.IndexOf("@") + 1);
                    if (strRemoteHostAndPort.IndexOf("/") > 0)
                    {
                        strSubUrl = strRemoteHostAndPort.Substring(strRemoteHostAndPort.IndexOf("/"));
                        strRemoteHostAndPort = strRemoteHostAndPort.Substring(0, strRemoteHostAndPort.IndexOf("/"));
                    }
                    string strRemoteHost = strRemoteHostAndPort;
                    int iRemotePort = 21;
                    if (strRemoteHostAndPort.IndexOf(":") > 0)
                    {
                        strRemoteHost = strRemoteHostAndPort.Substring(0, strRemoteHostAndPort.IndexOf(":"));
                        string strRemotePort = strRemoteHostAndPort.Substring(strRemoteHostAndPort.IndexOf(":") + 1);
                        if (!int.TryParse(strRemotePort, out iRemotePort))
                            iRemotePort = 21;
                    }
                    string strUserNameAndPassword = value.Substring(0, value.IndexOf("@")).Trim();
                    if (strUserNameAndPassword.ToLower().StartsWith("ftp://"))
                        strUserNameAndPassword = strUserNameAndPassword.Substring(6).Trim();
                    if (strUserNameAndPassword == string.Empty || strUserNameAndPassword.IndexOf(":") < 0)
                        throw (new Exception("FTP地址路径不合法!格式应为ftp://用户名:密码@地址[:端口]/"));
                    string strUserName = strUserNameAndPassword.Substring(0, strUserNameAndPassword.IndexOf(":"));
                    string strPassword = strUserNameAndPassword.Substring(strUserNameAndPassword.IndexOf(":") + 1);

                    IPAddress[] ips = Dns.GetHostAddresses(strRemoteHost);
                    if (ips.Length == 0)
                        throw (new Exception("FTP地址路径中主机地址无效!"));
                    strSubUrl = strSubUrl.Replace("//", "/");
                    this.m_RemoteHostIP = ips[0];
                    this.m_RemoteHost = strRemoteHost;
                    this.m_RemotePort = iRemotePort;
                    this.m_UserName = strUserName;
                    this.m_Password = strPassword;
                    this.m_SubUrl = strSubUrl;
                    this.m_Url = value;
                }
            }
            /// <summary>
            /// 主机地址
            /// </summary>
            public string RemoteHost
            {
                get { return this.m_RemoteHost; }
            }
            /// <summary>
            /// 主机IP
            /// </summary>
            public IPAddress RemoteHostIP
            {
                get { return this.m_RemoteHostIP; }
            }
            /// <summary>
            /// 主机端口
            /// </summary>
            public int RemotePort
            {
                get { return this.m_RemotePort; }
            }
            public string UserName
            {
                get { return this.m_UserName; }
            }
            public string Password
            {
                get { return this.m_Password; }
            }
            public string SubUrl
            {
                get { return this.m_SubUrl; }
            }
        }
        #endregion
        #region 传输模式
        /// <summary>
        /// 传输模式:二进制类型、ASCII类型
        /// </summary>
        public enum TransferType
        {
            /// <summary>
            /// Binary
            /// </summary>
            Binary,
            /// <summary>
            /// ASCII
            /// </summary>
            ASCII
        };
        #endregion
        #region 存贮FTP的连接结构类
        public class FTPConnect
        {
            #region 私有字段
            /// <summary>
            /// 数据传送套接字列表
            /// </summary>
            private List<Socket> m_DataSocketList;
            private string m_ID;
            /// <summary>
            /// 唯一ID
            /// </summary>
            public string ID
            {
                get { return this.m_ID; }
            }
            private object m_Tag = null;
            /// <summary>
            /// 扩展标记
            /// </summary>
            public object Tag
            {
                get { return this.m_Tag; }
                set { this.m_Tag = value; }
            }
            private bool m_DataTransmitting = false;
            /// <summary>
            /// 数据正在传输 标记
            /// </summary>
            public bool DataTransmitting
            {
                get { return this.m_DataTransmitting; }
                set { this.m_DataTransmitting = value; }
            }
            private Socket m_SocketControl;
            /// <summary>
            /// FTPUrl
            /// </summary>
            private FTPUrl m_FTPUrl;
            /// <summary>
            /// 是否已经连接
            /// </summary>
            private bool m_IsConnected;
            private Encoding m_EncodeType = Encoding.Default;
            /// <summary>
            /// 编码方式
            /// </summary>
            public Encoding EncodeType
            {
                get { return this.m_EncodeType; }
                set { this.m_EncodeType = value; }
            }
            /// <summary>
            /// 接收和发送数据的缓冲区
            /// </summary>
            private static int BLOCK_SIZE = 512;
            /// <summary>
            /// 缓冲区大小
            /// </summary>
            private Byte[] m_Buffer;
            public Byte[] Buffer
            {
                get { return this.m_Buffer; }
                set { this.m_Buffer = value; }
            }
            private string m_Message;
            /// <summary>
            /// 当前的消息
            /// </summary>
            public string Message
            {
                get { return this.m_Message; }
                set { this.m_Message = value; }
            }
            private string m_ReplyString;
            /// <summary>
            /// 应答字符串
            /// </summary>
            public string ReplyString
            {
                get { return this.m_ReplyString; }
                set { this.m_ReplyString = value; }
            }
            private int m_ReplyCode;
            /// <summary>
            /// 应答代码
            /// </summary>
            public int ReplyCode
            {
                get { return this.m_ReplyCode; }
                set { this.m_ReplyCode = value; }
            }
            /// <summary>
            /// 传输模式
            /// </summary>
            private TransferType trType;
            #endregion
            public FTPConnect()
            {
                this.m_ID = System.Guid.NewGuid().ToString();
                this.m_DataSocketList = new List<Socket>();
                this.m_Buffer = new Byte[BLOCK_SIZE];
            }
            public FTPConnect(FTPUrl ftpUrl)
            {
                this.m_ID = System.Guid.NewGuid().ToString();
                this.m_DataSocketList = new List<Socket>();
                this.m_Buffer = new Byte[BLOCK_SIZE];
                this.FTPUrl = ftpUrl;
            }
            public FTPConnect(FTPUrl ftpUrl, string ftpId)
            {
                if (String.IsNullOrEmpty(ftpId))
                    ftpId = System.Guid.NewGuid().ToString();
                this.m_ID = ftpId;
                this.m_DataSocketList = new List<Socket>();
                this.m_Buffer = new Byte[BLOCK_SIZE];
                this.FTPUrl = ftpUrl;
            }
            #region 公共字段
            /// <summary>
            /// 套接字连接
            /// </summary>
            public Socket SocketControl
            {
                get { return this.m_SocketControl; }
                set { this.m_SocketControl = value; }
            }
            /// <summary>
            /// 对应的URL
            /// </summary>
            public FTPUrl FTPUrl
            {
                get { return this.m_FTPUrl; }
                set { this.m_FTPUrl = value; }
            }
            /// <summary>
            /// 是否已经连接
            /// </summary>
            public bool IsConnected
            {
                get { return this.m_IsConnected; }
                set { this.m_IsConnected = value; }
            }
            #endregion
            #region 公共方法
            #region 取消传送数据
            public void CancelDataTransmit()
            {
                this.m_DataTransmitting = false;
            }
            #endregion
            #region 发送命令
            /// <summary>
            /// 发送命令并获取应答码和最后一行应答字符串
            /// </summary>
            /// <param name="strCommand">命令</param>
            public void SendCommand(string strCommand)
            {
                if (this.m_SocketControl == null)
                    throw (new Exception("请先连接服务器再进行操作!"));
                Byte[] cmdBytes = m_EncodeType.GetBytes((strCommand + "\r\n").ToCharArray());
                this.m_SocketControl.Send(cmdBytes, cmdBytes.Length, 0);
                this.ReadReply();
            }
            #endregion
            #region 读取最后一行的消息
            /// <summary>
            /// 读取Socket返回的所有字符串
            /// </summary>
            /// <returns>包含应答码的字符串行</returns>
            private string ReadLine()
            {
                if (this.m_SocketControl == null)
                    throw (new Exception("请先连接服务器再进行操作!"));
                while (true)
                {
                    int iBytes = this.m_SocketControl.Receive(m_Buffer, m_Buffer.Length, 0);
                    m_Message += m_EncodeType.GetString(m_Buffer, 0, iBytes);
                    if (iBytes < m_Buffer.Length)
                    {
                        break;
                    }
                }
                char[] seperator = { '\n' };
                string[] mess = m_Message.Split(seperator);
                if (m_Message.Length > 2)
                {
                    m_Message = mess[mess.Length - 2];
                    //seperator[0]是10,换行符是由13和0组成的,分隔后10后面虽没有字符串,
                    //但也会分配为空字符串给后面(也是最后一个)字符串数组,
                    //所以最后一个mess是没用的空字符串
                    //但为什么不直接取mess[0],因为只有最后一行字符串应答码与信息之间有空格
                }
                else
                {
                    m_Message = mess[0];
                }
                if (!m_Message.Substring(3, 1).Equals(" "))//返回字符串正确的是以应答码(如220开头,后面接一空格,再接问候字符串)
                {
                    return this.ReadLine();
                }
                return m_Message;
            }
            #endregion
            #region 读取应答代码
            /// <summary>
            /// 将一行应答字符串记录在strReply和strMsg
            /// 应答码记录在iReplyCode
            /// </summary>
            public void ReadReply()
            {
                this.m_Message = "";
                this.m_ReplyString = this.ReadLine();
                this.m_ReplyCode = Int32.Parse(m_ReplyString.Substring(0, 3));
            }
            #endregion
            #region 断开连接
            /// <summary>
            /// 关闭连接
            /// </summary>
            public void DisConnect()
            {
                this.m_DataTransmitting = false;
                while (this.m_DataSocketList.Count > 0)
                {
                    Socket socket = this.m_DataSocketList[0];
                    if (socket != null && socket.Connected)
                        socket.Close();
                    this.m_DataSocketList.RemoveAt(0);
                }
                if (this.m_IsConnected && this.m_SocketControl != null)
                    this.SendCommand("QUIT");
                this.CloseSocketConnect();
                this.m_Buffer = null;
            }
            /// <summary>
            /// 关闭socket连接(用于登录以前)
            /// </summary>
            private void CloseSocketConnect()
            {
                if (this.m_SocketControl != null && this.m_SocketControl.Connected)
                {
                    this.m_SocketControl.Close();
                    this.m_SocketControl = null;
                }
                this.m_IsConnected = false;
            }
            #endregion
            #region 连接服务器
            public void Connect()
            {
                this.DisConnect();//先断开现有连接
                this.m_Buffer = new byte[BLOCK_SIZE];
                this.m_SocketControl = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
                IPEndPoint ep = new IPEndPoint(this.m_FTPUrl.RemoteHostIP, this.m_FTPUrl.RemotePort);
                try
                {
                    this.m_SocketControl.Connect(ep);
                }
                catch (Exception)
                {
                    throw new IOException(String.Format("无法连接到远程服务器{0}!", this.m_FTPUrl.RemoteHost));
                }
                // 获取应答码
                this.ReadReply();
                if (m_ReplyCode != 220)
                {
                    this.DisConnect();
                    throw new IOException(m_ReplyString.Substring(4));
                }
                // 登陆
                this.SendCommand("USER " + this.m_FTPUrl.UserName);
                if (!(m_ReplyCode == 331 || m_ReplyCode == 230))
                {
                    this.CloseSocketConnect();//关闭连接
                    throw new IOException(m_ReplyString.Substring(4));
                }
                if (m_ReplyCode != 230)
                {
                    this.SendCommand("PASS " + this.m_FTPUrl.Password);
                    if (!(m_ReplyCode == 230 || m_ReplyCode == 202))
                    {
                        this.CloseSocketConnect();//关闭连接
                        throw new IOException(m_ReplyString.Substring(4));
                    }
                }
                this.m_IsConnected = true;
            }
            #endregion
            #region 改变目录
            /// <summary>
            /// 改变目录
            /// </summary>
            /// <param name="strDirName">新的工作目录名</param>
            public void ChangeDir(string dirName)
            {
                if (!this.m_IsConnected)
                    throw (new Exception("请先连接服务器然后再进行CWD操作!"));
                if (dirName.Equals(".") || dirName.Equals(""))
                    return;
                this.SendCommand("CWD " + dirName);
                if (m_ReplyCode != 250)
                    throw new IOException(m_ReplyString.Substring(4));
            }
            #endregion
            #region 传输模式
            /// <summary>
            /// 设置传输模式
            /// </summary>
            /// <param name="ttType">传输模式</param>
            public void SetTransferType(TransferType ttType)
            {
                if (ttType == TransferType.Binary)
                {
                    this.SendCommand("TYPE I");//binary类型传输
                }
                else
                {
                    this.SendCommand("TYPE A");//ASCII类型传输
                }
                if (m_ReplyCode != 200)
                {
                    throw new IOException(m_ReplyString.Substring(4));
                }
                else
                {
                    trType = ttType;
                }
            }
            /// <summary>
            /// 获得传输模式
            /// </summary>
            /// <returns>传输模式</returns>
            public TransferType GetTransferType()
            {
                return trType;
            }
            #endregion
            #region 建立进行数据连接的socket
            /// <summary>
            /// 建立进行数据连接的socket
            /// </summary>
            /// <returns>数据连接socket</returns>
            public Socket CreateDataSocket()
            {
                this.SendCommand("PASV");
                if (this.m_ReplyCode != 227)
                    throw new IOException(this.m_ReplyString.Substring(4));
                int index1 = this.m_ReplyString.IndexOf('(');
                int index2 = this.m_ReplyString.IndexOf(')');
                string ipData = this.m_ReplyString.Substring(index1 + 1, index2 - index1 - 1);
                int[] parts = new int[6];
                int len = ipData.Length;
                int partCount = 0;
                string buf = "";
                for (int i = 0; i < len && partCount <= 6; i++)
                {
                    char ch = Char.Parse(ipData.Substring(i, 1));
                    if (Char.IsDigit(ch))
                        buf += ch;
                    else if (ch != ',')
                    {
                        throw new IOException("Malformed PASV Reply: " + this.m_ReplyString);
                    }
                    if (ch == ',' || i + 1 == len)
                    {
                        try
                        {
                            parts[partCount++] = Int32.Parse(buf);
                            buf = "";
                        }
                        catch (Exception)
                        {
                            throw new IOException("Malformed PASV Reply: " + this.m_ReplyString);
                        }
                    }
                }
                string ipAddress = parts[0] + "." + parts[1] + "." + parts[2] + "." + parts[3];
                int port = (parts[4] << 8) + parts[5];
                Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
                IPEndPoint ep = new IPEndPoint(IPAddress.Parse(ipAddress), port);
                try
                {
                    socket.Connect(ep);
                }
                catch (Exception)
                {
                    throw new IOException(String.Format("无法连接到远程服务器{0}:{1}!", ipAddress, port));
                }
                this.m_DataSocketList.Add(socket);
                return socket;
            }
            #endregion
            #endregion
        }
        #endregion
        #region 变量区
        /// <summary>
        /// 进行控制连接的socket
        /// </summary>
        private List<FTPConnect> m_FTPConnectList;
        private object m_Tag;
        /// <summary>
        /// 标记
        /// </summary>
        public object Tag
        {
            get { return this.m_Tag; }
            set { this.m_Tag = value; }
        }
        #endregion

        #region 实例化
        public FTPHelper()
        {
            this.m_FTPConnectList = new List<FTPConnect>();
        }
        #endregion
        #region Dispose
        public void Dispose()
        {
            while (this.m_FTPConnectList.Count > 0)
            {
                FTPConnect ftpConnect = this.m_FTPConnectList[0];
                ftpConnect.DisConnect();
                this.m_FTPConnectList.Remove(ftpConnect);
            }
        }
        #endregion
        #region 得到文件大小
        /// <summary>
        /// 得到文件大小
        /// </summary>
        /// <param name="fileUrl">目标文件,包含用户名与密码。如:ftp://username:password@127.0.0.1/1.txt</param>
        /// <param name="UserName">用户名</param>
        /// <param name="Password">密码</param>
        /// <returns></returns>
        public long GetFileSize(string fileUrl)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            try
            {
                ftpUrl = new FTPUrl(fileUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                string strDirName = ftpUrl.SubUrl;
                if (strDirName.IndexOf("/") >= 0)
                    strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirName);
                ftpConnect.SendCommand("SIZE " + Path.GetFileName(ftpUrl.SubUrl));
                long lFileSize = 0;
                if (ftpConnect.ReplyCode == 213)
                    lFileSize = Int64.Parse(ftpConnect.ReplyString.Substring(4));
                else
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                return lFileSize;
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        #endregion
        #region 得到文件大小
        /// <summary>
        /// 得到文件大小
        /// </summary>
        /// <param name="fileUrl">目标文件,包含用户名与密码。如:ftp://username:password@127.0.0.1/1.txt</param>
        /// <param name="UserName">用户名</param>
        /// <param name="Password">密码</param>
        /// <returns></returns>
        public DateTime GetDateTimestamp(string fileUrl)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            try
            {
                ftpUrl = new FTPUrl(fileUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                string strDirName = ftpUrl.SubUrl;
                if (strDirName.IndexOf("/") >= 0)
                    strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirName);
                ftpConnect.SendCommand("MDTM " + Path.GetFileName(ftpUrl.SubUrl));
                if (ftpConnect.ReplyCode == 213)
                {
                    string strDateTime = ftpConnect.ReplyString.Substring(4);
                    int iYear = 0, iMonth = 0, iDay = 0, iHour = 0, iMinute = 0, iSecond = 0;
                    if (strDateTime.Length >= 4)
                        int.TryParse(strDateTime.Substring(0, 4), out iYear);
                    if (strDateTime.Length >= 6)
                        int.TryParse(strDateTime.Substring(4, 2), out iMonth);
                    if (strDateTime.Length >= 8)
                        int.TryParse(strDateTime.Substring(6, 2), out iDay);
                    if (strDateTime.Length >= 10)
                        int.TryParse(strDateTime.Substring(8, 2), out iHour);
                    if (strDateTime.Length >= 12)
                        int.TryParse(strDateTime.Substring(10, 2), out iMinute);
                    if (strDateTime.Length >= 14)
                        int.TryParse(strDateTime.Substring(12, 2), out iSecond);
                    return new DateTime(iYear, iMonth, iDay, iHour, iMinute, iSecond).ToLocalTime();
                }
                else
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        #endregion
        #region 删除指定的文件
        /// <summary>
        /// 删除指定的文件
        /// </summary>
        /// <param name="fileUrl">待删除文件名</param>
        public void DeleteFile(string fileUrl)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            try
            {
                ftpUrl = new FTPUrl(fileUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                string strDirName = ftpUrl.SubUrl;
                if (strDirName.IndexOf("/") >= 0)
                    strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirName);
                ftpConnect.SendCommand("DELE " + Path.GetFileName(ftpUrl.SubUrl));
                if (ftpConnect.ReplyCode != 250)
                    throw (new Exception(ftpConnect.ReplyString.Substring(4)));
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        #endregion
        #region 重命名
        /// <summary>
        /// 重命名(如果新文件名与已有文件重名,将覆盖已有文件)
        /// </summary>
        /// <param name="strOldFileName">旧文件名</param>
        /// <param name="strNewFileName">新文件名</param>
        public void Rename(string originalUrl, string newName)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            try
            {
                ftpUrl = new FTPUrl(originalUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                string strDirName = ftpUrl.SubUrl;
                if (strDirName.IndexOf("/") >= 0)
                    strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirName);
                ftpConnect.SendCommand("RNFR " + Path.GetFileName(ftpUrl.SubUrl));
                if (ftpConnect.ReplyCode != 350)
                    throw (new Exception(ftpConnect.ReplyString.Substring(4)));
                ftpConnect.SendCommand("RNTO " + newName);
                if (ftpConnect.ReplyCode != 250)
                    throw (new Exception(ftpConnect.ReplyString.Substring(4)));
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        #endregion
        #region 创建文件夹
        public void MakeDirectory(string dirUrl)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            int iCharIndex = 0;
            string strDirUrl, strDirUrlTemp;
            try
            {
                ftpUrl = new FTPUrl(dirUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                strDirUrl = ftpUrl.SubUrl;
                while (true && strDirUrl.Trim() != "")
                {
                    iCharIndex = strDirUrl.IndexOf("/", iCharIndex) + 1;
                    if (iCharIndex == 0)
                        strDirUrlTemp = strDirUrl;
                    else
                        strDirUrlTemp = strDirUrl.Substring(0, iCharIndex);
                    if (strDirUrlTemp == "")
                        continue;
                    ftpConnect.SendCommand("MKD " + strDirUrlTemp);
                    if (ftpConnect.ReplyCode != 257 && ftpConnect.ReplyCode != 550)
                        throw new IOException(ftpConnect.ReplyString.Substring(4));
                    if (strDirUrlTemp == strDirUrl)
                        break;
                }
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        #endregion
        #region 删除目录
        /// <summary>
        /// 删除目录
        /// </summary>
        /// <param name="dirUrl">目录名</param>
        public void RemoveDirectory(string dirUrl)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            try
            {
                ftpUrl = new FTPUrl(dirUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                string strDirPath = ftpUrl.SubUrl;
                string strDirName = ftpUrl.SubUrl;
                if (strDirPath.IndexOf("/") >= 0)
                {
                    strDirPath = strDirPath.Substring(0, strDirPath.LastIndexOf("/"));
                    strDirName = strDirPath.Substring(strDirPath.LastIndexOf("/") + 1);
                }
                ftpConnect.ChangeDir(strDirPath);
                ftpConnect.SendCommand("RMD " + strDirName);
                if (ftpConnect.ReplyCode != 250)
                    throw (new Exception(ftpConnect.ReplyString.Substring(4)));
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        #endregion
        #region 得到简单的文件列表
        public string[] ListDirectoryFile(string listUrl)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            try
            {
                ftpUrl = new FTPUrl(listUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                string strDirPath = ftpUrl.SubUrl;
                if (strDirPath.IndexOf("/") >= 0)
                    strDirPath = strDirPath.Substring(0, strDirPath.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirPath);
                socketData = ftpConnect.CreateDataSocket();
                ftpConnect.SendCommand("NLST");
                if (ftpConnect.ReplyCode != 150 && ftpConnect.ReplyCode != 125)
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                //获得结果
                ftpConnect.Message = "";
                while (true)
                {
                    int iBytes = socketData.Receive(ftpConnect.Buffer, ftpConnect.Buffer.Length, 0);
                    ftpConnect.Message += ftpConnect.EncodeType.GetString(ftpConnect.Buffer, 0, iBytes);
                    if (iBytes < ftpConnect.Buffer.Length)
                        break;
                }
                ftpConnect.Message = ftpConnect.Message.Replace('\r', ' ');
                char[] seperator = { '\n' };
                string[] strsFileList = ftpConnect.Message.Split(seperator);
                socketData.Close();//数据socket关闭时也会有返回码
                socketData = null;
                if (ftpConnect.ReplyCode != 226)
                {
                    ftpConnect.ReadReply();
                    if (ftpConnect.ReplyCode != 226)
                        throw new IOException(ftpConnect.ReplyString.Substring(4));
                }
                return strsFileList;
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                socketData = null;
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        //获取当前目录下的目录
        public List<string> ListDirectory(string listUrl)
        {
            List<string> strsFileList = new List<string>();
            List<FileStruct> files = ListDirectoryDetails(listUrl);
            foreach (FileStruct fs in files)
            {
                if (fs.IsDirectory)
                {
                    strsFileList.Add(fs.Name);
                }
            }
            return strsFileList;
        }
        #endregion
        #region 得到详细的文件列表
        public List<FileStruct> ListDirectoryDetails(string listUrl)
        {
            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            try
            {
                ftpUrl = new FTPUrl(listUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                string strDirPath = ftpUrl.SubUrl;
                if (strDirPath.IndexOf("/") >= 0)
                    strDirPath = strDirPath.Substring(0, strDirPath.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirPath);
                socketData = ftpConnect.CreateDataSocket();
                ftpConnect.SendCommand("LIST");
                if (ftpConnect.ReplyCode != 150 && ftpConnect.ReplyCode != 125)
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                //获得结果
                ftpConnect.Message = "";
                while (true)
                {
                    int iBytes = socketData.Receive(ftpConnect.Buffer, ftpConnect.Buffer.Length, 0);
                    ftpConnect.Message += ftpConnect.EncodeType.GetString(ftpConnect.Buffer, 0, iBytes);
                    if (iBytes < ftpConnect.Buffer.Length)
                        break;
                }
                if (ftpConnect.Message.StartsWith("t"))
                    ftpConnect.Message = ftpConnect.Message.Substring(ftpConnect.Message.IndexOf("\r\n") + 2);
                return this.GetList(ftpConnect.Message);
            }
            catch (Exception ex)
            {
                throw (ex);
            }
            finally
            {
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                socketData = null;
                if (ftpConnect != null)
                    ftpConnect.DisConnect();
            }
        }
        #region 用于得到文件列表的方法
        /// <summary>
        /// 文件列表类型枚举
        /// </summary>
        private enum FileListStyle
        {
            /// <summary>
            /// Unix系统类型
            /// </summary>
            UnixStyle,
            /// <summary>
            /// Windows系统类型
            /// </summary>
            WindowsStyle,
            /// <summary>
            /// 未知类型
            /// </summary>
            Unknown
        }
        /// <summary>
        /// 列出FTP服务器上面当前目录的所有文件
        /// </summary>
        /// <param name="listUrl">查看目标目录</param>
        /// <returns>返回文件信息列表</returns>
        public List<FileStruct> ListFiles(string listUrl)
        {
            List<FileStruct> listFile = null;
            try
            {
                List<FileStruct> listAll = this.ListDirectoryDetails(listUrl);
                listFile = new List<FileStruct>();
                foreach (FileStruct file in listAll)
                {
                    if (!file.IsDirectory)
                    {
                        listFile.Add(file);
                    }
                }
            }
            catch (Exception ex)
            {
                throw ex;
            }
            return listFile;
        }
        /// <summary>
        /// 列出FTP服务器上面当前目录的所有的目录
        /// </summary>
        /// <param name="listUrl">查看目标目录</param>
        /// <returns>返回目录信息列表</returns>
        public List<FileStruct> ListDirectories(string listUrl)
        {
            List<FileStruct> listAll = this.ListDirectoryDetails(listUrl);
            List<FileStruct> listDirectory = new List<FileStruct>();
            foreach (FileStruct file in listAll)
            {
                if (file.IsDirectory)
                {
                    listDirectory.Add(file);
                }
            }
            return listDirectory;
        }
        /// <summary>
        /// 获得文件和目录列表
        /// </summary>
        /// <param name="datastring">FTP返回的列表字符信息</param>
        private List<FileStruct> GetList(string datastring)
        {
            List<FileStruct> myListArray = new List<FileStruct>();
            string[] dataRecords = datastring.Split('\n');
            FileListStyle _directoryListStyle = GuessFileListStyle(dataRecords);
            foreach (string s in dataRecords)
            {
                if (_directoryListStyle != FileListStyle.Unknown && s != "")
                {
                    FileStruct f = new FileStruct();
                    f.Name = "..";
                    switch (_directoryListStyle)
                    {
                        case FileListStyle.UnixStyle:
                            f = ParseFileStructFromUnixStyleRecord(s);
                            break;
                        case FileListStyle.WindowsStyle:
                            f = ParseFileStructFromWindowsStyleRecord(s);
                            break;
                    }
                    if (!(f.Name == "." || f.Name == ".."))
                    {
                        myListArray.Add(f);
                    }
                }
            }
            return myListArray;
        }

        /// <summary>
        /// 从Windows格式中返回文件信息
        /// </summary>
        /// <param name="Record">文件信息</param>
        private FileStruct ParseFileStructFromWindowsStyleRecord(string Record)
        {
            FileStruct f = new FileStruct();
            string processstr = Record.Trim();
            string dateStr = processstr.Substring(0, 8);
            processstr = (processstr.Substring(8, processstr.Length - 8)).Trim();
            string timeStr = processstr.Substring(0, 7);
            processstr = (processstr.Substring(7, processstr.Length - 7)).Trim();
            DateTimeFormatInfo myDTFI = new CultureInfo("en-US", false).DateTimeFormat;
            myDTFI.ShortTimePattern = "t";
            f.CreateTime = DateTime.Parse(dateStr + " " + timeStr, myDTFI);
            if (processstr.Substring(0, 5) == "<DIR>")
            {
                f.IsDirectory = true;
                processstr = (processstr.Substring(5, processstr.Length - 5)).Trim();
            }
            else
            {
                //string[] strs = processstr.Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries);   // true);
                //processstr = strs[1];
                processstr = processstr.Substring(processstr.IndexOf(' ') + 1);
                f.IsDirectory = false;
            }
            f.Name = processstr;
            return f;
        }
        /// <summary>
        /// 根据文件列表记录猜想文件列表类型
        /// </summary>
        /// <param name="recordList"></param>
        /// <returns></returns>
        private FileListStyle GuessFileListStyle(string[] recordList)
        {
            foreach (string s in recordList)
            {
                if (s.Length > 10 && Regex.IsMatch(s.Substring(0, 10), "(-|d)(-|r)(-|w)(-|x)(-|r)(-|w)(-|x)(-|r)(-|w)(-|x)"))
                {
                    return FileListStyle.UnixStyle;
                }
                else if (s.Length > 8 && Regex.IsMatch(s.Substring(0, 8), "[0-9][0-9]-[0-9][0-9]-[0-9][0-9]"))
                {
                    return FileListStyle.WindowsStyle;
                }
            }
            return FileListStyle.Unknown;
        }
        /// <summary>
        /// 从Unix格式中返回文件信息
        /// </summary>
        /// <param name="Record">文件信息</param>
        private FileStruct ParseFileStructFromUnixStyleRecord(string Record)
        {
            FileStruct f = new FileStruct();
            string processstr = Record.Trim();
            f.Flags = processstr.Substring(0, 10);
            f.IsDirectory = (f.Flags[0] == 'd');
            processstr = (processstr.Substring(11)).Trim();
            _cutSubstringFromStringWithTrim(ref processstr, ' ', 0);   //跳过一部分
            f.Owner = _cutSubstringFromStringWithTrim(ref processstr, ' ', 0);
            f.Group = _cutSubstringFromStringWithTrim(ref processstr, ' ', 0);
            f.FileSize = Convert.ToInt32(_cutSubstringFromStringWithTrim(ref processstr, ' ', 0));
            //_cutSubstringFromStringWithTrim(ref processstr, ' ', 0);   //跳过一部分
            string yearOrTime = processstr.Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries)[2];
            if (yearOrTime.IndexOf(":") >= 0)  //time
            {
                processstr = processstr.Replace(yearOrTime, DateTime.Now.Year.ToString());
            }
            f.CreateTime = DateTime.Parse(_cutSubstringFromStringWithTrim(ref processstr, ' ', 8));
            f.Name = processstr;   //最后就是名称
            return f;
        }
        /// <summary>
        /// 按照一定的规则进行字符串截取
        /// </summary>
        /// <param name="s">截取的字符串</param>
        /// <param name="c">查找的字符</param>
        /// <param name="startIndex">查找的位置</param>
        private string _cutSubstringFromStringWithTrim(ref string s, char c, int startIndex)
        {
            int pos1 = s.IndexOf(c, startIndex);
            string retString = s.Substring(0, pos1);
            s = (s.Substring(pos1)).Trim();
            return retString;
        }
        #endregion
        #endregion
        #region 上传文件
        /// <summary>
        /// 直接上传文件
        /// </summary>
        /// <param name="uploadUrl">上传的目标全路径与文件名</param>
        /// <param name="isContinueUpload">是否断点续传</param>
        /// <returns>上传是否成功</returns>
        public bool UploadFileNow(byte[] fileBytes, string uploadUrl, bool isContinueUpload)
        {
            bool result = true;

            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            int iCharIndex = 0;
            int bytesRead;
            long lOffset = 0, lTotalReaded = 0;
            string strDirUrl, strDirUrlTemp;
            try
            {
                ftpUrl = new FTPUrl(uploadUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                this.m_FTPConnectList.Add(ftpConnect);//添加到控制器中
                #region 判断并创建文件夹
                strDirUrl = ftpUrl.SubUrl;
                if (strDirUrl.IndexOf("/") >= 0)
                    strDirUrl = strDirUrl.Substring(0, strDirUrl.LastIndexOf("/"));
                while (true && strDirUrl.Trim() != "")
                {
                    iCharIndex = strDirUrl.IndexOf("/", iCharIndex) + 1;
                    if (iCharIndex == 0)
                        strDirUrlTemp = strDirUrl;
                    else
                        strDirUrlTemp = strDirUrl.Substring(0, iCharIndex);
                    if (strDirUrlTemp == "")
                        continue;
                    ftpConnect.SendCommand("MKD " + strDirUrlTemp);
                    if (ftpConnect.ReplyCode != 257 && ftpConnect.ReplyCode != 550)
                        throw new IOException(ftpConnect.ReplyString.Substring(4));
                    if (strDirUrlTemp == strDirUrl)
                        break;
                }
                #endregion
                ftpConnect.ChangeDir(strDirUrl);
                #region 得到服务器上的文件大小
                if (isContinueUpload)
                {
                    string strDirName = ftpUrl.SubUrl;
                    if (strDirName.IndexOf("/") >= 0)
                        strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                    ftpConnect.ChangeDir(strDirName);
                    ftpConnect.SendCommand("SIZE " + Path.GetFileName(ftpUrl.SubUrl));
                    if (ftpConnect.ReplyCode == 213)
                        lOffset = Int64.Parse(ftpConnect.ReplyString.Substring(4));
                }
                #endregion
                #region 开始上传
                lTotalReaded = lOffset;
                socketData = ftpConnect.CreateDataSocket();
                if (lOffset > 0)
                    ftpConnect.SendCommand("APPE " + Path.GetFileName(uploadUrl));
                else
                    ftpConnect.SendCommand("STOR " + Path.GetFileName(uploadUrl));
                if (!(ftpConnect.ReplyCode == 125 || ftpConnect.ReplyCode == 150))
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                ftpConnect.DataTransmitting = true;
                while (true)
                {
                    if (!ftpConnect.DataTransmitting)
                    {
                        this.OnFileUploadCanceled(ftpConnect);
                        break;
                    }
                    this.OnFileUploading(ftpConnect, fileBytes.Length, lTotalReaded);
                    //开始上传资料
                    bytesRead = (int)((fileBytes.Length > lTotalReaded + ftpConnect.Buffer.Length) ? ftpConnect.Buffer.Length : (fileBytes.Length - lTotalReaded));
                    if (bytesRead == 0)
                        break;
                    Array.Copy(fileBytes, lTotalReaded, ftpConnect.Buffer, 0, bytesRead);
                    socketData.Send(ftpConnect.Buffer, bytesRead, 0);
                    lTotalReaded += bytesRead;
                }
                if (socketData.Connected)
                    socketData.Close();
                if (ftpConnect.DataTransmitting)
                {
                    if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    {
                        ftpConnect.ReadReply();
                        if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                            throw new IOException(ftpConnect.ReplyString.Substring(4));
                    }
                    this.OnFileUploadCompleted(ftpConnect);
                }
                #endregion
            }
            catch (Exception ex)
            {
                result = false;
                throw (ex);
            }
            finally
            {
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                if (ftpConnect != null)
                {
                    ftpConnect.DisConnect();
                    this.m_FTPConnectList.Remove(ftpConnect);
                }
            }
            return result;
        }
        /// <summary>
        /// 直接上传文件
        /// </summary>
        /// <param name="filePath">上传文件的全路径</param>
        /// <param name="uploadUrl">上传的目标全路径与文件名</param>
        /// <param name="isContinueUpload">是否断点续传</param>
        /// <returns>上传是否成功</returns>
        public bool UploadFileNow(string filePath, string uploadUrl, bool isContinueUpload)
        {
            bool result = true;

            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            FileStream fileStream = null;
            int iCharIndex = 0;
            int bytesRead;
            long lOffset = 0, lTotalReaded = 0;
            string strDirUrl, strDirUrlTemp;
            try
            {
                ftpUrl = new FTPUrl(uploadUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                this.m_FTPConnectList.Add(ftpConnect);//添加到控制器中
                #region 判断并创建文件夹
                strDirUrl = ftpUrl.SubUrl;
                if (strDirUrl.IndexOf("/") >= 0)
                    strDirUrl = strDirUrl.Substring(0, strDirUrl.LastIndexOf("/"));
                while (true && strDirUrl.Trim() != "")
                {
                    iCharIndex = strDirUrl.IndexOf("/", iCharIndex) + 1;
                    if (iCharIndex == 0)
                        strDirUrlTemp = strDirUrl;
                    else
                        strDirUrlTemp = strDirUrl.Substring(0, iCharIndex);
                    if (strDirUrlTemp == "")
                        continue;
                    ftpConnect.SendCommand("MKD " + strDirUrlTemp);
                    if (ftpConnect.ReplyCode != 257 && ftpConnect.ReplyCode != 550)
                        throw new IOException(ftpConnect.ReplyString.Substring(4));
                    if (strDirUrlTemp == strDirUrl)
                        break;
                }
                #endregion
                ftpConnect.ChangeDir(strDirUrl);
                #region 得到服务器上的文件大小
                if (isContinueUpload)
                {
                    string strDirName = ftpUrl.SubUrl;
                    if (strDirName.IndexOf("/") >= 0)
                        strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                    ftpConnect.ChangeDir(strDirName);
                    ftpConnect.SendCommand("SIZE " + Path.GetFileName(ftpUrl.SubUrl));
                    if (ftpConnect.ReplyCode == 213)
                        lOffset = Int64.Parse(ftpConnect.ReplyString.Substring(4));
                }
                #endregion
                #region 开始上传
                fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.Read);
                lTotalReaded = lOffset;
                fileStream.Seek(lOffset, SeekOrigin.Begin);
                socketData = ftpConnect.CreateDataSocket();
                if (lOffset > 0)
                    ftpConnect.SendCommand("APPE " + Path.GetFileName(filePath));
                else
                    ftpConnect.SendCommand("STOR " + Path.GetFileName(filePath));
                if (!(ftpConnect.ReplyCode == 125 || ftpConnect.ReplyCode == 150))
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                ftpConnect.DataTransmitting = true;
                while (true)
                {
                    if (!ftpConnect.DataTransmitting)
                    {
                        this.OnFileUploadCanceled(ftpConnect);
                        break;
                    }
                    this.OnFileUploading(ftpConnect, fileStream.Length, lTotalReaded);
                    //开始上传资料
                    bytesRead = fileStream.Read(ftpConnect.Buffer, 0, ftpConnect.Buffer.Length);
                    if (bytesRead == 0)
                        break;
                    socketData.Send(ftpConnect.Buffer, bytesRead, 0);
                    lTotalReaded += bytesRead;
                }
                fileStream.Close();
                if (socketData.Connected)
                    socketData.Close();
                if (ftpConnect.DataTransmitting)
                {
                    if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    {
                        ftpConnect.ReadReply();
                        if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                            throw new IOException(ftpConnect.ReplyString.Substring(4));
                    }
                    this.OnFileUploadCompleted(ftpConnect);
                }
                #endregion
            }
            catch (Exception ex)
            {
                result = false;
                throw (ex);
            }
            finally
            {
                if (fileStream != null)
                    fileStream.Close();
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                if (ftpConnect != null)
                {
                    ftpConnect.DisConnect();
                    this.m_FTPConnectList.Remove(ftpConnect);
                }     
            }
            return result;
        }
        /// <summary>
        /// 上传文件
        /// </summary>
        /// <param name="filePath"></param>
        /// <param name="uploadUrl"></param>
        public string UploadFile(string filePath, string uploadUrl)
        {
            return this.UploadFile(filePath, uploadUrl, false);
        }
        /// <summary>
        /// 上传文件
        /// </summary>
        /// <param name="filePath">上传文件的全路径</param>
        /// <param name="uploadUrl">上传的目标全路径 包含了用户名用户密码与文件名</param>
        /// <param name="isContinueUpload">是否断点续传</param>
        /// <returns>返回控制上传下载的ID</returns>
        public string UploadFile(string filePath, string uploadUrl, bool isContinueUpload)
        {
            String strFTPId = System.Guid.NewGuid().ToString();
            IList<object> objList = new List<object> { filePath, uploadUrl, isContinueUpload, strFTPId };
            System.Threading.Thread threadUpload = new System.Threading.Thread(new System.Threading.ParameterizedThreadStart(ThreadUploadFile));
            threadUpload.Start(objList);  //开始采用线程方式下载
            return strFTPId;
        }
        /// <summary>
        /// 线程接收上传
        /// </summary>
        /// <param name="obj"></param>
        private void ThreadUploadFile(object obj)
        {
            string filePath;
            string uploadUrl;
            bool isContinueUpload;
            string strFTPId;
            IList<object> objList = obj as IList<object>;
            if (objList != null && objList.Count == 4)
            {
                filePath = objList[0] as string;
                uploadUrl = objList[1] as string;
                isContinueUpload = (bool)objList[2];
                strFTPId = objList[3] as string;
                this.ThreadUploadFile(filePath, uploadUrl, isContinueUpload, strFTPId);
            }
        }
        /// <summary>
        /// 线程上传文件
        /// </summary>
        /// <param name="filePath">上传文件的全路径</param>
        /// <param name="uploadUrl">上传的目标全路径与文件名</param>
        /// <param name="isContinueUpload">是否断点续传</param>
        /// <returns>上传是否成功</returns>
        private bool ThreadUploadFile(string filePath, string uploadUrl, bool isContinueUpload, string strFTPId)
        {
            bool result = true;

            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            FileStream fileStream = null;
            int iCharIndex = 0;
            int bytesRead;
            long lOffset = 0, lTotalReaded = 0;
            string strDirUrl, strDirUrlTemp;
            try
            {
                ftpUrl = new FTPUrl(uploadUrl);
                ftpConnect = new FTPConnect(ftpUrl, strFTPId);
                ftpConnect.Connect();
                this.m_FTPConnectList.Add(ftpConnect);//添加到控制器中
                #region 判断并创建文件夹
                strDirUrl = ftpUrl.SubUrl;
                if (strDirUrl.IndexOf("/") >= 0)
                    strDirUrl = strDirUrl.Substring(0, strDirUrl.LastIndexOf("/"));
                while (true && strDirUrl.Trim() != "")
                {
                    iCharIndex = strDirUrl.IndexOf("/", iCharIndex) + 1;
                    if (iCharIndex == 0)
                        strDirUrlTemp = strDirUrl;
                    else
                        strDirUrlTemp = strDirUrl.Substring(0, iCharIndex);
                    if (strDirUrlTemp == "")
                        continue;
                    ftpConnect.SendCommand("MKD " + strDirUrlTemp);
                    if (ftpConnect.ReplyCode != 257 && ftpConnect.ReplyCode != 550)
                        throw new IOException(ftpConnect.ReplyString.Substring(4));
                    if (strDirUrlTemp == strDirUrl)
                        break;
                }
                #endregion
                ftpConnect.ChangeDir(strDirUrl);
                #region 得到服务器上的文件大小
                if (isContinueUpload)
                {
                    string strDirName = ftpUrl.SubUrl;
                    if (strDirName.IndexOf("/") >= 0)
                        strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                    ftpConnect.ChangeDir(strDirName);
                    ftpConnect.SendCommand("SIZE " + Path.GetFileName(ftpUrl.SubUrl));
                    if (ftpConnect.ReplyCode == 213)
                        lOffset = Int64.Parse(ftpConnect.ReplyString.Substring(4));
                }
                #endregion
                #region 开始上传
                fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.Read);
                lTotalReaded = lOffset;
                fileStream.Seek(lOffset, SeekOrigin.Begin);
                socketData = ftpConnect.CreateDataSocket();
                if (lOffset > 0)
                    ftpConnect.SendCommand("APPE " + Path.GetFileName(uploadUrl));
                else
                    ftpConnect.SendCommand("STOR " + Path.GetFileName(uploadUrl));
                if (!(ftpConnect.ReplyCode == 125 || ftpConnect.ReplyCode == 150))
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                ftpConnect.DataTransmitting = true;
                while (true)
                {
                    if (!ftpConnect.DataTransmitting)
                    {
                        this.OnFileUploadCanceled(ftpConnect);
                        break;
                    }
                    this.OnFileUploading(ftpConnect, fileStream.Length, lTotalReaded);
                    //开始上传资料
                    bytesRead = fileStream.Read(ftpConnect.Buffer, 0, ftpConnect.Buffer.Length);
                    if (bytesRead == 0)
                        break;
                    socketData.Send(ftpConnect.Buffer, bytesRead, 0);
                    lTotalReaded += bytesRead;
                }
                fileStream.Close();
                if (socketData.Connected)
                    socketData.Close();
                if (ftpConnect.DataTransmitting)
                {
                    if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    {
                        ftpConnect.ReadReply();
                        if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                            throw new IOException(ftpConnect.ReplyString.Substring(4));
                    }
                    this.OnFileUploadCompleted(ftpConnect);
                }
                #endregion
            }
            catch (Exception ex)
            {
                result = false;
                this.OnFtpError(ftpConnect, ex);
            }
            finally
            {
                if (fileStream != null)
                    fileStream.Close();
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                if (ftpConnect != null)
                {
                    ftpConnect.DisConnect();
                    this.m_FTPConnectList.Remove(ftpConnect);
                } 
            }
            return result;
        }
        /// <summary>
        /// 取消正在上传的文件
        /// </summary>
        /// <returns></returns>
        public void CancelUploadFile(FTPConnect ftpConnect)
        {
            if (ftpConnect != null)
                ftpConnect.DataTransmitting = false;
        }
        /// <summary>
        /// 取消正在上传的文件
        /// </summary>
        /// <param name="strID"></param>
        public void CancelUploadFile(string strID)
        {
            foreach (FTPConnect ftp in this.m_FTPConnectList)
            {
                if (ftp != null && ftp.ID == strID)
                {
                    ftp.DataTransmitting = false;
                    break;
                }
            }
        }
        #endregion
        #region 下载文件
        /// <summary>
        /// 直接下载文件
        /// </summary>
        /// <param name="downloadUrl">要下载文件的路径</param>
        /// <param name="fileBytes">存贮的内容</param>
        /// <returns>下载是否成功</returns>
        public bool DownloadFileNow(string downloadUrl, out byte[] fileBytes)
        {
            bool result = true;

            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            fileBytes = new byte[] { };
            int bytesRead;
            long lTotalReaded = 0, lFileSize;
            try
            {
                ftpUrl = new FTPUrl(downloadUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                this.m_FTPConnectList.Add(ftpConnect);//添加到控制器中
                string strDirUrl = ftpUrl.SubUrl;
                if (strDirUrl.IndexOf("/") >= 0)
                    strDirUrl = strDirUrl.Substring(0, strDirUrl.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirUrl);
                #region 得到服务器上的文件大小
                string strDirName = ftpUrl.SubUrl;
                if (strDirName.IndexOf("/") >= 0)
                    strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirName);
                ftpConnect.SendCommand("SIZE " + Path.GetFileName(ftpUrl.SubUrl));
                if (ftpConnect.ReplyCode == 213)
                    lFileSize = Int64.Parse(ftpConnect.ReplyString.Substring(4));
                else
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                #endregion

                socketData = ftpConnect.CreateDataSocket();
                fileBytes = new byte[lFileSize];
                ftpConnect.SendCommand("RETR " + Path.GetFileName(ftpUrl.SubUrl));
                if (!(ftpConnect.ReplyCode == 150 || ftpConnect.ReplyCode == 125 || ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                #region 开始下载
                ftpConnect.DataTransmitting = true;
                while (true)
                {
                    if (!ftpConnect.DataTransmitting)    //判断取消是否取消了下载
                    {
                        this.OnFileDownloadCanceled(ftpConnect);
                        break;
                    }
                    this.OnFileDownloading(ftpConnect, lFileSize, lTotalReaded);
                    //开始将文件流写入本地
                    bytesRead = socketData.Receive(ftpConnect.Buffer, ftpConnect.Buffer.Length, 0);
                    if (bytesRead <= 0)
                        break;
                    Array.Copy(ftpConnect.Buffer, 0, fileBytes, lTotalReaded, bytesRead);
                    lTotalReaded += bytesRead;
                }
                if (socketData.Connected)
                    socketData.Close();
                if (ftpConnect.DataTransmitting)
                {
                    if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    {
                        ftpConnect.ReadReply();
                        if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                            throw new IOException(ftpConnect.ReplyString.Substring(4));
                    }
                    this.OnFileDownloadCompleted(ftpConnect);
                }
                #endregion
            }
            catch (Exception ex)
            {
                result = false;
                throw (ex);
            }
            finally
            {
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                if (ftpConnect != null)
                {
                    ftpConnect.DisConnect();
                    this.m_FTPConnectList.Remove(ftpConnect);
                } 
            }
            return result;
        }
        /// <summary>
        /// 直接下载文件
        /// </summary>
        /// <param name="downloadUrl">要下载文件的路径</param>
        /// <param name="targetFile">目标存在全路径</param>
        /// <param name="isContinueDownload">是否断点续传</param>
        /// <returns>下载是否成功</returns>
        public bool DownloadFileNow(string downloadUrl, string targetFile, bool isContinueDownload)
        {
            bool result = true;

            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            FileStream fileStream = null;
            int bytesRead;
            long lTotalReaded = 0, lFileSize;
            try
            {
                ftpUrl = new FTPUrl(downloadUrl);
                ftpConnect = new FTPConnect(ftpUrl);
                ftpConnect.Connect();
                this.m_FTPConnectList.Add(ftpConnect);//添加到控制器中
                string strDirUrl = ftpUrl.SubUrl;
                if (strDirUrl.IndexOf("/") >= 0)
                    strDirUrl = strDirUrl.Substring(0, strDirUrl.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirUrl);
                #region 得到服务器上的文件大小
                string strDirName = ftpUrl.SubUrl;
                if (strDirName.IndexOf("/") >= 0)
                    strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirName);
                ftpConnect.SendCommand("SIZE " + Path.GetFileName(ftpUrl.SubUrl));
                if (ftpConnect.ReplyCode == 213)
                    lFileSize = Int64.Parse(ftpConnect.ReplyString.Substring(4));
                else
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                #endregion

                socketData = ftpConnect.CreateDataSocket();
                //断点续传长度的偏移量
                if (System.IO.File.Exists(targetFile) && isContinueDownload)
                {
                    System.IO.FileInfo fiInfo = new FileInfo(targetFile);
                    lTotalReaded = fiInfo.Length;
                    ftpConnect.SendCommand("REST " + fiInfo.Length.ToString());
                    if (ftpConnect.ReplyCode != 350)
                        throw new IOException(ftpConnect.ReplyString.Substring(4));
                }
                ftpConnect.SendCommand("RETR " + Path.GetFileName(ftpUrl.SubUrl));
                if (!(ftpConnect.ReplyCode == 150 || ftpConnect.ReplyCode == 125 || ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                #region 开始下载
                string strTargetPath = targetFile;
                strTargetPath = strTargetPath.Substring(0, strTargetPath.LastIndexOf("\\"));//这里本来是//
                if (!System.IO.Directory.Exists(strTargetPath)) //判断目标路径是否存在,如果不存在就创建
                    System.IO.Directory.CreateDirectory(strTargetPath);
                if (System.IO.File.Exists(targetFile) && isContinueDownload)  //目标文件已经是全路径了 断点续传
                    fileStream = new System.IO.FileStream(targetFile, System.IO.FileMode.Append, System.IO.FileAccess.Write);
                else
                    fileStream = new System.IO.FileStream(targetFile, System.IO.FileMode.Create, System.IO.FileAccess.Write);
                ftpConnect.DataTransmitting = true;
                while (true)
                {
                    if (!ftpConnect.DataTransmitting)    //判断取消是否取消了下载
                    {
                        this.OnFileDownloadCanceled(ftpConnect);
                        break;
                    }
                    this.OnFileDownloading(ftpConnect, lFileSize, lTotalReaded);
                    //开始将文件流写入本地
                    bytesRead = socketData.Receive(ftpConnect.Buffer, ftpConnect.Buffer.Length, 0);
                    if (bytesRead <= 0)
                        break;
                    fileStream.Write(ftpConnect.Buffer, 0, bytesRead);
                    lTotalReaded += bytesRead;
                }
                fileStream.Close();
                if (socketData.Connected)
                    socketData.Close();
                if (ftpConnect.DataTransmitting)
                {
                    if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    {
                        ftpConnect.ReadReply();
                        if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                            throw new IOException(ftpConnect.ReplyString.Substring(4));
                    }
                    this.OnFileDownloadCompleted(ftpConnect);
                }
                #endregion
            }
            catch (Exception ex)
            {
                result = false;
                throw (ex);
            }
            finally
            {
                if (fileStream != null)
                    fileStream.Close();
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                if (ftpConnect != null)
                {
                    ftpConnect.DisConnect();
                    this.m_FTPConnectList.Remove(ftpConnect);
                }
            }
            return result;
        }
        /// <summary>
        /// 下载文件
        /// </summary>
        /// <param name="downloadUrl"></param>
        /// <param name="targetFile"></param>
        public string DownloadFile(string downloadUrl, string targetFile)
        {
            return this.DownloadFile(downloadUrl, targetFile, false);
        }
        /// <summary>
        /// 下载文件
        /// </summary>
        /// <param name="downloadUrl">要下载文件的路径 包含了登录名与登录密码</param>
        /// <param name="TargetFile">目标存在路径包含文件名</param>
        /// <param name="isContinueDownload">是否断点续传</param>
        /// <returns>返回下载控制ID</returns>
        public string DownloadFile(string downloadUrl, string targetFile, bool isContinueDownload)
        {
            String strFTPId = System.Guid.NewGuid().ToString();
            IList<object> objList = new List<object> { downloadUrl, targetFile, isContinueDownload, strFTPId };
            System.Threading.Thread threadDownload = new System.Threading.Thread(new System.Threading.ParameterizedThreadStart(ThreadDownloadFile));
            threadDownload.Start(objList);  //开始采用线程方式下载
            return strFTPId;
        }
        /// <summary>
        /// 线程接收下载
        /// </summary>
        /// <param name="obj"></param>
        private void ThreadDownloadFile(object obj)
        {
            string downloadUrl;
            string targetFile;
            bool isContinueDownload;
            string strFTPId;
            IList<object> objList = obj as IList<object>;
            if (objList != null && objList.Count == 4)
            {
                downloadUrl = objList[0] as string;
                targetFile = objList[1] as string;
                isContinueDownload = (bool)objList[2];
                strFTPId = objList[3] as String;
                this.ThreadDownloadFile(downloadUrl, targetFile, isContinueDownload, strFTPId);
            }
        }
        /// <summary>
        /// 线程下载文件
        /// </summary>
        /// <param name="downloadUrl">要下载文件的路径</param>
        /// <param name="targetFile">目标存在全路径</param>
        /// <param name="isContinueDownload">是否断点续传</param>
        /// <returns>下载是否成功</returns>
        private bool ThreadDownloadFile(string downloadUrl, string targetFile, bool isContinueDownload, string strFTPId)
        {
            bool result = true;

            FTPUrl ftpUrl = null;
            FTPConnect ftpConnect = null;
            Socket socketData = null;
            FileStream fileStream = null;
            int bytesRead;
            long lTotalReaded = 0, lFileSize;
            try
            {
                ftpUrl = new FTPUrl(downloadUrl);
                ftpConnect = new FTPConnect(ftpUrl, strFTPId);
                ftpConnect.Connect();
                this.m_FTPConnectList.Add(ftpConnect);//添加到控制器中
                string strDirUrl = ftpUrl.SubUrl;
                if (strDirUrl.IndexOf("/") >= 0)
                    strDirUrl = strDirUrl.Substring(0, strDirUrl.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirUrl);
                #region 得到服务器上的文件大小
                string strDirName = ftpUrl.SubUrl;
                if (strDirName.IndexOf("/") >= 0)
                    strDirName = strDirName.Substring(0, strDirName.LastIndexOf("/"));
                ftpConnect.ChangeDir(strDirName);
                ftpConnect.SendCommand("SIZE " + Path.GetFileName(ftpUrl.SubUrl));
                if (ftpConnect.ReplyCode == 213)
                    lFileSize = Int64.Parse(ftpConnect.ReplyString.Substring(4));
                else
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                #endregion

                socketData = ftpConnect.CreateDataSocket();
                //断点续传长度的偏移量
                if (System.IO.File.Exists(targetFile) && isContinueDownload)
                {
                    System.IO.FileInfo fiInfo = new FileInfo(targetFile);
                    lTotalReaded = fiInfo.Length;
                    ftpConnect.SendCommand("REST " + fiInfo.Length.ToString());
                    if (ftpConnect.ReplyCode != 350)
                        throw new IOException(ftpConnect.ReplyString.Substring(4));
                }
                ftpConnect.SendCommand("RETR " + Path.GetFileName(ftpUrl.SubUrl));
                if (!(ftpConnect.ReplyCode == 150 || ftpConnect.ReplyCode == 125 || ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    throw new IOException(ftpConnect.ReplyString.Substring(4));
                #region 开始下载
                string strTargetPath = targetFile;
                strTargetPath = strTargetPath.Substring(0, strTargetPath.LastIndexOf("//"));
                if (!System.IO.Directory.Exists(strTargetPath)) //判断目标路径是否存在,如果不存在就创建
                    System.IO.Directory.CreateDirectory(strTargetPath);
                if (System.IO.File.Exists(targetFile) && isContinueDownload)  //目标文件已经是全路径了 断点续传
                    fileStream = new System.IO.FileStream(targetFile, System.IO.FileMode.Append, System.IO.FileAccess.Write);
                else
                    fileStream = new System.IO.FileStream(targetFile, System.IO.FileMode.Create, System.IO.FileAccess.Write);
                ftpConnect.DataTransmitting = true;
                while (true)
                {
                    if (!ftpConnect.DataTransmitting)    //判断取消是否取消了下载
                    {
                        this.OnFileDownloadCanceled(ftpConnect);
                        break;
                    }
                    this.OnFileDownloading(ftpConnect, lFileSize, lTotalReaded);
                    //开始将文件流写入本地
                    bytesRead = socketData.Receive(ftpConnect.Buffer, ftpConnect.Buffer.Length, 0);
                    if (bytesRead <= 0)
                        break;
                    fileStream.Write(ftpConnect.Buffer, 0, bytesRead);
                    lTotalReaded += bytesRead;
                }
                fileStream.Close();
                if (socketData.Connected)
                    socketData.Close();
                if (ftpConnect.DataTransmitting)
                {
                    if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                    {
                        ftpConnect.ReadReply();
                        if (!(ftpConnect.ReplyCode == 226 || ftpConnect.ReplyCode == 250))
                            throw new IOException(ftpConnect.ReplyString.Substring(4));
                    }
                    this.OnFileDownloadCompleted(ftpConnect);
                }
                #endregion
            }
            catch (Exception ex)
            {
                result = false;
                this.OnFtpError(ftpConnect, ex);
            }
            finally
            {
                if (fileStream != null)
                    fileStream.Close();
                if (socketData != null && socketData.Connected)
                    socketData.Close();
                if (ftpConnect != null)
                {
                    ftpConnect.DisConnect();
                    this.m_FTPConnectList.Remove(ftpConnect);
                }
            }
            return result;
        }
        /// <summary>
        /// 取消正在下载的文件
        /// </summary>
        /// <returns></returns>
        public void CancelDownloadFile(FTPConnect ftpConnect)
        {
            if (ftpConnect != null)
                ftpConnect.DataTransmitting = false;
        }
        /// <summary>
        /// 取消正在下载的文件
        /// </summary>
        /// <param name="strID"></param>
        public void CancelDownloadFile(string strID)
        {
            foreach (FTPConnect ftp in this.m_FTPConnectList)
            {
                if (ftp != null && ftp.ID == strID)
                {
                    ftp.DataTransmitting = false;
                    break;
                }
            }
        }
        #endregion
        #region 根据指定的ID查找FTPConnect
        /// <summary>
        /// 根据指定的ID查找FTPConnect
        /// </summary>
        /// <param name="id"></param>
        /// <returns></returns>
        public FTPConnect FindFTPConnectByID(string id)
        {
            foreach (FTPConnect ftpConnect in this.m_FTPConnectList)
            {
                if (ftpConnect != null && ftpConnect.ID == id)
                    return ftpConnect;
            }
            return null;
        }
        #endregion

        #region 传输类型
        public enum FTPType
        {
            /// <summary>
            /// 无状态
            /// </summary>
            None,
            /// <summary>
            /// 上传
            /// </summary>
            Upload,
            /// <summary>
            /// 下载
            /// </summary>
            Download
        }
        #endregion
    }
    #region 文件信息结构
    public struct FileStruct
    {
        public string Flags;
        public string Owner;
        public string Group;
        public int FileSize;
        public bool IsDirectory;
        public DateTime CreateTime;
        public string Name;
    }
    #endregion
    #region 文件传输进度控制事件
    public delegate void FTPSendEventHandler(object sender, FTPSendEventArgs e);
    public class FTPSendEventArgs : System.EventArgs
    {
        private long m_totalbytes;			// Total Bytes
        private long m_bytestransfered;

        public FTPSendEventArgs()
        {
            m_totalbytes = 0;
            m_bytestransfered = 0;
        }
        public FTPSendEventArgs(long lTotalBytes, long lBytesTransfered)
        {
            m_totalbytes = lTotalBytes;
            m_bytestransfered = lBytesTransfered;
        }
        /// <summary>
        /// 总字节数
        /// </summary>
        public long TotalBytes
        {
            get { return m_totalbytes; }
            set { m_totalbytes = value; }
        }
        /// <summary>
        /// 已传输字节数
        /// </summary>
        public long BytesTransfered
        {
            get { return m_bytestransfered; }
            set { m_bytestransfered = value; }
        }
    }
    public delegate void FTPErrorEventHandler(object sender, FTPErrorEventArgs e);
    public class FTPErrorEventArgs : System.EventArgs
    {
        private Exception m_Error = null;
        public FTPErrorEventArgs() { }
        public FTPErrorEventArgs(Exception error)
        {
            this.m_Error = error;
        }
        /// <summary>
        /// 错误消息
        /// </summary>
        public Exception Error
        {
            get { return this.m_Error; }
            set { this.m_Error = value; }
        }
    }
    #endregion
}

 使用示例:

private void Download()
        {
            FTPHelper ftphelper = new FTPHelper();
            ftphelper.FileDownloading += new FTPSendEventHandler(ftphelper_FileDownloading);
            ftphelper.FileDownloadCompleted += new EventHandler(ftphelper_FileDownloadCompleted);
            ftphelper.DownloadFileNow("ftp://userName:password@address/test.mp3", "D://office//test.mp3", false);
        }
private void ftphelper_FileDownloadCompleted(object sender, EventArgs e)
        {
            MessageBox.Show("下载完成");
        }

private void ftphelper_FileDownloading(object sender, FTPSendEventArgs e)
        {
            progressBar.Maximum = (int)e.TotalBytes;
            progressBar.Value = ((int)e.BytesTransfered);
        }

 使用WebRequest的另一个Ftp类:

using System;
using System.Collections.Generic;
using System.Text;
using System.Net;
using System.IO;
using System.Globalization;
using System.Text.RegularExpressions;

namespace Ftp
{
    #region 文件信息结构
    public struct FileStruct
    {
        public string Flags;
        public string Owner;
        public string Group;
        public bool IsDirectory;
        public DateTime CreateTime;
        public string Name;
    }
    public enum FileListStyle
    {
        UnixStyle,
        WindowsStyle,
        Unknown
    }
    #endregion


    /// <summary>
    /// FTP处理操作类
    /// 功能:
    /// 下载文件
    /// 上传文件
    /// 上传文件的进度信息
    /// 下载文件的进度信息
    /// 删除文件
    /// 列出文件
    /// 列出目录
    /// 进入子目录
    /// 退出当前目录返回上一层目录
    /// 判断远程文件是否存在
    /// 判断远程文件是否存在
    /// 删除远程文件    
    /// 建立目录
    /// 删除目录
    /// 文件(目录)改名
    /// </summary>
    public class FTPHelper
    {
        #region 属性信息
        /// <summary>
        /// FTP请求对象
        /// </summary>
        FtpWebRequest Request = null;
        /// <summary>
        /// FTP响应对象
        /// </summary>
        FtpWebResponse Response = null;
        /// <summary>
        /// FTP服务器地址
        /// </summary>
        private Uri _Uri;
        /// <summary>
        /// FTP服务器地址
        /// </summary>
        public Uri Uri
        {
            get
            {
                if (_DirectoryPath == "/")
                {
                    return _Uri;
                }
                else
                {
                    string strUri = _Uri.ToString();
                    if (strUri.EndsWith("/"))
                    {
                        strUri = strUri.Substring(0, strUri.Length - 1);
                    }
                    return new Uri(strUri + this.DirectoryPath);
                }
            }
            set
            {
                if (value.Scheme != Uri.UriSchemeFtp)
                {
                    throw new Exception("Ftp 地址格式错误!");
                }
                _Uri = new Uri(value.GetLeftPart(UriPartial.Authority));
                _DirectoryPath = value.AbsolutePath;
                if (!_DirectoryPath.EndsWith("/"))
                {
                    _DirectoryPath += "/";
                }
            }
        }

        /// <summary>
        /// 当前工作目录
        /// </summary>
        private string _DirectoryPath;

        /// <summary>
        /// 当前工作目录
        /// </summary>
        public string DirectoryPath
        {
            get { return _DirectoryPath; }
            set { _DirectoryPath = value; }
        }

        /// <summary>
        /// FTP登录用户
        /// </summary>
        private string _UserName;
        /// <summary>
        /// FTP登录用户
        /// </summary>
        public string UserName
        {
            get { return _UserName; }
            set { _UserName = value; }
        }

        /// <summary>
        /// 错误信息
        /// </summary>
        private string _ErrorMsg;
        /// <summary>
        /// 错误信息
        /// </summary>
        public string ErrorMsg
        {
            get { return _ErrorMsg; }
            set { _ErrorMsg = value; }
        }

        /// <summary>
        /// FTP登录密码
        /// </summary>
        private string _Password;
        /// <summary>
        /// FTP登录密码
        /// </summary>
        public string Password
        {
            get { return _Password; }
            set { _Password = value; }
        }

        /// <summary>
        /// 连接FTP服务器的代理服务
        /// </summary>
        private WebProxy _Proxy = null;
        /// <summary>
        /// 连接FTP服务器的代理服务
        /// </summary>
        public WebProxy Proxy
        {
            get
            {
                return _Proxy;
            }
            set
            {
                _Proxy = value;
            }
        }

        /// <summary>
        /// 是否需要删除临时文件
        /// </summary>
        private bool _isDeleteTempFile = false;
        /// <summary>
        /// 异步上传所临时生成的文件
        /// </summary>
        private string _UploadTempFile = "";
        #endregion


        #region 事件
        public delegate void De_DownloadProgressChanged(object sender, DownloadProgressChangedEventArgs e);
        public delegate void De_DownloadDataCompleted(object sender, System.ComponentModel.AsyncCompletedEventArgs e);
        public delegate void De_UploadProgressChanged(object sender, UploadProgressChangedEventArgs e);
        public delegate void De_UploadFileCompleted(object sender, UploadFileCompletedEventArgs e);

        /// <summary>
        /// 异步下载进度发生改变触发的事件
        /// </summary>
        public event De_DownloadProgressChanged DownloadProgressChanged;
        /// <summary>
        /// 异步下载文件完成之后触发的事件
        /// </summary>
        public event De_DownloadDataCompleted DownloadDataCompleted;
        /// <summary>
        /// 异步上传进度发生改变触发的事件
        /// </summary>
        public event De_UploadProgressChanged UploadProgressChanged;
        /// <summary>
        /// 异步上传文件完成之后触发的事件
        /// </summary>
        public event De_UploadFileCompleted UploadFileCompleted;
        #endregion


        #region 构造析构函数
        /// <summary>
        /// 构造函数
        /// </summary>
        /// <param name="FtpUri">FTP地址</param>
        /// <param name="strUserName">登录用户名</param>
        /// <param name="strPassword">登录密码</param>
        public FTPHelper(Uri FtpUri, string strUserName, string strPassword)
        {
            this._Uri = new Uri(FtpUri.GetLeftPart(UriPartial.Authority));
            _DirectoryPath = FtpUri.AbsolutePath;
            if (!_DirectoryPath.EndsWith("/"))
            {
                _DirectoryPath += "/";
            }
            this._UserName = strUserName;
            this._Password = strPassword;
            this._Proxy = null;
        }
        /// <summary>
        /// 构造函数
        /// </summary>
        /// <param name="FtpUri">FTP地址</param>
        /// <param name="strUserName">登录用户名</param>
        /// <param name="strPassword">登录密码</param>
        /// <param name="objProxy">连接代理</param>
        public FTPHelper(Uri FtpUri, string strUserName, string strPassword, WebProxy objProxy)
        {
            this._Uri = new Uri(FtpUri.GetLeftPart(UriPartial.Authority));
            _DirectoryPath = FtpUri.AbsolutePath;
            if (!_DirectoryPath.EndsWith("/"))
            {
                _DirectoryPath += "/";
            }
            this._UserName = strUserName;
            this._Password = strPassword;
            this._Proxy = objProxy;
        }
        /// <summary>
        /// 构造函数
        /// </summary>
        public FTPHelper()
        {
            this._UserName = "anonymous";  //匿名用户
            this._Password = "@anonymous";
            this._Uri = null;
            this._Proxy = null;
        }

        /// <summary>
        /// 析构函数
        /// </summary>
        ~FTPHelper()
        {
            if (Response != null)
            {
                Response.Close();
                Response = null;
            }
            if (Request != null)
            {
                Request.Abort();
                Request = null;
            }
        }
        #endregion


        #region 建立连接
        /// <summary>
        /// 建立FTP链接,返回响应对象
        /// </summary>
        /// <param name="uri">FTP地址</param>
        /// <param name="FtpMathod">操作命令</param>
        private FtpWebResponse Open(Uri uri, string FtpMathod)
        {
            try
            {
                Request = (FtpWebRequest)WebRequest.Create(uri);
                Request.Method = FtpMathod;
                Request.UseBinary = true;
                Request.KeepAlive = false;
                Request.Credentials = new NetworkCredential(this.UserName, this.Password);

                if (this.Proxy != null)
                {
                    Request.Proxy = this.Proxy;
                }
                return (FtpWebResponse)Request.GetResponse();
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 建立FTP链接,返回请求对象
        /// </summary>
        /// <param name="uri">FTP地址</param>
        /// <param name="FtpMathod">操作命令</param>
        private FtpWebRequest OpenRequest(Uri uri, string FtpMathod)
        {
            try
            {
                Request = (FtpWebRequest)WebRequest.Create(uri);
                Request.Method = FtpMathod;
                Request.UseBinary = true;
                Request.KeepAlive = false;
                Request.Credentials = new NetworkCredential(this.UserName, this.Password);

                if (this.Proxy != null)
                {
                    Request.Proxy = this.Proxy;
                }
                return Request;
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 下载文件
        /// <summary>
        /// 从FTP服务器下载文件,使用与远程文件同名的文件名来保存文件
        /// </summary>
        /// <param name="RemoteFileName">远程文件名</param>
        /// <param name="LocalPath">本地路径</param>

        public bool DownloadFile(string RemoteFileName, string LocalPath)
        {
            return DownloadFile(RemoteFileName, LocalPath, RemoteFileName);
        }
        /// <summary>
        /// 从FTP服务器下载文件,指定本地路径和本地文件名
        /// </summary>
        /// <param name="RemoteFileName">远程文件名</param>
        /// <param name="LocalPath">本地路径</param>
        /// <param name="LocalFilePath">保存文件的本地路径,后面带有"\"</param>
        /// <param name="LocalFileName">保存本地的文件名</param>
        public bool DownloadFile(string RemoteFileName, string LocalPath, string LocalFileName)
        {
            byte[] bt = null;
            try
            {
                if (!IsValidFileChars(RemoteFileName) || !IsValidFileChars(LocalFileName) || !IsValidPathChars(LocalPath))
                {
                    throw new Exception("非法文件名或目录名!");
                }
                if (!Directory.Exists(LocalPath))
                {
                    throw new Exception("本地文件路径不存在!");
                }

                string LocalFullPath = Path.Combine(LocalPath, LocalFileName);
                if (File.Exists(LocalFullPath))
                {
                    throw new Exception("当前路径下已经存在同名文件!");
                }
                bt = DownloadFile(RemoteFileName);
                if (bt != null)
                {
                    FileStream stream = new FileStream(LocalFullPath, FileMode.Create);
                    stream.Write(bt, 0, bt.Length);
                    stream.Flush();
                    stream.Close();
                    return true;
                }
                else
                {
                    return false;
                }
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }

        /// <summary>
        /// 从FTP服务器下载文件,返回文件二进制数据
        /// </summary>
        /// <param name="RemoteFileName">远程文件名</param>
        public byte[] DownloadFile(string RemoteFileName)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName))
                {
                    throw new Exception("非法文件名或目录名!");
                }
                Response = Open(new Uri(this.Uri.ToString() + RemoteFileName), WebRequestMethods.Ftp.DownloadFile);
                Stream Reader = Response.GetResponseStream();

                MemoryStream mem = new MemoryStream(1024 * 500);
                byte[] buffer = new byte[1024];
                int bytesRead = 0;
                int TotalByteRead = 0;
                while (true)
                {
                    bytesRead = Reader.Read(buffer, 0, buffer.Length);
                    TotalByteRead += bytesRead;
                    if (bytesRead == 0)
                        break;
                    mem.Write(buffer, 0, bytesRead);
                }
                if (mem.Length >= 0)
                {
                    return mem.ToArray();
                }
                else
                {
                    return null;
                }
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 异步下载文件
        /// <summary>
        /// 从FTP服务器异步下载文件,指定本地路径和本地文件名
        /// </summary>
        /// <param name="RemoteFileName">远程文件名</param>        
        /// <param name="LocalPath">保存文件的本地路径,后面带有"\"</param>
        /// <param name="LocalFileName">保存本地的文件名</param>
        public void DownloadFileAsync(string RemoteFileName, string LocalPath, string LocalFileName)
        {
            //byte[] bt = null;
            try
            {
                if (!IsValidFileChars(RemoteFileName) || !IsValidFileChars(LocalFileName) || !IsValidPathChars(LocalPath))
                {
                    throw new Exception("非法文件名或目录名!");
                }
                if (!Directory.Exists(LocalPath))
                {
                    throw new Exception("本地文件路径不存在!");
                }

                string LocalFullPath = Path.Combine(LocalPath, LocalFileName);
                if (File.Exists(LocalFullPath))
                {
                    throw new Exception("当前路径下已经存在同名文件!");
                }
                DownloadFileAsync(RemoteFileName, LocalFullPath);

            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }

        /// <summary>
        /// 从FTP服务器异步下载文件,指定本地完整路径文件名
        /// </summary>
        /// <param name="RemoteFileName">远程文件名</param>
        /// <param name="LocalFullPath">本地完整路径文件名</param>
        public void DownloadFileAsync(string RemoteFileName, string LocalFullPath)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName))
                {
                    throw new Exception("非法文件名或目录名!");
                }
                if (File.Exists(LocalFullPath))
                {
                    throw new Exception("当前路径下已经存在同名文件!");
                }
                MyWebClient client = new MyWebClient();

                client.DownloadProgressChanged += new DownloadProgressChangedEventHandler(client_DownloadProgressChanged);
                client.DownloadFileCompleted += new System.ComponentModel.AsyncCompletedEventHandler(client_DownloadFileCompleted);
                client.Credentials = new NetworkCredential(this.UserName, this.Password);
                if (this.Proxy != null)
                {
                    client.Proxy = this.Proxy;
                }
                client.DownloadFileAsync(new Uri(this.Uri.ToString() + RemoteFileName), LocalFullPath);
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }

        /// <summary>
        /// 异步下载文件完成之后触发的事件
        /// </summary>
        /// <param name="sender">下载对象</param>
        /// <param name="e">数据信息对象</param>
        void client_DownloadFileCompleted(object sender, System.ComponentModel.AsyncCompletedEventArgs e)
        {
            if (DownloadDataCompleted != null)
            {
                DownloadDataCompleted(sender, e);
            }
        }

        /// <summary>
        /// 异步下载进度发生改变触发的事件
        /// </summary>
        /// <param name="sender">下载对象</param>
        /// <param name="e">进度信息对象</param>
        void client_DownloadProgressChanged(object sender, DownloadProgressChangedEventArgs e)
        {
            if (DownloadProgressChanged != null)
            {
                DownloadProgressChanged(sender, e);
            }
        }
        #endregion


        #region 上传文件
        /// <summary>
        /// 上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件名</param>
        public bool UploadFile(string LocalFullPath)
        {
            return UploadFile(LocalFullPath, Path.GetFileName(LocalFullPath), false);
        }
        /// <summary>
        /// 上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件</param>
        /// <param name="OverWriteRemoteFile">是否覆盖远程服务器上面同名的文件</param>
        public bool UploadFile(string LocalFullPath, bool OverWriteRemoteFile)
        {
            return UploadFile(LocalFullPath, Path.GetFileName(LocalFullPath), OverWriteRemoteFile);
        }
        /// <summary>
        /// 上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        public bool UploadFile(string LocalFullPath, string RemoteFileName)
        {
            return UploadFile(LocalFullPath, RemoteFileName, false);
        }
        /// <summary>
        /// 上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件名</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        /// <param name="OverWriteRemoteFile">是否覆盖远程服务器上面同名的文件</param>
        public bool UploadFile(string LocalFullPath, string RemoteFileName, bool OverWriteRemoteFile)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName) || !IsValidFileChars(Path.GetFileName(LocalFullPath)) || !IsValidPathChars(Path.GetDirectoryName(LocalFullPath)))
                {
                    throw new Exception("非法文件名或目录名!");
                }
                if (File.Exists(LocalFullPath))
                {
                    FileStream Stream = new FileStream(LocalFullPath, FileMode.Open, FileAccess.Read);
                    byte[] bt = new byte[Stream.Length];
                    Stream.Read(bt, 0, (Int32)Stream.Length);   //注意,因为Int32的最大限制,最大上传文件只能是大约2G多一点
                    Stream.Close();
                    return UploadFile(bt, RemoteFileName, OverWriteRemoteFile);
                }
                else
                {
                    throw new Exception("本地文件不存在!");
                }
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 上传文件到FTP服务器
        /// </summary>
        /// <param name="FileBytes">上传的二进制数据</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        public bool UploadFile(byte[] FileBytes, string RemoteFileName)
        {
            if (!IsValidFileChars(RemoteFileName))
            {
                throw new Exception("非法文件名或目录名!");
            }
            return UploadFile(FileBytes, RemoteFileName, false);
        }
        /// <summary>
        /// 上传文件到FTP服务器
        /// </summary>
        /// <param name="FileBytes">文件二进制内容</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        /// <param name="OverWriteRemoteFile">是否覆盖远程服务器上面同名的文件</param>
        public bool UploadFile(byte[] FileBytes, string RemoteFileName, bool OverWriteRemoteFile)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName))
                {
                    throw new Exception("非法文件名!");
                }
                if (!OverWriteRemoteFile && FileExist(RemoteFileName))
                {
                    throw new Exception("FTP服务上面已经存在同名文件!");
                }
                OpenRequest(new Uri(this.Uri.ToString() + RemoteFileName), WebRequestMethods.Ftp.UploadFile);
                Stream requestStream = Request.GetRequestStream();
                MemoryStream mem = new MemoryStream(FileBytes);

                byte[] buffer = new byte[1024];
                int bytesRead = 0;
                int TotalRead = 0;
                while (true)
                {
                    bytesRead = mem.Read(buffer, 0, buffer.Length);
                    if (bytesRead == 0)
                        break;
                    TotalRead += bytesRead;
                    requestStream.Write(buffer, 0, bytesRead);
                }
                requestStream.Close();
                Response = (FtpWebResponse)Request.GetResponse();
                mem.Close();
                mem.Dispose();
                FileBytes = null;
                return true;
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 异步上传文件
        /// <summary>
        /// 异步上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件名</param>
        public void UploadFileAsync(string LocalFullPath)
        {
            UploadFileAsync(LocalFullPath, Path.GetFileName(LocalFullPath), false);
        }
        /// <summary>
        /// 异步上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件</param>
        /// <param name="OverWriteRemoteFile">是否覆盖远程服务器上面同名的文件</param>
        public void UploadFileAsync(string LocalFullPath, bool OverWriteRemoteFile)
        {
            UploadFileAsync(LocalFullPath, Path.GetFileName(LocalFullPath), OverWriteRemoteFile);
        }
        /// <summary>
        /// 异步上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        public void UploadFileAsync(string LocalFullPath, string RemoteFileName)
        {
            UploadFileAsync(LocalFullPath, RemoteFileName, false);
        }
        /// <summary>
        /// 异步上传文件到FTP服务器
        /// </summary>
        /// <param name="LocalFullPath">本地带有完整路径的文件名</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        /// <param name="OverWriteRemoteFile">是否覆盖远程服务器上面同名的文件</param>
        public void UploadFileAsync(string LocalFullPath, string RemoteFileName, bool OverWriteRemoteFile)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName) || !IsValidFileChars(Path.GetFileName(LocalFullPath)) || !IsValidPathChars(Path.GetDirectoryName(LocalFullPath)))
                {
                    throw new Exception("非法文件名或目录名!");
                }
                if (!OverWriteRemoteFile && FileExist(RemoteFileName))
                {
                    throw new Exception("FTP服务上面已经存在同名文件!");
                }
                if (File.Exists(LocalFullPath))
                {
                    MyWebClient client = new MyWebClient();

                    client.UploadProgressChanged += new UploadProgressChangedEventHandler(client_UploadProgressChanged);
                    client.UploadFileCompleted += new UploadFileCompletedEventHandler(client_UploadFileCompleted);
                    client.Credentials = new NetworkCredential(this.UserName, this.Password);
                    if (this.Proxy != null)
                    {
                        client.Proxy = this.Proxy;
                    }
                    client.UploadFileAsync(new Uri(this.Uri.ToString() + RemoteFileName), LocalFullPath);

                }
                else
                {
                    throw new Exception("本地文件不存在!");
                }
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 异步上传文件到FTP服务器
        /// </summary>
        /// <param name="FileBytes">上传的二进制数据</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        public void UploadFileAsync(byte[] FileBytes, string RemoteFileName)
        {
            if (!IsValidFileChars(RemoteFileName))
            {
                throw new Exception("非法文件名或目录名!");
            }
            UploadFileAsync(FileBytes, RemoteFileName, false);
        }
        /// <summary>
        /// 异步上传文件到FTP服务器
        /// </summary>
        /// <param name="FileBytes">文件二进制内容</param>
        /// <param name="RemoteFileName">要在FTP服务器上面保存文件名</param>
        /// <param name="OverWriteRemoteFile">是否覆盖远程服务器上面同名的文件</param>
        public void UploadFileAsync(byte[] FileBytes, string RemoteFileName, bool OverWriteRemoteFile)
        {
            try
            {

                if (!IsValidFileChars(RemoteFileName))
                {
                    throw new Exception("非法文件名!");
                }
                if (!OverWriteRemoteFile && FileExist(RemoteFileName))
                {
                    throw new Exception("FTP服务上面已经存在同名文件!");
                }
                string TempPath = System.Environment.GetFolderPath(Environment.SpecialFolder.Templates);
                if (!TempPath.EndsWith("\\"))
                {
                    TempPath += "\\";
                }
                string TempFile = TempPath + Path.GetRandomFileName();
                TempFile = Path.ChangeExtension(TempFile, Path.GetExtension(RemoteFileName));
                FileStream Stream = new FileStream(TempFile, FileMode.CreateNew, FileAccess.Write);
                Stream.Write(FileBytes, 0, FileBytes.Length);   //注意,因为Int32的最大限制,最大上传文件只能是大约2G多一点
                Stream.Flush();
                Stream.Close();
                Stream.Dispose();
                _isDeleteTempFile = true;
                _UploadTempFile = TempFile;
                FileBytes = null;
                UploadFileAsync(TempFile, RemoteFileName, OverWriteRemoteFile);



            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }

        /// <summary>
        /// 异步上传文件完成之后触发的事件
        /// </summary>
        /// <param name="sender">下载对象</param>
        /// <param name="e">数据信息对象</param>
        void client_UploadFileCompleted(object sender, UploadFileCompletedEventArgs e)
        {
            if (_isDeleteTempFile)
            {
                if (File.Exists(_UploadTempFile))
                {
                    File.SetAttributes(_UploadTempFile, FileAttributes.Normal);
                    File.Delete(_UploadTempFile);
                }
                _isDeleteTempFile = false;
            }
            if (UploadFileCompleted != null)
            {
                UploadFileCompleted(sender, e);
            }
        }

        /// <summary>
        /// 异步上传进度发生改变触发的事件
        /// </summary>
        /// <param name="sender">下载对象</param>
        /// <param name="e">进度信息对象</param>
        void client_UploadProgressChanged(object sender, UploadProgressChangedEventArgs e)
        {
            if (UploadProgressChanged != null)
            {
                UploadProgressChanged(sender, e);
            }
        }
        #endregion


        #region 列出目录文件信息
        /// <summary>
        /// 列出FTP服务器上面当前目录的所有文件和目录
        /// </summary>
        public FileStruct[] ListFilesAndDirectories()
        {
            Response = Open(this.Uri, WebRequestMethods.Ftp.ListDirectoryDetails);
            StreamReader stream = new StreamReader(Response.GetResponseStream(), Encoding.Default);
            string Datastring = stream.ReadToEnd();
            FileStruct[] list = GetList(Datastring);
            return list;
        }
        /// <summary>
        /// 列出FTP服务器上面当前目录的所有文件
        /// </summary>
        public FileStruct[] ListFiles()
        {
            FileStruct[] listAll = ListFilesAndDirectories();
            List<FileStruct> listFile = new List<FileStruct>();
            foreach (FileStruct file in listAll)
            {
                if (!file.IsDirectory)
                {
                    listFile.Add(file);
                }
            }
            return listFile.ToArray();
        }

        /// <summary>
        /// 列出FTP服务器上面当前目录的所有的目录
        /// </summary>
        public FileStruct[] ListDirectories()
        {
            FileStruct[] listAll = ListFilesAndDirectories();
            List<FileStruct> listDirectory = new List<FileStruct>();
            foreach (FileStruct file in listAll)
            {
                if (file.IsDirectory)
                {
                    listDirectory.Add(file);
                }
            }
            return listDirectory.ToArray();
        }
        /// <summary>
        /// 获得文件和目录列表
        /// </summary>
        /// <param name="datastring">FTP返回的列表字符信息</param>
        private FileStruct[] GetList(string datastring)
        {
            List<FileStruct> myListArray = new List<FileStruct>();
            string[] dataRecords = datastring.Split('\n');
            FileListStyle _directoryListStyle = GuessFileListStyle(dataRecords);
            foreach (string s in dataRecords)
            {
                if (_directoryListStyle != FileListStyle.Unknown && s != "")
                {
                    FileStruct f = new FileStruct();
                    f.Name = "..";
                    switch (_directoryListStyle)
                    {
                        case FileListStyle.UnixStyle:
                            f = ParseFileStructFromUnixStyleRecord(s);
                            break;
                        case FileListStyle.WindowsStyle:
                            f = ParseFileStructFromWindowsStyleRecord(s);
                            break;
                    }
                    if (!(f.Name == "." || f.Name == ".."))
                    {
                        myListArray.Add(f);
                    }
                }
            }
            return myListArray.ToArray();
        }

        /// <summary>
        /// 从Windows格式中返回文件信息
        /// </summary>
        /// <param name="Record">文件信息</param>
        private FileStruct ParseFileStructFromWindowsStyleRecord(string Record)
        {
            FileStruct f = new FileStruct();
            string processstr = Record.Trim();
            string dateStr = processstr.Substring(0, 8);
            processstr = (processstr.Substring(8, processstr.Length - 8)).Trim();
            string timeStr = processstr.Substring(0, 7);
            processstr = (processstr.Substring(7, processstr.Length - 7)).Trim();
            DateTimeFormatInfo myDTFI = new CultureInfo("en-US", false).DateTimeFormat;
            myDTFI.ShortTimePattern = "t";
            f.CreateTime = DateTime.Parse(dateStr + " " + timeStr, myDTFI);
            if (processstr.Substring(0, 5) == "<DIR>")
            {
                f.IsDirectory = true;
                processstr = (processstr.Substring(5, processstr.Length - 5)).Trim();
            }
            else
            {
                string[] strs = processstr.Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries);   // true);
                processstr = strs[1];
                f.IsDirectory = false;
            }
            f.Name = processstr;
            return f;
        }


        /// <summary>
        /// 判断文件列表的方式Window方式还是Unix方式
        /// </summary>
        /// <param name="recordList">文件信息列表</param>
        private FileListStyle GuessFileListStyle(string[] recordList)
        {
            foreach (string s in recordList)
            {
                if (s.Length > 10
                 && Regex.IsMatch(s.Substring(0, 10), "(-|d)(-|r)(-|w)(-|x)(-|r)(-|w)(-|x)(-|r)(-|w)(-|x)"))
                {
                    return FileListStyle.UnixStyle;
                }
                else if (s.Length > 8
                 && Regex.IsMatch(s.Substring(0, 8), "[0-9][0-9]-[0-9][0-9]-[0-9][0-9]"))
                {
                    return FileListStyle.WindowsStyle;
                }
            }
            return FileListStyle.Unknown;
        }

        /// <summary>
        /// 从Unix格式中返回文件信息
        /// </summary>
        /// <param name="Record">文件信息</param>
        private FileStruct ParseFileStructFromUnixStyleRecord(string Record)
        {
            FileStruct f = new FileStruct();
            string processstr = Record.Trim();
            f.Flags = processstr.Substring(0, 10);
            f.IsDirectory = (f.Flags[0] == 'd');
            processstr = (processstr.Substring(11)).Trim();
            _cutSubstringFromStringWithTrim(ref processstr, ' ', 0);   //跳过一部分
            f.Owner = _cutSubstringFromStringWithTrim(ref processstr, ' ', 0);
            f.Group = _cutSubstringFromStringWithTrim(ref processstr, ' ', 0);
            _cutSubstringFromStringWithTrim(ref processstr, ' ', 0);   //跳过一部分
            string yearOrTime = processstr.Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries)[2];
            if (yearOrTime.IndexOf(":") >= 0)  //time
            {
                processstr = processstr.Replace(yearOrTime, DateTime.Now.Year.ToString());
            }
            f.CreateTime = DateTime.Parse(_cutSubstringFromStringWithTrim(ref processstr, ' ', 8));
            f.Name = processstr;   //最后就是名称
            return f;
        }

        /// <summary>
        /// 按照一定的规则进行字符串截取
        /// </summary>
        /// <param name="s">截取的字符串</param>
        /// <param name="c">查找的字符</param>
        /// <param name="startIndex">查找的位置</param>
        private string _cutSubstringFromStringWithTrim(ref string s, char c, int startIndex)
        {
            int pos1 = s.IndexOf(c, startIndex);
            string retString = s.Substring(0, pos1);
            s = (s.Substring(pos1)).Trim();
            return retString;
        }
        #endregion


        #region 目录或文件存在的判断
        /// <summary>
        /// 判断当前目录下指定的子目录是否存在
        /// </summary>
        /// <param name="RemoteDirectoryName">指定的目录名</param>
        public bool DirectoryExist(string RemoteDirectoryName)
        {
            try
            {
                if (!IsValidPathChars(RemoteDirectoryName))
                {
                    throw new Exception("目录名非法!");
                }
                FileStruct[] listDir = ListDirectories();
                foreach (FileStruct dir in listDir)
                {
                    if (dir.Name == RemoteDirectoryName)
                    {
                        return true;
                    }
                }
                return false;
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 判断一个远程文件是否存在服务器当前目录下面
        /// </summary>
        /// <param name="RemoteFileName">远程文件名</param>
        public bool FileExist(string RemoteFileName)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName))
                {
                    throw new Exception("文件名非法!");
                }
                FileStruct[] listFile = ListFiles();
                foreach (FileStruct file in listFile)
                {
                    if (file.Name == RemoteFileName)
                    {
                        return true;
                    }
                }
                return false;
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 删除文件
        /// <summary>
        /// 从FTP服务器上面删除一个文件
        /// </summary>
        /// <param name="RemoteFileName">远程文件名</param>
        public void DeleteFile(string RemoteFileName)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName))
                {
                    throw new Exception("文件名非法!");
                }
                Response = Open(new Uri(this.Uri.ToString() + RemoteFileName), WebRequestMethods.Ftp.DeleteFile);
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 重命名文件
        /// <summary>
        /// 更改一个文件的名称或一个目录的名称
        /// </summary>
        /// <param name="RemoteFileName">原始文件或目录名称</param>
        /// <param name="NewFileName">新的文件或目录的名称</param>
        public bool ReName(string RemoteFileName, string NewFileName)
        {
            try
            {
                if (!IsValidFileChars(RemoteFileName) || !IsValidFileChars(NewFileName))
                {
                    throw new Exception("文件名非法!");
                }
                if (RemoteFileName == NewFileName)
                {
                    return true;
                }
                if (FileExist(RemoteFileName))
                {
                    Request = OpenRequest(new Uri(this.Uri.ToString() + RemoteFileName), WebRequestMethods.Ftp.Rename);
                    Request.RenameTo = NewFileName;
                    Response = (FtpWebResponse)Request.GetResponse();

                }
                else
                {
                    throw new Exception("文件在服务器上不存在!");
                }
                return true;
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 拷贝、移动文件
        /// <summary>
        /// 把当前目录下面的一个文件拷贝到服务器上面另外的目录中,注意,拷贝文件之后,当前工作目录还是文件原来所在的目录
        /// </summary>
        /// <param name="RemoteFile">当前目录下的文件名</param>
        /// <param name="DirectoryName">新目录名称。
        /// 说明:如果新目录是当前目录的子目录,则直接指定子目录。如: SubDirectory1/SubDirectory2 ;
        /// 如果新目录不是当前目录的子目录,则必须从根目录一级一级的指定。如: ./NewDirectory/SubDirectory1/SubDirectory2
        /// </param>
        /// <returns></returns>
        public bool CopyFileToAnotherDirectory(string RemoteFile, string DirectoryName)
        {
            string CurrentWorkDir = this.DirectoryPath;
            try
            {
                byte[] bt = DownloadFile(RemoteFile);
                GotoDirectory(DirectoryName);
                bool Success = UploadFile(bt, RemoteFile, false);
                this.DirectoryPath = CurrentWorkDir;
                return Success;
            }
            catch (Exception ep)
            {
                this.DirectoryPath = CurrentWorkDir;
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 把当前目录下面的一个文件移动到服务器上面另外的目录中,注意,移动文件之后,当前工作目录还是文件原来所在的目录
        /// </summary>
        /// <param name="RemoteFile">当前目录下的文件名</param>
        /// <param name="DirectoryName">新目录名称。
        /// 说明:如果新目录是当前目录的子目录,则直接指定子目录。如: SubDirectory1/SubDirectory2 ;
        /// 如果新目录不是当前目录的子目录,则必须从根目录一级一级的指定。如: ./NewDirectory/SubDirectory1/SubDirectory2
        /// </param>
        /// <returns></returns>
        public bool MoveFileToAnotherDirectory(string RemoteFile, string DirectoryName)
        {
            string CurrentWorkDir = this.DirectoryPath;
            try
            {
                if (DirectoryName == "")
                    return false;
                if (!DirectoryName.StartsWith("/"))
                    DirectoryName = "/" + DirectoryName;
                if (!DirectoryName.EndsWith("/"))
                    DirectoryName += "/";
                bool Success = ReName(RemoteFile, DirectoryName + RemoteFile);
                this.DirectoryPath = CurrentWorkDir;
                return Success;
            }
            catch (Exception ep)
            {
                this.DirectoryPath = CurrentWorkDir;
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 建立、删除子目录
        /// <summary>
        /// 在FTP服务器上当前工作目录建立一个子目录
        /// </summary>
        /// <param name="DirectoryName">子目录名称</param>
        public bool MakeDirectory(string DirectoryName)
        {
            try
            {
                if (!IsValidPathChars(DirectoryName))
                {
                    throw new Exception("目录名非法!");
                }
                if (DirectoryExist(DirectoryName))
                {
                    throw new Exception("服务器上面已经存在同名的文件名或目录名!");
                }
                Response = Open(new Uri(this.Uri.ToString() + DirectoryName), WebRequestMethods.Ftp.MakeDirectory);
                return true;
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 从当前工作目录中删除一个子目录
        /// </summary>
        /// <param name="DirectoryName">子目录名称</param>
        public bool RemoveDirectory(string DirectoryName)
        {
            try
            {
                if (!IsValidPathChars(DirectoryName))
                {
                    throw new Exception("目录名非法!");
                }
                if (!DirectoryExist(DirectoryName))
                {
                    throw new Exception("服务器上面不存在指定的文件名或目录名!");
                }
                Response = Open(new Uri(this.Uri.ToString() + DirectoryName), WebRequestMethods.Ftp.RemoveDirectory);
                return true;
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        #endregion


        #region 文件、目录名称有效性判断
        /// <summary>
        /// 判断目录名中字符是否合法
        /// </summary>
        /// <param name="DirectoryName">目录名称</param>
        public bool IsValidPathChars(string DirectoryName)
        {
            char[] invalidPathChars = Path.GetInvalidPathChars();
            char[] DirChar = DirectoryName.ToCharArray();
            foreach (char C in DirChar)
            {
                if (Array.BinarySearch(invalidPathChars, C) >= 0)
                {
                    return false;
                }
            }
            return true;
        }
        /// <summary>
        /// 判断文件名中字符是否合法
        /// </summary>
        /// <param name="FileName">文件名称</param>
        public bool IsValidFileChars(string FileName)
        {
            char[] invalidFileChars = Path.GetInvalidFileNameChars();
            char[] NameChar = FileName.ToCharArray();
            foreach (char C in NameChar)
            {
                if (Array.BinarySearch(invalidFileChars, C) >= 0)
                {
                    return false;
                }
            }
            return true;
        }
        #endregion


        #region 目录切换操作
        /// <summary>
        /// 进入一个目录
        /// </summary>
        /// <param name="DirectoryName">
        /// 新目录的名字。 
        /// 说明:如果新目录是当前目录的子目录,则直接指定子目录。如: SubDirectory1/SubDirectory2 ; 
        /// 如果新目录不是当前目录的子目录,则必须从根目录一级一级的指定。如: ./NewDirectory/SubDirectory1/SubDirectory2
        /// </param>
        public bool GotoDirectory(string DirectoryName)
        {
            string CurrentWorkPath = this.DirectoryPath;
            try
            {
                DirectoryName = DirectoryName.Replace("\\", "/");
                string[] DirectoryNames = DirectoryName.Split(new char[] { '/' });
                if (DirectoryNames[0] == ".")
                {
                    this.DirectoryPath = "/";
                    if (DirectoryNames.Length == 1)
                    {
                        return true;
                    }
                    Array.Clear(DirectoryNames, 0, 1);
                }
                bool Success = false;
                foreach (string dir in DirectoryNames)
                {
                    if (dir != null)
                    {
                        Success = EnterOneSubDirectory(dir);
                        if (!Success)
                        {
                            this.DirectoryPath = CurrentWorkPath;
                            return false;
                        }
                    }
                }
                return Success;

            }
            catch (Exception ep)
            {
                this.DirectoryPath = CurrentWorkPath;
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 从当前工作目录进入一个子目录
        /// </summary>
        /// <param name="DirectoryName">子目录名称</param>
        private bool EnterOneSubDirectory(string DirectoryName)
        {
            try
            {
                if (DirectoryName.IndexOf("/") >= 0 || !IsValidPathChars(DirectoryName))
                {
                    throw new Exception("目录名非法!");
                }
                if (DirectoryName.Length > 0 && DirectoryExist(DirectoryName))
                {
                    if (!DirectoryName.EndsWith("/"))
                    {
                        DirectoryName += "/";
                    }
                    _DirectoryPath += DirectoryName;
                    return true;
                }
                else
                {
                    return false;
                }
            }
            catch (Exception ep)
            {
                ErrorMsg = ep.ToString();
                throw ep;
            }
        }
        /// <summary>
        /// 从当前工作目录往上一级目录
        /// </summary>
        public bool ComeoutDirectory()
        {
            if (_DirectoryPath == "/")
            {
                ErrorMsg = "当前目录已经是根目录!";
                throw new Exception("当前目录已经是根目录!");
            }
            char[] sp = new char[1] { '/' };

            string[] strDir = _DirectoryPath.Split(sp, StringSplitOptions.RemoveEmptyEntries);
            if (strDir.Length == 1)
            {
                _DirectoryPath = "/";
            }
            else
            {
                _DirectoryPath = String.Join("/", strDir, 0, strDir.Length - 1);
            }
            return true;

        }
        #endregion


        #region 重载WebClient,支持FTP进度
        internal class MyWebClient : WebClient
        {
            protected override WebRequest GetWebRequest(Uri address)
            {
                FtpWebRequest req = (FtpWebRequest)base.GetWebRequest(address);
                req.UsePassive = false;
                return req;
            }
        }
        #endregion
    }
}

 使用示例:

Uri ftpUrl = new Uri("ftp://192.168.1.100/");
FTPHelper ftp = new FTPHelper(ftpUrl, "用户名", "密码");
FileStruct[] files = ftp.ListFiles();
foreach (FileStruct fs in files)
{
if (fs.Name == filename)
{
    MessageUtil.ShowTips("服务器中已经含有" + filename + "文件");
    break;
}
}
bool result = ftp.UploadFile(filePosition);

 

posted @ 2012-08-03 19:08  MagiCube  阅读(5505)  评论(0)    收藏  举报