Visualizador de equações de arte ASCII

10

Lidar com equações na ausência de um bom editor de equações é confuso e desagradável. Por exemplo, se eu quisesse expressar uma integral e sua solução, ela seria algo como isto:

Integral [x ^ 3 e ^ (- mx ^ 2 b / 2), dx] = - ((2 + b m x ^ 2) / (b ^ 2 * e ^ ((b m x ^ 2) / 2) * m ^ 2))

Em integrals.wolfram.com , isso é chamado de "formulário de entrada". Ninguém gosta de ver uma equação em "formulário de entrada". A maneira ideal de visualizar esta equação seria:

insira a descrição da imagem aqui

(Wolfram chama isso de "forma tradicional")

Para esse codegolf, escreva um programa que terá alguma equação na "forma de entrada" como entrada e visualize essa equação em uma representação ascii da "forma tradicional". Portanto, neste exemplo, podemos obter algo parecido com isto:

       /\      3
       |      x
       | ------------  dx = 
       |       2
      \/   (m x  b)/2
          e

              2
     2 + b m x
-(-----------------)
            2
   2  (b m x )/2  2
  b  e           m

Requisitos:

  1. Não embaralhe, simplifique ou reorganize a entrada de forma alguma. Renderize-o exatamente da mesma forma que foi descrita pela entrada.
  2. Suporte as quatro operações matemáticas básicas (+, -, *, /). Quando não multiplicar dois números adjacentes, o símbolo * está implícito e deve ser omitido.
  3. O suporte à integração (como mostrado no exemplo acima) não é necessário. Ser capaz de dar suporte à entrada com funções como [...] Integrate ou Sqrt [...] é um bônus.
  4. Poderes de suporte, como mostrado no exemplo acima (a enésima raiz pode ser modelada aumentando-a para a enésima potência).
  5. Parênteses redundantes (como aqueles ao redor do denominador e numerador da fração grande no exemplo acima) devem ser omitidos.
  6. A expressão no denominador e numerador de uma fração deve estar centralizada acima e abaixo da linha de divisão horizontal.
  7. Você pode escolher se deseja ou não iniciar uma nova linha após um sinal de igual. No exemplo acima, uma nova linha é iniciada.
  8. A ordem das operações deve ser exatamente a mesma na saída e na entrada.

Alguns exemplos de entrada e saída associada para testar sua solução:

Entrada:

1/2 + 1/3 + 1/4

Resultado:

1   1   1
- + - + -
2   3   4

Entrada:

3x^2 / 2 + x^3^3

Resultado:

   2     3
3 x     3
---- + x   
 2

Entrada:

(2 / x) / (5 / 4^2)

Resultado:

2
-
x
--
5
--
 2
4

Entrada:

(3x^2)^(1/2)

Resultado:

    2 1/2
(3 x )
Ami
fonte
Sua pergunta geralmente deve ter uma tag que expresse que tipo de competição é essa. Tomei a liberdade de adicionar um porque você disse "codegolf" no texto.
dmckee --- ex-moderador gatinho
3
Esse problema é nebuloso demais para ser um código de golfe. Você não diz que construções devem ser suportadas ou como devem ser. Apenas apoiar +, -, * e / seria suficiente? O Sigma precisa ser suportado? E as letras gregas? As possíveis soluções para a pergunta, como você a colocou, podem ter funcionalidades muito variadas para serem comparadas no tamanho do código.
precisa
@MtnViewMark, adicionei alguns "Requisitos" ... deixe-me saber se o golfe está melhor agora.
Ami
@ Ami - sim, muito.
precisa saber é o seguinte
Concordo com o MtnViewMark, isso parece muito aberto e vago. Talvez seja melhor limitar a entrada e a saída a um conjunto bem definido de caixas de teste para fins de golfe. Você fez uma implementação de referência?
Gnibbler #

Respostas:

10

Caracteres Python 2, 1666

O layout é realmente muito fácil - é a análise da entrada que é uma dor real. Ainda não tenho certeza se está completamente correto.

import re,shlex
s=' '
R=range

# Tokenize.  The regex is because shlex doesn't treat 3x and x3 as two separate tokens.  The regex jams a space in between.                                                 
r=r'\1 \2'
f=re.sub
x=shlex.shlex(f('([^\d])(\d)',r,f('(\d)([^\d])',r,raw_input())))
T=[s]
while T[-1]:T+=[x.get_token()]
T[-1]=s

# convert implicit * to explicit *                                                                                                                                          
i=1
while T[i:]:
 if(T[i-1].isalnum()or T[i-1]in')]')and(T[i].isalnum()or T[i]in'('):T=T[:i]+['*']+T[i:]
 i+=1

# detect unary -, replace with !                                                                                                                                            
for i in R(len(T)):
 if T[i]=='-'and T[i-1]in'=+-*/^![( ':T[i]='!'
print T

# parse expression: returns tuple of op and args (if any)                                                                                                                   
B={'=':1,',':2,'+':3,'-':3,'*':4,'/':4,'^':5}
def P(t):
 d,m=0,9
 for i in R(len(t)):
  c=t[i];d+=c in'([';d-=c in')]'
  if d==0and c in B and(B[c]<m or m==B[c]and'^'!=c):m=B[c];r=(c,P(t[:i]),P(t[i+1:]))
 if m<9:return r
 if'!'==t[0]:return('!',P(t[1:]))
 if'('==t[0]:return P(t[1:-1])
 if'I'==t[0][0]:return('I',P(t[2:-1]))
 return(t[0],)

# parenthesize a layout                                                                                                                                                     
def S(x):
 A,a,b,c=x
 if b>1:A=['/'+A[0]+'\\']+['|'+A[i]+'|'for i in R(1,b-1)]+['\\'+A[-1]+'/']
 else:A=['('+A[0]+')']
 return[A,a+2,b,c]

# layout a parsed expression.  Returns array of strings (one for each line), width, height, centerline                                                                      
def L(z):
 p,g=z[0],map(L,z[1:])
 if p=='*':
  if z[1][0]in'+-':g[0]=S(g[0])
  if z[2][0]in'+-':g[1]=S(g[1])
 if p=='^'and z[1][0]in'+-*/^!':g[0]=S(g[0])
 if g:(A,a,b,c)=g[0]
 if g[1:]:(D,d,e,f)=g[1]
 if p in'-+*=,':
  C=max(c,f);E=max(b-c,e-f);F=C+E;U=[s+s+s]*F;U[C]=s+p+s;V=3
  if p in'*,':U=[s]*F;V=1
  return([x+u+y for x,u,y in zip((C-c)*[s*a]+A+(E-b+c)*[s*a],U,(C-f)*[s*d]+D+(E-e+f)*[s*d])],a+d+V,F,C)
 if'^'==p:return([s*a+x for x in D]+[x+s*d for x in A],a+d,b+e,c+e)
 if'/'==p:w=max(a,d);return([(w-a+1)/2*s+x+(w-a)/2*s for x in A]+['-'*w]+[(w-d+1)/2*s+x+(w-d)/2*s for x in D],w,b+e+1,b)
 if'!'==p:return([' -  '[i==c::2]+A[i]for i in R(b)],a+2,b,c)
 if'I'==p:h=max(3,b);A=(h-b)/2*[s*a]+A+(h-b+1)/2*[s*a];return(['  \\/|/\\  '[(i>0)+(i==h-1)::3]+A[i]for i in R(h)],a+3,h,h/2)
 return([p],len(p),1,0)

print'\n'.join(L(P(T[1:-1]))[0])

Para a grande entrada na pergunta, recebo:

 /\         2                     2 
 |     - m x  b          2 + b m x  
 |     --------    = - -------------
 |  3      2                    2   
\/ x  e         dx         b m x    
                           ------   
                        2     2    2
                       b  e       m 

Aqui estão alguns casos de teste mais complicados:

I:(2^3)^4
O:    4
  / 3\ 
  \2 / 

I:(2(3+4)5)^6
O:             6
  (2 (3 + 4) 5) 

I:x Integral[x^2,dx] y
O:   /\ 2     
  x  | x  dx y
    \/        

I:(-x)^y
O:     y
  (- x) 

I:-x^y
O:     y
  (- x)

Esse último está errado, algum erro de precedência no analisador.

Keith Randall
fonte
a linha de base do argumento da integral não deve ser centralizada verticalmente após a integral? Atualmente, parece mais um subscrito à integral.
JoeyMay
Não é difícil mudar, mas desperdiçaria algum espaço. Atualmente, faço o sinal integral apenas o suficiente para abranger seu argumento (com um mínimo de 3).
perfil completo de Keith Randall
Golfe leve: use guias em vez de espaços duplos.
CalculatorFeline