7

想用“张量”替换乘法符号“*”,用“p_tensor”替换幂符号“^”,使用以下替换规则:

    a(k)^n --> p_tensor(n,a(k))
    a(i)*a(j) --> tensor(a(i),a(j)), when i=/=j

但是当符号“*”在数字和a(i)之间时,例如3*a(i),我们应该保持符号“*”的原样。

例如,

    5*a(i)*a(j)*(a(k1)+3*a(k2)) --> 5*tensor(tensor(a(i),a(j)),a(k1)+3*a(k2))
    a(i)^2*a(j)^2  --> tensor(p_tensor(2,a(i)),p_tensor(2,a(j)))
    ...

现在我想使用 AWK 或 sed 或 Perl 重新格式化以下表达式:

    3*a(3)^2+6*a(1)^2*(5*a(2)^2-2*a(4))+6*a(2)*a(4)+6*a(1)*(-4*a(2)*a(3)+a(5))

有什么想法吗?

替换后的预期结果应该是

    3*p_tensor(2,a(3))+6*tensor(p_tensor(2,a(1)),(5*p_tensor(2,a(2))-2*a(4))+6*tensor(a(2),a(4))+6*tensor(a(1),(-4*tensor(a(2),a(3))+a(5))
4

1 回答 1

7

正则表达式不能做任意嵌套,也不能做优先级和关联性。为此需要解析器;但是,您可以从以下开始足够接近:

珀尔:

while(<>) {
   s/(a\(\d+\))\^(\d+)/p_tensor($2,$1)/g;
   s/(a\((\d+)\))\*(a\((\d+)\))/tensor($1, $3)/g if $2 != $4;
   print;
}

这是接近的,并为您提供一个级别。然后可以通过添加额外的递归定义的模式来“伪造”额外的嵌套,这些模式可以达到您需要的任何最大嵌套深度(通常不是很多......实际上表达式很少有 3-4 层,这对您来说可能很好)。

试试看:

echo "3*a(3)^2+6*a(1)^2*(5*a(2)^2-2*a(4))+6*a(2)*a(4)+6*a(1)*(-4*a(2)*a(3)+a(5))" | perl t.pl

或类似的东西。

于 2012-12-13T04:57:46.173 回答