`
piperzero
  • 浏览: 3555283 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
文章分类
社区版块
存档分类
最新评论

[Perl多线程]自动上传FTP文件的多线程扩展

阅读更多
<iframe align="top" marginwidth="0" marginheight="0" src="http://www.zealware.com/csdnblog01.html" frameborder="0" width="728" scrolling="no" height="90"></iframe>

Perl脚本多线程上传FTP文件

<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

本脚本是对《[Perl]FTP自动上传文件的脚本以及配置文件》的多线程扩展,当然首先对方FTP站点允许同一个IP发起多个连接。

ithreads支持

Perl<?xml:namespace prefix = st1 ns = "urn:schemas-microsoft-com:office:smarttags" /><chsdate year="1899" month="12" day="30" islunardate="False" isrocdate="False" w:st="on">5.6.0</chsdate>已经加入了ithreads支持,我们通过

use threads;

导入threads多线程处理包;并且通过

use threads::shared;

使用线程间共享变量。

在定义全局并线程间共享的变量时,要这样写:

my $CurrentThreads: shared = 0; #当前线程总数

因为在Perl的实现中,线程并不是像pthread那样共享变量,而是大家都分开,如同原来的进程一样。如果想让一个变量共享,须要显式地指定它才行。

当然我们还要限制并发活动线程数目,因为最多线程数是1-65perl最多允许64个子线程,加上主线程因此最多65个线程,而且FTP站点允许多大连接数。这里$MaxThreads设置是从配置文件读取的。

配置文件的读取我们是采用

use Config::IniFiles;

库的。它的读取很简单:

my $cfg = Config::IniFiles->new( -file => "FTPUpload.config" );

等待线程的完全退出

通过不断地调用

my $thread = threads->create('processFile', $srcpath, $dstpath, $dstdir);

启动了多干个线程后,我们这里一定要调用

push(@$self, \$thread);

因为,创建一个thread以后要用join取得该thread的返回值,然后系统才会对thread进行清理,否则所有thread的信息都会保留下来,当然越积越多了。对返回值不关心的时候要用detach显式剥离该thread

所以,在最后我们要等待这些线程的完全退出:

foreach my $thread (@$self)

{

print("Joining thread\n");

$$thread->join();

}

否则,threads库会在最后打印如下信息:

A thread exited while 2 threads were running.

程序大致的流程是:

第一步,尝试用配置信息登陆ftp站点,这只是验证能够登陆而已;

第二步,在指定文件夹A类下寻找符合条件的文件,并将对每一个A类文件创建一个线程上传。每个线程单独创建一个FTP连接,并不断地尝试上传这个文件直至成功;

第三步,如果A类文件们全部上传成功,那么在指定文件夹B类下寻找指定文件, 并且上传到FTP指定目录下

第四步,写成功/失败日志。

最后,我们要写的成功/失败日志的格式如下所示:

成功: 生成一个名为Upload_Succ_2005_01_04_17_23.log的日志文件

文件格式:输出上传时间,以及所有上传文件名及其大小和耗费的时间。

失败: 生成一个名为Upload_Fail_2005_01_04_17_23.log的日志文件

文件格式:输出上传时间,以及已经上传的文件名及其大小和耗费的时间,和失败的文件名及原因。

配置perl脚本运行有两个办法:

u 您可以在Windows计划任务中配置运行“Perl Upload.pl”的时间,这需要在Windows环境中配置ActivePerl <chsdate year="1899" month="12" day="30" islunardate="False" isrocdate="False" w:st="on"><b style="mso-bidi-font-weight: normal"><span lang="EN-US" style="FONT-SIZE: 10.5pt; COLOR: #666699; FONT-FAMILY: Arial; mso-bidi-font-size: 12.0pt">5.8.4</span></b></chsdate>.810

u 您也可以利用Perl2Exe(p2x-8.40-Win32)来将perl脚本编译为一个exe可执行程序,在计划任务中运行这个exe(这需要PerlCRT.dll在系统路径下)

[注意!]在运行之前,您必须修改“Upload.config”文件以配置所需的重要参数。

外部配置参数

在和perl脚本同一目录下的“Upload.config”配置文件中,是事先配置的10个外部参数:

# 参数1: ftp_server

# FTP服务器的IP地址。

# 参数2: ftp_dir:

# 指定的FTP上传目录路径;

# 参数3: ftp_uid:

# FTP的登陆用户名;

# 参数4: ftp_pw:

# FTP的登陆密码;

# 参数5: src_dir_WAVFiles,这是一个数组:

指定A类文件夹,放置所有要上传的语音文件;

注意:这个目录下的子文件夹也会被上传。

# 参数6: src_dir_NamesListFile,这是一个数组:

指定B类文件夹,放置B类文件. 注意:这个目录下的子文件夹也会被上传。

# 参数7: ftp_timeout:

# FTP的访问超时时间,以秒为单位;

# 参数8: ftp_retrytimes:

# FTP上传一个文件的重试次数,如果重试这么多次都失败之后,报告错误;

# 参数9: ftp_debug

# FTP的详细的调试信息输出

# 参数10: maxthreads

# 启动同时访问FTP站点的线程最大数目

附录:
Upoad.pl
内容:
#!/usr/bin/perl-w

####################################################################
#
#
工程项目: FTP自动上传两类文件
#
#
模块名称: FTPAutoUpload
#
#
模块任务: 按照指定的文件夹目录,自动将该文件夹下的所有文件上传到指定ftp站点的指定目录下
#
#
程序名称: Upload.pl
#
#
程序员: Uwe Keim
#
郑昀
#
历史记录:
#
编号 日期 作者 备注
# 1 2000 Uwe Keim
创建
# 2 2005.1.5
郑昀 加入容错处理和读取外部配置文件部分

# 3 2005.1.14 郑昀 加入多线程支持
#
####################################################################

#use strict;

##================================================================##

## 引用的库声明 1

## 读取ini配置文件的库

use Config::IniFiles;

my $cfg = Config::IniFiles->new( -file => "Upload.config" );

## 启动同时访问FTP站点的线程最大数目 ##

$config_maxthreads = $cfg->val('Threads', 'maxthreads') || 1;

##================================================================##

##================================================================##

## 引用的库声明 2

use File::Copy;

use File::stat;

use File::Find;

use Net::FTP;

use Date::Pcalc qw(Delta_DHMS);

use Date::Parse;

#!!use Win32::OLE; # 加入这个库会导致Perl执行线程时崩溃

#!!use Win32::OLE::Variant; # 加入这个库会导致Perl执行线程时崩溃

##================================================================##

##================================================================##

## 引用的库声明 3

# Perl<chsdate year="1899" month="12" day="30" islunardate="False" isrocdate="False" w:st="on">5.6.0</chsdate>已经加入了ithreads支持

use threads; #导入threads多线程处理包

use threads::shared; #使用线程间共享变量

my $MaxThreads = $config_maxthreads; #最多线程数(1-65),perl最多允许64个子线程,加上主线程因此最多65个线程

#此次设置是从配置文件读取的

# 在定义全局并线程间共享的变量时,要这样写

my $CurrentThreads: shared = 0; #当前线程总数

## $total_files是上传文件的数目

my $total_files: shared = 0;

## $processed_files是已上传文件的数目

my $processed_files: shared = 0;

## $skipped_files是跳过文件的数目

my $skipped_files: shared = 0;

## FTP重试次数

my $ftp_retrytimes: shared = 0;

## $g_nIsAllWAVsFile_UploadSuccess代表是否已经完全将语音文件上传,-1为不是,1为是:

my $g_nIsAllWAVsFile_UploadSuccess: shared = 1;

# 因为在Perl的实现中,线程并不是像pthread那样共享变量,而是大家都分开,如同原来的进程一样。

# 如果想让一个变量共享,须要显式地指定它才行。

#

##================================================================##

##================================================================##

## 从配置文件读取外部参数 ##

##

## FTP服务器的IP地址 ##

my $ftp_server = $cfg->val('FTPServer', 'ftp_server') || '';

## 指定的FTP上传目录路径 ##

#! 切记:文件夹最后不要加"/"符号 !#

my $ftp_dir = $cfg->val('FTPServer', 'ftp_dir') || '';

## FTP的登陆用户名 ##

my $ftp_uid = $cfg->val('FTPServer', 'ftp_uid') || '';

## FTP的登陆密码 ##

my $ftp_pw = $cfg->val('FTPServer', 'ftp_pw') || '';

## FTP的访问超时时间,以秒为单位,默认设置为30分钟 ##

my $ftp_timeout = $cfg->val('FTPServer', 'ftp_timeout') || 1800;

####$ftp_timeout = 1800;

## FTP上传一个文件的重试次数,如果重试这么多次都失败之后,报告错误 ##

$ftp_retrytimes = $cfg->val('FTPServer', 'ftp_retrytimes') || 3;

####$ftp_retrytimes = 3;

## FTP的详细的调试信息输出 ##

$ftp_debug = $cfg->val('FTPServer', 'ftp_debug') || 0;

####$ftp_debug = 0;

## 指定文件夹“语音文件”,放置所有要上传的语音文件 ##

#! 切记:文件夹最后不要加"\\"符号 !#

my @src_dir_WAVFiles = $cfg->val('SrcDirectory', 'src_dir_WAVFiles');

## 指定文件夹“命名对照列表文件TXT”,放置命名对照列表文件 ##

#! 切记:文件夹最后不要加"\\"符号 !#

my @src_dir_NamesListFile = $cfg->val('SrcDirectory', 'src_dir_NamesListFile');

## 一个字符串集合,表明哪些类型的文件/文件夹将不被上传到服务器 ##

my @wc_exclude = ("_vti",".log","\\bak","\\data","server.inc");

##================================================================##

##================================================================##

## 记录全部过程的日志文件准备

my $logfilename = 'upload.log';

my $log_cnt = 0;

my $span = 0;

LOG("");

LOG("郑昀(R) 自动上传两类文件 Version 0.1");

LOG("没有版权(C) Linktone 2005-2006。不保留所有权利。");

LOG("");

LOG("用法: Perl FTPUpload.MultiThread.pl");

LOG("");

##================================================================##

##================================================================##

##=== 程序执行的第一步:尝试登陆ftp站点 ==========================##

## $start_date计算出当前开始的时间

my $start_date = timeString(time);

## $g_nUploadSuccess代表是否已经完全上传,-1为不是,1为是:

my $g_nUploadSuccess = 1;

## $g_strLastError代表上次错误原因:

my $g_strLastError = "";

LOG("正在链接至指定FTP服务器($ftp_server)...");

# 第二个参数用于指定超时时间

my $ftp = Net::FTP->new($ftp_server, Debug=>$ftp_debug, Timeout=>$ftp_timeout);

if($@)

{

$g_strLastError = "不能连接到FTP服务器,错误原因:".$@;

LOG("$g_strLastError@\n");

$g_nUploadSuccess = -1;

}

else

{

$ftp->login($ftp_uid, $ftp_pw);

if($@)

{

$g_strLastError = "不能登陆FTP服务器,错误原因:".$@;

LOG("$g_strLastError\n");

$g_nUploadSuccess = -1;

}

else

{

LOG("链接FTP服务器成功!");

## 关闭连接

$ftp->quit() or warn "咦,为什么不能够和FTP服务器断开连接呢?: $@\n";

##================================================================##

##================================================================##

##=== 程序执行的第二步,将指定文件夹“语音文件”下所有语音文件上传到FTP站点指定目录下 ===##

#my %lookup;

LOG("准备上传“语音文件”目录(@src_dir_WAVFiles)下的所有文件!");

find(\&processFiles, @src_dir_WAVFiles);

LOG("目录(@src_dir_WAVFiles)已经处理完毕,结果是:");

foreach my $thread (@$self)

{

print("Joining thread\n");

# join() does three things: it waits for a thread to exit,

# cleans up after it, and returns any data the thread may

# have produced.

$$thread->join();

}

##================================================================##

##=== 程序执行的第三步,将指定文件夹“命名对照列表文件TXT”下文件上传到FTP站点指定目录下 ===##

if($g_nIsAllWAVsFile_UploadSuccess > 0)

{

LOG("+===============================+");

LOG("准备上传“语音文件”目录(@src_dir_NamesListFile)下的所有文件!");

@$self = {};

find(\&processFiles, @src_dir_NamesListFile);

foreach my $thread (@$self)

{

print("Joining thread\n");

$$thread->join();

}

LOG("目录(@src_dir_NamesListFile)已经处理完毕,结果是:");

LOG("-===============================-");

}

else

{

LOG("-===============================-");

LOG("由于语音文件目录并没有完全上传,所以本命名对照文件不上传!");

LOG("-===============================-");

}

##================================================================##

styl

分享到:
评论

相关推荐

    一个多线程实现的ftp扫描程序

    总的来说,这个多线程FTP扫描程序是一个实用的工具,它利用多线程技术并优化资源管理,能够快速有效地对FTP服务器进行安全评估。通过分析和使用这个程序,我们可以学习到如何设计和实现网络扫描工具,以及如何利用...

    VB下FTP多线程技术的实现.doc

    "VB下FTP多线程技术的实现" VB 下 FTP 多线程技术的实现是指在 VB 语言中使用多线程机制来实现 FTP 文件传输协议的技术。FTP 是因特网上较早出现的服务功能之一,迄今其仍是 Internet 上最常用也是最重要的服务功能...

    ftp同步文件

    - **性能优化**:针对大量文件的同步场景,考虑使用多线程或多进程技术来提高同步效率。 #### 七、总结 通过使用Perl编写的`ftpsync`脚本,可以有效地自动化FTP文件同步过程,减轻了维护人员的工作负担。该脚本不仅...

    国外一个非常强大的FTP服务器,只有一个文件

    7. **脚本和自动化**:某些FTP服务器支持使用脚本语言(如Perl或Python)编写自定义脚本,实现自动化的任务,如定时备份、文件同步等。 8. **安全强化**:除了基本的用户认证,FTP服务器可能还支持更安全的身份验证...

    如何使用Perl脚本实现下载服务器端数据到客户端

    通过Perl脚本,我们可以自动登录到FTP服务器,执行文件上传或下载操作,并在完成后断开连接,这一过程完全无需人工干预。 #### 2. 参数设置与路径管理 在Perl脚本中,参数设置和路径管理是实现自动化下载的关键...

    ftp_file.rar_32 37

    5. **多线程或异步处理**:如果要同时下载多个文件,可能需要使用Perl的线程模块或者异步I/O来提高效率。 6. **配置文件**:可能有一个配置文件存储FTP服务器的连接信息,以便于更改和重用。 7. **日志记录**:为了...

    Perl语言编程(第三版中文版)

    在并发编程上,Perl支持线程和 fork 进程,可以编写多任务并行的程序。 面向对象编程也是Perl的重要特性,书中会介绍如何定义类、创建对象,以及继承、封装和多态等面向对象的基本概念。Perl5引入了MOP(Meta ...

    Perl 6 技术参考手册.docx

    - **多线程编程**:Perl 提供了多线程支持,使得并发处理更为便捷。 - **跨平台兼容性**:Perl 具有优秀的跨平台兼容性,可以在多种操作系统上运行,包括但不限于 Unix-like 和 Windows。 #### 四、Perl 6 的新特性...

    Perl Material(2)

    在这个压缩包文件中,我们期待找到更多关于Perl编程的知识点,帮助初学者深入理解和实践。 Perl的特性包括但不限于以下几点: 1. **灵活性**:Perl的设计理念是“There's More Than One Way To Do It”(TIMTOWTDI...

    跨平台Perl开发指南(第二版)

    10. **并发和并行处理**:Perl支持线程和并发编程,这对于处理大量数据或执行多任务的应用程序来说至关重要。 11. **perlport和perlsys**:介绍Perl的平台差异和如何编写可移植的代码,确保程序能在多种操作系统上...

    高级perl编程(第二版)pdf

    7. **进程和线程**:Perl支持多线程编程,可以创建子进程,这对于并发处理和并行计算至关重要。书中会讲解如何在Perl中实现并发编程。 8. **网络编程**:Perl可以用于构建网络客户端和服务端应用程序,涉及套接字...

    Network Programming With Perl

    5. **FTP编程**:Perl的`Net::FTP`模块允许开发者实现FTP客户端功能,进行文件上传、下载、列出目录等操作。 6. **SMTP邮件发送**:`Net::SMTP`模块支持发送电子邮件,包括设置发件人、收件人、主题和正文内容。 7...

    网络编程基础篇系列-perl

    5. **FTP编程**:Perl也可以处理FTP(文件传输协议)通信,例如`Net::FTP`模块可以让你的Perl程序连接到FTP服务器,上传或下载文件。 6. **网络错误处理**:在网络编程中,必须考虑到网络异常情况,如连接失败、...

    Advanced Perl Programming, 2nd Edition

    4. **并发与并行编程**:Perl可以处理多线程和异步任务,书中有章节讲解如何在Perl中实现并发和并行程序,以提高程序性能。 5. **文件和系统接口**:Perl可以直接操作文件和系统资源,书中会讲解高级的文件操作技巧...

    perl语言编程

    8. **并发和多线程**:Perl支持线程编程,尽管其线程模型在早期版本中有一定限制,但最新版本已得到改进。这部分可能涉及Perl的线程和进程创建、信号处理等。 9. **脚本和脚本调优**:如何编写高效、可维护的Perl...

    PERL编程24学时教程

    11. **进程和线程**:理解Perl中的进程控制(fork、exec)和线程(threads模块),以及如何在多线程环境中编写安全的代码。 12. **数据库交互**:使用DBI模块连接和操作各种数据库,如MySQL、Oracle等。 13. **...

    ftp工具包下载

    此外,FTP工具包通常会支持一些高级特性,如断点续传、被动模式(解决NAT和防火墙问题)、多线程传输、文件列表的获取等。这些特性使得FTP工具包在处理大量或大文件时更加高效和可靠。 总结起来,"ftp工具包下载...

    perl cookbook

    8. **过程和并发编程**:Perl支持多种并发模型,如线程和 fork。书中展示了如何在Perl中进行并发编程,以实现多任务并行执行。 9. **系统交互**:Perl可以方便地与操作系统进行交互,执行外部命令,管理进程等。书...

    perl IO-SOCKET

    Perl的IO-Socket模块是Perl语言中用于网络通信的核心库,它提供了低级套接字接口,使得开发者能够创建各种网络应用程序,如客户端、服务器、甚至是复杂的多线程网络服务。这个模块支持TCP、UDP以及UNIX域socket等...

    iperl中级资料

    7. **并发编程**:Perl支持线程(threads)和进程(forks)编程,了解它们的工作原理和使用方法,可以帮助你编写高效的多任务程序。 8. **数据库接口**:Perl的DBI(Database Interface)模块提供了与各种数据库...

Global site tag (gtag.js) - Google Analytics