当前位置: 首页 > news >正文

[翻译].net 2.0(c#)下简单的FTP应用程序(转)

转自:http://www.cnblogs.com/webabcd/archive/2007/01/21/626242.aspx#726261

 

原文地址:http://www.c-sharpcorner.com/UploadFile/neo_matrix/SimpleFTP01172007082222AM/SimpleFTP.aspx
[原文源码下载]


.net 2.0(c#)下简单的FTP应用程序



原文发布日期:2007.01.18
作者:Neo Matrix
翻译:webabcd


本文使用.net 2.0(c#)来实现一般的FTP功能


介绍
微软的.net framework 2.0相对于1.x来说增加了对FTP的支持。以前为了符合我的需求,我不等不使用第三方类库来实现FTP功能,但是为了可靠,还是使用.net framework的类比较好。我的这段代码没有做成可重复使用的类库的形式,但它却是比较容易理解的并能满足你的需求。它可以实现上传,下载,删除等任意功能。在这篇文章的后面将给大家出示.net 2.0下实现ftp的简单代码,使用的语言是c#。或许是因为这是.net新增的类,又或许是第三方类库已经能很好的实现你的需求,.net 2.0的这部分类库并没有得到足够的关注。


背景
作为我的工作的一部分,我已经使用了ftp模块,但是我只能在.net 1.1中去使用它,所以我不能深入的研究.net 2.0下ftp的实现。但是我相信,.ne 2.0下对ftp的支持是非常好的。


代码
不要忘记引入命名空间

using  System.Net;
using  System.IO;


下面的几个步骤包括了使用FtpWebRequest类实现ftp功能的一般过程

1、创建一个FtpWebRequest对象,指向ftp服务器的uri
2、设置ftp的执行方法(上传,下载等)
3、给FtpWebRequest对象设置属性(是否支持ssl,是否使用二进制传输等)
4、设置登录验证(用户名,密码)
5、执行请求
6、接收相应流(如果需要的话)
7、如果没有打开的流,则关闭ftp请求

开发任何ftp应用程序都需要一个相关的ftp服务器及它的配置信息。FtpWebRequest暴露了一些属性来设置这些信息。

接下来的代码示例了上传功能

首先设置一个uri地址,包括路径和文件名。这个uri被使用在FtpWebRequest实例中。

然后根据ftp请求设置FtpWebRequest对象的属性

其中一些重要的属性如下:
    ·Credentials - 指定登录ftp服务器的用户名和密码。
    ·KeepAlive - 指定连接是应该关闭还是在请求完成之后关闭,默认为true
    ·UseBinary - 指定文件传输的类型。有两种文件传输模式,一种是Binary,另一种是ASCII。两种方法在传输时,字节的第8位是不同的。ASCII使用第8位作为错误控制,而Binary的8位都是有意义的。所以当你使用ASCII传输时要小心一些。简单的说,如果能用记事本读和写的文件用ASCII传输就是安全的,而其他的则必须使用Binary模式。当然使用Binary模式发送ASCII文件也是非常好的。
    ·UsePassive - 指定使用主动模式还是被动模式。早先所有客户端都使用主动模式,而且工作的很好,而现在因为客户端防火墙的存在,将会关闭一些端口,这样主动模式将会失败。在这种情况下就要使用被动模式,但是一些端口也可能被服务器的防火墙封掉。不过因为ftp服务器需要它的ftp服务连接到一定数量的客户端,所以他们总是支持被动模式的。这就是我们为什么要使用被动模式的原意,为了确保数据可以正确的传输,使用被动模式要明显优于主动模式。(译者注:主动(PORT)模式建立数据传输通道是由服务器端发起的,服务器使用20端口连接客户端的某一个大于1024的端口;在被动(PASV)模式中,数据传输的通道的建立是由FTP客户端发起的,他使用一个大于1024的端口连接服务器的1024以上的某一个端口)
    ·ContentLength - 设置这个属性对于ftp服务器是有用的,但是客户端不使用它,因为FtpWebRequest忽略这个属性,所以在这种情况下,该属性是无效的。但是如果我们设置了这个属性,ftp服务器将会提前预知文件的大小(在upload时会有这种情况)
    ·Method - 指定当前请求是什么命令(upload,download,filelist等)。这个值定义在结构体WebRequestMethods.Ftp中。

private   void  Upload( string  filename)
ExpandedBlockStart.gifContractedBlock.gif
{
    FileInfo fileInf 
= new FileInfo(filename);
    
string uri = "ftp://" + ftpServerIP + "/" + fileInf.Name;
    FtpWebRequest reqFTP; 

    
// 根据uri创建FtpWebRequest对象 
    reqFTP = (FtpWebRequest)FtpWebRequest.Create(new Uri("ftp://" + ftpServerIP + "/" + fileInf.Name)); 

    
// ftp用户名和密码
    reqFTP.Credentials = new NetworkCredential(ftpUserID, ftpPassword); 

    
// 默认为true,连接不会被关闭
    
// 在一个命令之后被执行
    reqFTP.KeepAlive = false

    
// 指定执行什么命令
    reqFTP.Method = WebRequestMethods.Ftp.UploadFile; 

    
// 指定数据传输类型
    reqFTP.UseBinary = true

    
// 上传文件时通知服务器文件的大小
    reqFTP.ContentLength = fileInf.Length; 

    
// 缓冲大小设置为2kb
    int buffLength = 2048;

    
byte[] buff = new byte[buffLength];
    
int contentLen; 

    
// 打开一个文件流 (System.IO.FileStream) 去读上传的文件
    FileStream fs = fileInf.OpenRead(); 
    
try
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        
// 把上传的文件写入流
        Stream strm = reqFTP.GetRequestStream(); 

        
// 每次读文件流的2kb
        contentLen = fs.Read(buff, 0, buffLength); 

        
// 流内容没有结束
        while (contentLen != 0)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            
// 把内容从file stream 写入 upload stream
            strm.Write(buff, 0, contentLen);

            contentLen 
= fs.Read(buff, 0, buffLength);
        }
 

        
// 关闭两个流
        strm.Close();
        fs.Close();
    }

    
catch (Exception ex)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        MessageBox.Show(ex.Message, 
"Upload Error");
    }

}


以上代码简单的示例了ftp的上传功能。创建一个指向某ftp服务器的FtpWebRequest对象,然后设置其不同的属性Credentials,KeepAlive,Method,UseBinary,ContentLength。

打开本地机器上的文件,把其内容写入ftp请求流。缓冲的大小为2kb,无论上传大文件还是小文件,这都是一个合适的大小。

private   void  Download( string  filePath,  string  fileName)
ExpandedBlockStart.gifContractedBlock.gif
{
    FtpWebRequest reqFTP;

    
try
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        FileStream outputStream 
= new FileStream(filePath + "\\" + fileName, FileMode.Create); 

        reqFTP 
= (FtpWebRequest)FtpWebRequest.Create(new Uri("ftp://" + ftpServerIP + "/" + fileName));

        reqFTP.Method 
= WebRequestMethods.Ftp.DownloadFile;

        reqFTP.UseBinary 
= true;

        reqFTP.Credentials 
= new NetworkCredential(ftpUserID, ftpPassword);

        FtpWebResponse response 
= (FtpWebResponse)reqFTP.GetResponse();

        Stream ftpStream 
= response.GetResponseStream();

        
long cl = response.ContentLength;

        
int bufferSize = 2048;

        
int readCount;

        
byte[] buffer = new byte[bufferSize];

        readCount 
= ftpStream.Read(buffer, 0, bufferSize);

        
while (readCount > 0)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{
            outputStream.Write(buffer, 
0, readCount);

            readCount 
= ftpStream.Read(buffer, 0, bufferSize);
        }


        ftpStream.Close();

        outputStream.Close();

        response.Close();
    }

    
catch (Exception ex)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{
        MessageBox.Show(ex.Message);
    }

}


上面的代码实现了从ftp服务器上下载文件的功能。这不同于之前所提到的上传功能,下载需要一个响应流,它包含着下载文件的内容。这个下载的文件是在FtpWebRequest对象中的uri指定的。在得到所请求的文件后,通过FtpWebRequest对象的GetResponse()方法下载文件。它将把文件作为一个流下载到你的客户端的机器上。

注意:我们可以设置文件在我们本地机器上的存放路径和名称。

public   string [] GetFileList()
ExpandedBlockStart.gifContractedBlock.gif
{    
    
string[] downloadFiles;    
    StringBuilder result 
= new StringBuilder();    
    FtpWebRequest reqFTP;    
    
try    
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{        
        reqFTP 
= (FtpWebRequest)FtpWebRequest.Create(new Uri("ftp://" + ftpServerIP + "/"));        
        reqFTP.UseBinary 
= true;        
        reqFTP.Credentials 
= new NetworkCredential(ftpUserID, ftpPassword);        
        reqFTP.Method 
= WebRequestMethods.Ftp.ListDirectory;        
        WebResponse response 
= reqFTP.GetResponse();        
        StreamReader reader 
= new StreamReader(response.GetResponseStream());        
        
string line = reader.ReadLine();        
        
while (line != null)        
ExpandedSubBlockStart.gifContractedSubBlock.gif        
{            
            result.Append(line);            
            result.Append(
"\n");            
            line 
= reader.ReadLine();        
        }
        
        
// to remove the trailing '\n'        
        result.Remove(result.ToString().LastIndexOf('\n'), 1);        
        reader.Close();        
        response.Close();        
        
return result.ToString().Split('\n');    
    }
    
    
catch (Exception ex)    
ExpandedSubBlockStart.gifContractedSubBlock.gif    
{        
        System.Windows.Forms.MessageBox.Show(ex.Message);        
        downloadFiles 
= null;        
        
return downloadFiles;    
    }

}


上面的代码示例了如何从ftp服务器上获得文件列表。uri指向ftp服务器的地址。我们使用StreamReader对象来存储一个流,文件名称列表通过“\r\n”分隔开,也就是说每一个文件名称都占一行。你可以使用StreamReader对象的ReadToEnd()方法来得到文件列表。上面的代码中我们用一个StringBuilder对象来保存文件名称,然后把结果通过分隔符分开后作为一个数组返回。我确定只是一个比较好的方法。

其他的实现如Rename,Delete,GetFileSize,FileListDetails,MakeDir等与上面的几段代码类似,就不多说了。

注意:实现重命名的功能时,要把新的名字设置给FtpWebRequest对象的RenameTo属性。连接指定目录的时候,需要在FtpWebRequest对象所使用的uri中指明。


需要注意的地方
你在编码时需要注意以下几点:
    ·除非EnableSsl属性被设置成true,否作所有数据,包括你的用户名和密码都将明文发给服务器,任何监视网络的人都可以获取到你连接服务器的验证信息。如果你连接的ftp服务器提供了SSL,你就应当把EnableSsl属性设置为true。
    ·如果你没有访问ftp服务器的权限,将会抛出SecurityException错误
    ·发送请求到ftp服务器需要调用GetResponse方法。当请求的操作完成后,一个FtpWebResponse对象将返回。这个FtpWebResponse对象提供了操作的状态和已经从ftp服务器上下载的数据。FtpWebResponse对象的StatusCode属性提供了ftp服务器返回的最后的状态代码。FtpWebResponse对象的StatusDescription属性为这个状态代码的描述。

转载于:https://www.cnblogs.com/ziyouyong/archive/2009/11/11/1601160.html

相关文章:

  • Python Unicode 转换 字符串
  • java中关于、、|、||之间的区别和运算
  • 生成静态页面的方法
  • 项目质量量化考核建议
  • Linux命令(磁盘的卸载与挂载)
  • 手势识别论文解读
  • 用于在弹出窗口中计算窗口位置的函数
  • 《领域特定语言》一2.2为何需要DSL
  • 关注数据中心NFV性能
  • 取页面内各种尽寸数据。备档
  • Kotlin和Java EE系列之—— 如何让Kotlin类对Java EE友好
  • 多个项目共享一个app.config 的解决办法
  • 程序员每周编程80+小时 健康如何保证
  • 网站性能愈来愈差怎么办?
  • 《C语言程序设计:问题与求解方法》——第2章C语言程序结构和基本语法要素...
  • 【刷算法】求1+2+3+...+n
  • 3.7、@ResponseBody 和 @RestController
  • C++入门教程(10):for 语句
  • IDEA常用插件整理
  • iOS编译提示和导航提示
  • jquery ajax学习笔记
  • mysql innodb 索引使用指南
  • React组件设计模式(一)
  • RedisSerializer之JdkSerializationRedisSerializer分析
  • vue 配置sass、scss全局变量
  • 得到一个数组中任意X个元素的所有组合 即C(n,m)
  • 等保2.0 | 几维安全发布等保检测、等保加固专版 加速企业等保合规
  • 翻译:Hystrix - How To Use
  • 规范化安全开发 KOA 手脚架
  • ------- 计算机网络基础
  • 利用jquery编写加法运算验证码
  • 每个JavaScript开发人员应阅读的书【1】 - JavaScript: The Good Parts
  • 前端 CSS : 5# 纯 CSS 实现24小时超市
  • 如何借助 NoSQL 提高 JPA 应用性能
  • 入手阿里云新服务器的部署NODE
  • 实习面试笔记
  • 它承受着该等级不该有的简单, leetcode 564 寻找最近的回文数
  • 通过git安装npm私有模块
  • 运行时添加log4j2的appender
  • 智能合约Solidity教程-事件和日志(一)
  • AI算硅基生命吗,为什么?
  • 湖北分布式智能数据采集方法有哪些?
  • 整理一些计算机基础知识!
  • ​secrets --- 生成管理密码的安全随机数​
  • #pragma 指令
  • #我与Java虚拟机的故事#连载11: JVM学习之路
  • ${factoryList }后面有空格不影响
  • (3)Dubbo启动时qos-server can not bind localhost22222错误解决
  • (Bean工厂的后处理器入门)学习Spring的第七天
  • (八)五种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (附源码)spring boot公选课在线选课系统 毕业设计 142011
  • (篇九)MySQL常用内置函数
  • (十六)Flask之蓝图
  • (五)MySQL的备份及恢复
  • (转载)跟我一起学习VIM - The Life Changing Editor