ExcelHome技术论坛

 找回密码
 免费注册

QQ登录

只需一步,快速开始

快捷登录

搜索
EH技术汇-专业的职场技能充电站 妙哉!函数段子手趣味讲函数 Excel服务器-会Excel,做管理系统 效率神器,一键搞定繁琐工作
HR薪酬管理数字化实战 Excel 2021函数公式学习大典 Excel数据透视表实战秘技 打造核心竞争力的职场宝典
让更多数据处理,一键完成 数据工作者的案头书 免费直播课集锦 ExcelHome出品 - VBA代码宝免费下载
用ChatGPT与VBA一键搞定Excel WPS表格从入门到精通 Excel VBA经典代码实践指南
楼主: figfig

[原创] 在vba里使用PERL

[复制链接]

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 15:13 | 显示全部楼层
本帖已被收录到知识树中,索引项:脚本语言应用
七、元素的增删
    增加元素已经讲过,可以通过给一个未出现过的元素赋值来向关联数组中增加新元素,如$fruit{"lime"} = 1;创建下标为lime、值为1的新元素。
    删除元素的方法是用内嵌函数delete,如欲删除上述元素,则:
    delete ($fruit{"lime"});
注意:

1、一定要使用delete函数来删除关联数组的元素,这是唯一的方法。
2、一定不要对关联数组使用内嵌函数push、pop、shift及splice,因为其元素位置是随机的。

 

Sub perl()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
x.eval "%fruits = (apples, 9,bananas, 23,cherries, 11); @fruits=%fruits;"


Debug.Print x.eval("@fruits")
x.eval "delete @fruits{apples};@fruits=%fruits;  "
Debug.Print x.eval("@fruits")


End Sub

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 15:37 | 显示全部楼层
7.5.2 在数组中寻找惟一的元素
上面这个代码中展示的方法也可以用来寻找数组中只出现一次的元素。假设已经将输入
的全部单词放入一个数组而不是哈希结构,同时没有专门采取措施来保证在将一个单词放入
列表之前,该列表中还没有这个单词。在这种情况下,列表中可能存在许多重复的单词。
如果输入的文本的开始行是One Fish,Two Fish,那么该列表看上去如下所示:
如果你被赋予这个单词列表(在@ f i s h w o r d s中),同时你只需要该列表的独一无二的元素,
那么使用哈希结构就非常适合你的需要,如程序清单7 - 1所示:
程序清单7-1 寻找数组中的惟一的元素
第1行:用于对临时哈希结构% s e e n进行初始化,该哈希结构用于存放你的所有单词。
第2行:对单词列表进行迭代操作,依次将$ 设置为每个单词。
第3行:用于创建哈希结构% s e e n中的关键字,以$ 中的该单词作为关键字,并为该数据
创建一个名义上的值。
82使用第一部分Perl 基础
下载
第5行:只是从哈希结构中取出所有关键字,并将它们存放在@ u n i q u e w o r d s中。哈希结
构中的任何重复单词(例如f i s h)将互相改写,然后只以一个关键字出现。
Sub perl()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
x.eval "@fish=qw(one fish two fish red fish blue fish);%seen=(); foreach(@fish){$seen{$_}=1;};$aa=join(',',keys(%seen));"


Debug.Print x.eval("$aa")


End Sub

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 15:53 | 显示全部楼层
7.5.3 寻找两个数组之间的交汇部分和不同部分
对数组经常要进行的一项操作是寻找两个数组之间的交汇部分(即它们的重叠部分)和
两个数组之间的不同部分(它们不重叠的部分)。在这个例子中,你有两个列表,一个是包含
电影明星的列表,另一个是包含政治家的列表。你的任务是找出是电影明星的所有政治家。
下面是你的两个(非常不完整的)数组:
程序清单7 - 2显示了寻找交汇部分的代码。
程序清单7-2 寻找两个数组的交汇部分
第1行:用于对哈希结构% s e e n进行初始化。这个临时哈希结构用于存放所有电影明星的
名字。
第2行:对电影明星的列表进行迭代操作,依次将$ 设置为每个名字。
第3行:用电影明星的名字填入哈希结构% s e e n的各个关键字,并将值设置为1,这个值可
以是你想要的任何真值。
第5行:这一行看起来比它实际上更复杂一些。@ p o l s中的G r e p函数对政治家的列表进行
迭代操作,依次将$ 设置给每个政治家。然后,在哈希结构% s e e n中寻找该名字。如果该名字
返回真,那么它就位于哈希结构中,表达式$ s e e n { $ }计算的结果为真。如果该表达式计算的
结果是真,那么g r e p返回$ 的值,然后该值被放入@ i n t e r s e c t i o n中。这个过程将重复进行,直
到@ p o l s被g r e p全部查看完毕。当该代码段运行结束时, @ i n t e r s e c t i o n便包含既是@ s t a r s又是
@ p o l s的所有成员的名字。
用于寻找两个数组之间的不同部分(即在一个数组中存在,而在另一个数组中不存在的
那些元素)的代码与上面这个代码几乎是相同的,可以使用程序清单7 - 3来查找不是电影明星
的所有政治家。
程序清单7-3 寻找两个数组之间的不同部分
惟一有变化的一行是第5行。它仍然用于查找哈希结构% s e e n中的每个政治家的名字,但
是,现在如果它找到了政治家的名字,则返回假。相反,如果没有找到政治家的名字,则返
回真。出现在哈希结构% s e e n中的所有政治家的名字并不返回给@ d i ff e r e n c e。如果你想要找
到不是政治家的所有电影明星,使用的代码几乎完全一样,但是必须将@ s t a r s切换成@ p o l s。
Sub perl()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
x.eval "@stars=qw(aa bb cc);@pols=qw(cc dd ee) ;%seen=();foreach(@stars){$seen{$_}=1;};@aa=grep($seen{$_},@pols);$bb=join(',',@aa);"


Debug.Print x.eval("$bb")


End Sub
Sub perl2()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
x.eval "@stars=qw(aa bb cc);@pols=qw(cc dd ee) ;%seen=();foreach(@stars){$seen{$_}=1;};@aa=grep(! $seen{$_},@pols);$bb=join(',',@aa);"


Debug.Print x.eval("$bb")


End Sub

[ 本帖最后由 figfig 于 2010-5-6 15:55 编辑 ]

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 16:06 | 显示全部楼层
P e r l有一个非常有用的能够对列表进行操作的运算符,称为范围运算符。范围运算符由一
对圆点(. .)来表示。下面是该运算符的用法的例子:
( 1 . . 1 0 )
范围运算符用一个左边的操作数( 1)和右边的操作数( 1 0)构成了一个包含1到1 0(含1
与1 0)之间的所有数的列表。如果需要在列表中使用若干个范围,那么只要使用多个范围运
算符即可:
(1..10, 20..30);
上面这个例子创建了一个包含2 1个元素的列表,即包含1到1 0和2 0到3 0(含1、1 0、2 0和
3 0)之间的数。如果范围运算符的右边的操作数小于左边的操作数,比如( 1 0 . . 1),那么将产
生一个空列表。范围运算符既可以用于字符串,也可以用于数字。范围( a . . z )可以产生一个包
含所有2 6个小写字母的列表。范围( a a . . z z )可以生成一个大得多的列表,它由6 7 5个字母对组成,
从a a、a b、a c、a d开始,到z x、z y、z z结束。

Sub perl2()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
x.eval "@aa=(1..100);$bb=join(',',@aa);"


Debug.Print x.eval("$bb")


End Sub

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 16:26 | 显示全部楼层
[



9    Perl 中的正则表达式
正则表达式的三种形式

正则表达式中的常用模式

正则表达式的 8 大原则

  

      正则表达式是 Perl 语言的一大特色,也是 Perl 程序中的一点难点,不过如果大家能够很好的掌握他,就可以轻易地用正则表达式来完成字符串处理的任务,当然在 CGI 程序设计中就更能得心应手了。下面我们列出一些正则表达式书写时的一些基本语法规则。


--------------------------------------------------------------------------------

9.1 正则表达式的三种形式
首先我们应该知道 Perl 程序中,正则表达式有三种存在形式,他们分别是:

匹配:m/<regexp>;/ (还可以简写为 /<regexp>;/ ,略去 m)

替换:s/<pattern>;/<replacement>;/

转化:tr/<pattern>;/<replacemnt>;/

这三种形式一般都和 =~ 或 !~ 搭配使用(其中 "=~" 表示相匹配,在整条语句中读作 does,"!~" 表示不匹配,在整条语句中读作 doesn't),并在左侧有待处理的标量变量。如果没有该变量和 =~ !~ 操作符,则默认为处理 $_ 变量中的内容。举例如下:

$str = "I love Perl";
$str =~ m/Perl/; # 表示如果在 $str 中发现 "Perl" 字符串,则返回 "1" 否则返回 "0"。
$str =~ s/Perl/BASH/; # 表示将变量 $str 中的 "Perl" 字符串替换为 "BASH",如果发生此替换则返回 "1",否则返回 "0"。
$str !~ tr/A-Z/a-z/; # 表示将变量 $str 中的所有大写字母转化为小写字母,如果转化发生了则返回 "0",否则返回 "1"。

另外还有:

foreach (@array) { s/a/b/; } # 此处每次循环将从 @array 数组中取出一个元素存放在 $_ 变量中,并对 $_ 进行替换处理。
while (<FILE>;) { print if (m/error/); } # 这一句稍微复杂一些,他将打印 FILE 文件中所有包含 error 字符串的行。

Perl 的正则表达式中如果出现 () ,则发生匹配或替换后 () 内的模式被 Perl 解释器自动依次赋给系统 $1, $2 ...... 请看下面的例子:

$string = "I love perl";
$string =~ s/(love)/<$1>;/; # 此时 $1 = "love",并且该替换的结果是将 $string 变为 "I <love>; perl"
$string = "i love perl";
$string =~ s/(i)(.*)(perl)/<$3>;$2<$1>;/; # 这里 $1 = "i",$2 = " love ",$3 = "perl",并且替换后 $string 变为 "<perl>; love <i>;"

替换操作 s/<pattern>;/<replacement>;/ 还可以在末尾加上 e 或 g 参数,他们的含义分别为:

s/<pattern>;/<replacement>;/g 表示把待处理字符串中所有符合 <pattern>; 的模式全部替换为 <replacement>; 字符串,而不是只替换第一个出现的模式。
s/<pattern>;/<replacement>;/e 表示将把 <replacemnet>; 部分当作一个运算符,这个参数用的不多。

比如下面的例子:

$string = "i:love:perl";
$string =~ s/:/*/; #此时 $string="i*love:perl";
$string = "i:love:perl";
$string =~ s/:/*/g; #此时 $string="i*love*perl";
$string =~ tr/*/ /; #此时 $string="i love perl";
$string = "www22cgi44";
$string =~ s/(\d+)/$1*2/e; # (/d+)代表 $string 中的一个或多个数字字符,将这些数字字符执行 *2 的操作,因此最后 $string 变成了 "www44cgi88"。

下面给出一个完整的例子:

#!/usr/bin/perl

print"请输入一个字符串!\n";
$string = <STDIN>;; # <STIDN>;代表标准输入,会让使用者输入一字符串
chop($string); # 将$string最后一个换行的字符\n删除掉
if($string =~ /perl/){
  print("输入的字符串中有 perl 这个字符串!\n";
}

如果输入的字符串含有 perl 这个字符串的话,就会显示后面的提示信息。



9.2 正则表达式中的常用模式
下面是正则表达式中的一些常用模式。

/pattern/  结果  
. 匹配除换行符以外的所有字符
x? 匹配 0 次或一次 x 字符串
x* 匹配 0 次或多次 x 字符串,但匹配可能的最少次数
x+ 匹配 1 次或多次 x 字符串,但匹配可能的最少次数
.* 匹配 0 次或一次的任何字符
.+ 匹配 1 次或多次的任何字符
{m} 匹配刚好是 m 个 的指定字符串
{m,n} 匹配在 m个 以上 n个 以下 的指定字符串
{m,} 匹配 m个 以上 的指定字符串
[] 匹配符合 [] 内的字符
[^] 匹配不符合 [] 内的字符
[0-9] 匹配所有数字字符
[a-z] 匹配所有小写字母字符
[^0-9] 匹配所有非数字字符
[^a-z] 匹配所有非小写字母字符
^ 匹配字符开头的字符
$ 匹配字符结尾的字符
\d 匹配一个数字的字符,和 [0-9] 语法一样
\d+ 匹配多个数字字符串,和 [0-9]+ 语法一样
\D 非数字,其他同 \d
\D+ 非数字,其他同 \d+
\w 英文字母或数字的字符串,和 [a-zA-Z0-9] 语法一样
\w+ 和 [a-zA-Z0-9]+ 语法一样
\W 非英文字母或数字的字符串,和 [^a-zA-Z0-9] 语法一样
\W+ 和 [^a-zA-Z0-9]+ 语法一样
\s 空格,和 [\n\t\r\f] 语法一样
\s+ 和 [\n\t\r\f]+ 一样
\S 非空格,和 [^\n\t\r\f] 语法一样
\S+ 和 [^\n\t\r\f]+ 语法一样
\b 匹配以英文字母,数字为边界的字符串
\B 匹配不以英文字母,数值为边界的字符串
a|b|c 匹配符合a字符 或是b字符 或是c字符 的字符串
abc 匹配含有 abc 的字符串
(pattern) () 这个符号会记住所找寻到的字符串,是一个很实用的语法。第一个 () 内所找到的字符串变成 $1 这个变量或是 \1 变量,第二个 () 内所找到的字符串变成 $2 这个变量或是 \2 变量,以此类推下去。  
/pattern/i i 这个参数表示忽略英文大小写,也就是在匹配字符串的时候,不考虑英文的大小写问题。
\ 如果要在 pattern 模式中找寻一个特殊字符,如 "*",则要在这个字符前加上 \ 符号,这样才会让特殊字符失效


下面给出一些例子:   

范例 说明
/perl/ 找到含有 perl 的字符串
/^perl/ 找到开头是 perl 的字符串
/perl$/ 找到结尾是 perl 的字符串
/c|g|i/ 找到含有 c 或 g 或 i 的字符串
/cg{2,4}i/ 找到 c 后面跟着 2个到 4个 g ,再跟着 i 的字符串
/cg{2,}i/ 找到 c 后面跟着 2个以上 g ,再跟着 i 的字符串
/cg{2}i/ 找到 c 后面跟着 2个 g,再跟着 i 的字符串
/cg*i/ 找到 c 后面跟着 0个或多个 g ,再跟着 i 的字符串,如同/cg{0,1}i/
/cg+i/ 找到 c 后面跟着一个以上 g,再跟着 i 的字符串,如同/cg{1,}i/
/cg?i/ 找到 c 后面跟着 0个或是 1个 g ,再跟着 i 的字符串,如同/cg{0,1}i/
/c.i/ 找到 c 后面跟着一个任意字符,再跟着 i 的字符串
/c..i/ 找到 c 后面跟着二个任意字符,再跟着 i 的字符串
/[cgi]/ 找到符合有这三个字符任意一个的字符串
/[^cgi]/ 找到没有这三个字符中任意一个的字符串
/\d/ 找寻符合数字的字符,可以使用/\d+/来表示一个或是多个数字组成的字符串
/\D/ 找寻符合不是数字的字符,可以使用/\D+/来表示一个或是更多个非数字组成的字符串
/\*/ 找寻符合 * 这个字符,因为 * 在常规表达式中有它的特殊意思,所以要在这个特殊符号前加上 \ 符号,这样才会让这个特殊字符失效
/abc/i 找寻符合 abc 的字符串而且不考虑这些字符串的大小写



9.3 正则表达式的八大原则
  如果在 Unix 中曾经使用过 sed、awk、grep 这些命令的话,相信对于 Perl 语言中的正则表达式(Regular Expression)不会感到陌生。Perl 语言由于有这个功能,所以对字符串的处理能力非常强。在Perl语言的程序中,经常可以看到正则表达式的运用,在 CGI 程序设计中也不例外。

  正则表达式是初学 Perl 的难点所在,不过只要一旦掌握其语法,你就可以拥有几乎无限的模式匹配能力,而且 Perl 编程的大部分工作都是掌握常规表达式。下面给大家介绍几条正则表达式使用过程中的 8 大原则。

  正则表达式在对付数据的战斗中可形成庞大的联盟——这常常是一场战争。我们要记住下面八条原则:

· 原则1:正则表达式有三种不同形式(匹配(m/ /),替换(s/ / /eg)和转换(tr/ / /))。

· 原则2:正则表达式仅对标量进行匹配( $scalar =~ m/a/; 可以工作; @array =~ m/a/ 将把@array作为标量对待,因此可能不会成功)。

· 原则3:正则表达式匹配一个给定模式的最早的可能匹配。缺省时,仅匹配或替换正则表达式一次( $a = 'string string2'; $a =~ s/string/ /; 导致 $a = 'string 2')。

· 原则4:正则表达式能够处理双引号所能处理的任意和全部字符( $a =~ m/$varb/ 在匹配前把varb扩展为变量;如果 $varb = 'a' $a = 'as',$a =~ s/$varb/ /; 等价于 $a =~ s/a/ /; ,执行结果使 $a = " s" )。

· 原则5:正则表达式在求值过程中产生两种情况:结果状态和反向引用: $a=~ m/pattern/ 表示 $a 中是否有子串 pattern 出现,$a =~ s/(word1)(word2)/$2$1/ 则“调换”这两个单词。

· 原则6:正则表达式的核心能力在于通配符和多重匹配运算符以及它们如何操作。$a =~ m/\w+/ 匹配一个或多个单词字符;$a =~ m/\d/" 匹配零个或多个数字。

· 原则7:如果欲匹配不止一个字符集合,Perl使用 "|" 来增加灵活性。如果输入 m/(cat|dog)/ 则相当于“匹配字符串 cat 或者 dog。

· 原则8:Perl用 (?..) 语法给正则表达式提供扩展功能。(这一点请同学们课后看相关资料)

想要学习所有这些原则?我建议大家先从简单的开始,并且不断的尝试和实验。实际上如果学会了 $a =~ m/ERROR/ 是在 $a 中查找子串ERROR,那么你就已经比在 C 这样的低层语言中得到了更大的处理能力。
Sub perl2()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
x.eval "$str = 'I love Perl';$str =~s/Perl/BASH/; "


Debug.Print x.eval("$str")


End Sub

[ 本帖最后由 figfig 于 2010-5-6 16:33 编辑 ]

TA的精华主题

TA的得分主题

发表于 2010-5-6 16:32 | 显示全部楼层
[广告] VBA代码宝 - VBA编程加强工具 · VBA代码随查随用  · 内置多项VBA编程加强工具       ★ 免费下载 ★      ★使用手册
很久不来,又见精彩!

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 18:49 | 显示全部楼层

文字操作更加方便

Sub perl2()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
   x.eval " $hour = 5;$min = 12;$sec = 6;$time = sprintf(qq(%02d:%02d:%02d), $hour,$min,$sec);"
         

Debug.Print x.eval("$time")

End Sub

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 18:57 | 显示全部楼层

读取文件目录

Sub perl2()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
   x.eval " opendir(DIR, 'c:\\');@file = readdir(DIR);closedir(DIR);$bb=join(',',@file); "
         

Debug.Print x.eval("$bb")

End Sub

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 19:00 | 显示全部楼层

从末尾开始删除3个字符

Sub perl2()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
   x.eval " $word = milktea;substr($word, -3, 3) = '';"


         

Debug.Print x.eval("$word")

End Sub

TA的精华主题

TA的得分主题

 楼主| 发表于 2010-5-6 19:02 | 显示全部楼层

文字替换

Sub perl2()

Set x = CreateObject("scriptcontrol")
x.Language = "perlscript"
   x.eval " $word = milktea;substr($word, 0,4) = lemon;"


         

Debug.Print x.eval("$word")

End Sub
您需要登录后才可以回帖 登录 | 免费注册

本版积分规则

手机版|关于我们|联系我们|ExcelHome

GMT+8, 2024-12-22 18:15 , Processed in 0.042465 second(s), 6 queries , Gzip On, MemCache On.

Powered by Discuz! X3.4

© 1999-2023 Wooffice Inc.

沪公网安备 31011702000001号 沪ICP备11019229号-2

本论坛言论纯属发表者个人意见,任何违反国家相关法律的言论,本站将协助国家相关部门追究发言者责任!     本站特聘法律顾问:李志群律师

快速回复 返回顶部 返回列表