免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 5687 | 回复: 15
打印 上一主题 下一主题

求助:除去成对的括号 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2011-02-19 22:41 |只看该作者 |倒序浏览
设最外层括号为第 1 层,请问怎么样能够除去 1 对第 2 层的括号,保留其他括号?

谢谢!

例如:
  1. (((1,2),3),4)   =>  ((1,2),3,4)

  2. ((1,2),(3,4))   =>  ((1,2),3,4)
  3.                         or
  4.                     (1,2,(3,4))
  5.                     
  6. (1,(2,(3,4)))   => (1,2,(3,4))
复制代码

论坛徽章:
0
2 [报告]
发表于 2011-02-20 15:17 |只看该作者
基本能够完成任务的代码,请各位指正。

  1. #!/usr/bin/perl

  2. use strict;
  3. use warnings;

  4. while (<DATA>) {
  5.     next if /^#/;
  6.     next if /^\s*$/;
  7.     chomp;
  8.    
  9.     print $_, "==>\n";
  10.    
  11.     my ($lnode, $rnode) = getNodes($_);
  12.    
  13.     if ($lnode =~ /^\(/) {
  14.         my $str = $lnode;
  15.         
  16.         my ($lnode, $mnode) = getNodes($str);
  17.         
  18.         print join(',', ($lnode, $mnode, $rnode)), "\n";
  19.     }
  20.     else {
  21.         my $str = $rnode;
  22.         
  23.         my ($mnode, $rnode) = getNodes($str);
  24.         
  25.         print join(',', ($lnode, $mnode, $rnode)), "\n";
  26.     }
  27. }

  28. =begin
  29.   Name:     getNodes
  30.   Desc:     Get 2 nodes of a binary tree
  31.   Usage:    getNodes($str)
  32.   Args:     A string
  33.   Return:   2 strings
  34. =cut

  35. sub getNodes {
  36.     my ($str) = @_;
  37.    
  38.     # Remove leading and tailing parentheses
  39.     # $str =~ s/^\(|\)$//g;
  40.    
  41.     # '((1,2),3)'
  42.     if ($str =~ /^\((\(.+\)),(\w+)\)$/) {
  43.     #    print "L\n";
  44.         return ($1, $2);
  45.     }
  46.     # '((1,2),(3,4))'
  47.     elsif ($str =~ /^\((\(.+\)),(\(.+\))\)$/) {
  48.     #    print "M\n";
  49.         return ($1, $2);
  50.     }
  51.     # '(1,(2,3))'
  52.     elsif ($str =~ /^\((\w+),(\(.+\))\)$/) {
  53.     #    print "R\n";
  54.         return ($1, $2);
  55.     }
  56.     # '(1,2)'
  57.     elsif ($str =~ /^\((\w+),(\w+)\)$/) {
  58.         return ($1, $2);
  59.     }
  60.     else {
  61.         print "Error!\n";
  62.         return (0, 0);
  63.     }
  64. }

  65. __DATA__
  66. (((1,2),3),4)
  67. ((1,2),(3,4))
  68. (1,(2,(3,4)))
  69. ((1,2),(3,(4,5)))
  70. (((((1,2),3),4),(5,6))
复制代码
输出

  1. (((1,2),3),4)==>
  2. (1,2),3,4
  3. ((1,2),(3,4))==>
  4. 1,2,(3,4)
  5. (1,(2,(3,4)))==>
  6. 1,2,(3,4)
  7. ((1,2),(3,(4,5)))==>
  8. 1,2,(3,(4,5))
  9. (((((1,2),3),4),(5,6))==>
  10. (((1,2),3),4,(5,6)

复制代码

论坛徽章:
0
3 [报告]
发表于 2011-02-20 20:38 |只看该作者
回复 1# longbow0
  1. SubstrIndex( $_ ) for "(((1,2),3),4)", "((1,2),(3,4))", "(1,(2,(3,4)))", "(((((1,2),3),4),(5,6))";

  2. sub SubstrIndex {
  3.     my $string = shift;
  4.     print "$string\t==>\t";
  5.     my $posi = Index( $string );
  6.     substr( $string, $posi->[0], 1 ) = "";
  7.     substr( $string, $posi->[1], 1 ) = "";
  8.     print "$string\n";
  9. }
  10. sub Index {
  11.     my $string = shift;
  12.     my $n = 2;
  13.     my $where1 = -1;
  14.     $where1 = index $string, "(", $where1 + 1 for 1 .. $n;
  15.     my $where2 = index $string, ")", $where1 + 1;
  16.     my $where3 = index $string, "(", $where1 + 1;
  17.     while( $where3 < $where2 and $where3 != -1 ){
  18.         $where2 = index $string, ")", $where2 + 1;
  19.         $where3 = index $string, "(", $where3 + 1 ;
  20.     }
  21.     return [ $where1, $where2 - 1 ];
  22. }
复制代码

论坛徽章:
0
4 [报告]
发表于 2011-02-21 00:39 |只看该作者
本帖最后由 黑色阳光_cu 于 2011-02-21 03:28 编辑

正则版本
  1. #!/bin/env perl

  2. use strict;
  3. use warnings;
  4. use 5.010;

  5. while (my $str = <DATA>)
  6. {
  7.         chomp $str;

  8.         print "$str => ";
  9.         my @stack;
  10.         foreach (0 .. 1)
  11.         {
  12.                 $str =~ /(\()(?=((?:[^()]|(?1)(?2))+(\))))/g;
  13.                 push(@stack, [$-[1], $-[3]]);
  14.         }

  15.         substr($str, $stack[1][1], 1) = "";
  16.         substr($str, $stack[1][0], 1) = "";
  17.         print "$str\n";
  18. }

  19. __DATA__
  20. (((1,2),3),4)
  21. ((1,2),(3,4))
  22. (1,(2,(3,4)))
复制代码

论坛徽章:
0
5 [报告]
发表于 2011-02-21 00:55 |只看该作者
  1. $str =~ /
  2. (\()        # $1: 匹配左括号
  3. (?=        # 整体是1个环视,这样,第1次匹配成功会从第1个左括号开始,第2个次匹配成功会从第2个左括号开始,以此类推
  4.         (        # $2: 匹配括号里的内容加上$3
  5.                 (?:        # 分组不捕获
  6.                         [^()] # 要么不包括括号
  7.                         |
  8.                         (?1)(?2)        # 要么是分组1加上分组2的递归
  9.                 )+
  10.                 (\)) # $3: 匹配右括号
  11.         )
  12. )
  13. /xg;
复制代码

论坛徽章:
0
6 [报告]
发表于 2011-02-21 03:09 |只看该作者
二叉树版本
  1. #!/bin/env perl

  2. use strict;
  3. use warnings;
  4. use Data::Dumper;

  5. while (my $str = <DATA>)
  6. {
  7.     chomp $str;

  8.     print "$str => ";
  9.     my $binary_tree = [undef, undef, undef, -1];    # [左孩子,右孩子,父节点,数据]
  10.     my $cnt = $binary_tree;
  11.     my $pos = -1;
  12.     do
  13.     {
  14.         (my $c, $pos) = find_parenthese($str, $pos);
  15.         if ($pos != -1)
  16.         {
  17.             if ($c eq "(")
  18.             {
  19.                 while (defined $cnt->[0])
  20.                 {
  21.                     $cnt = $cnt->[0];
  22.                 }

  23.                 $cnt->[0] = [undef, undef, $cnt, $pos];
  24.             }
  25.             else
  26.             {
  27.                 $cnt->[1] = [undef, undef, $cnt, $pos];
  28.                 while (defined $cnt->[1])
  29.                 {
  30.                     $cnt = $cnt->[2];
  31.                 }
  32.             }

  33.             $pos = -1 if (defined $binary_tree->[0][0] and defined $binary_tree->[0][1]);
  34.         }
  35.     } while ($pos != -1);

  36.     substr($str, $binary_tree->[0][1][3], 1) = "";
  37.     substr($str, $binary_tree->[0][0][3], 1) = "";
  38.     print "$str\n";
  39. }


  40. sub find_parenthese
  41. {
  42.     my ($str, $base) = @_;
  43.     $base = -1 if (not defined $base);
  44.     my ($c, $pos) = ("", -1);

  45.     my $pos1 = index($str, "(", $base + 1);
  46.     my $pos2 = index($str, ")", $base + 1);

  47.     if ($pos1 != -1 and ($pos2 == -1 or $pos1 < $pos2))
  48.     {
  49.         $c = "(";
  50.         $pos = $pos1;
  51.     }
  52.     elsif ($pos2 != -1 and ($pos1 == -1 or $pos2 < $pos1))
  53.     {
  54.         $c = ")";
  55.         $pos = $pos2;
  56.     }

  57.     return ($c, $pos);
  58. }

  59. __DATA__
  60. (((1,2),3),4)
  61. ((1,2),(3,4))
  62. (1,(2,(3,4)))
复制代码

论坛徽章:
0
7 [报告]
发表于 2011-02-21 03:26 |只看该作者
堆栈
  1. #!/bin/env perl

  2. use strict;
  3. use warnings;

  4. while (my $str = <DATA>)
  5. {
  6.     chomp $str;

  7.     print "$str => ";
  8.     my @stack;
  9.     my ($l_pos, $r_pos);
  10.     my $pos = -1;
  11.     do
  12.     {
  13.         (my $c, $pos) = find_parenthese($str, $pos);
  14.         if ($pos != -1)
  15.         {
  16.             if ($c eq "(")
  17.             {
  18.                 push(@stack, $pos);
  19.             }
  20.             else
  21.             {
  22.                 $l_pos = pop(@stack);
  23.                 if ($#stack == 0)
  24.                 {
  25.                     $r_pos = $pos;
  26.                     $pos = -1;
  27.                 }
  28.             }
  29.         }
  30.     } while ($pos != -1);

  31.     substr($str, $r_pos, 1) = "";
  32.     substr($str, $l_pos, 1) = "";
  33.     print "$str\n";
  34. }

  35. sub find_parenthese
  36. {
  37.     my ($str, $base) = @_;
  38.     $base = -1 if (not defined $base);
  39.     my ($c, $pos) = ("", -1);

  40.     my $pos1 = index($str, "(", $base + 1);
  41.     my $pos2 = index($str, ")", $base + 1);

  42.     if ($pos1 != -1 and ($pos2 == -1 or $pos1 < $pos2))
  43.     {
  44.         $c = "(";
  45.         $pos = $pos1;
  46.     }
  47.     elsif ($pos2 != -1 and ($pos1 == -1 or $pos2 < $pos1))
  48.     {
  49.         $c = ")";
  50.         $pos = $pos2;
  51.     }

  52.     return ($c, $pos);
  53. }

  54. __DATA__
  55. (((1,2),3),4)
  56. ((1,2),(3,4))
  57. (1,(2,(3,4)))
复制代码

论坛徽章:
0
8 [报告]
发表于 2011-02-21 08:14 |只看该作者
越短越好呢。

论坛徽章:
1
丑牛
日期:2014-02-14 17:07:04
9 [报告]
发表于 2011-02-21 15:52 |只看该作者
黑色阳光_cu 发表于 2011-02-21 00:55


我拷过去执行出现错误
Sequence (?1...) not recognized in regex; marked by <-- HERE in m/(\()(?=((?:[^()]|(?1 <-- HERE )(?2))+(\))))/ at 2.pl line 11.

论坛徽章:
0
10 [报告]
发表于 2011-02-21 16:09 |只看该作者
回复 9# ace_fei


    检查下perl的版本,这个是5.10的新特性
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP