Ascii art Уравнение визуализации

10

Работа с уравнениями в отсутствие хорошего редактора уравнений - это грязно и неприятно. Например, если бы я хотел выразить интеграл и его решение, он мог бы выглядеть примерно так:

Интеграл [x ^ 3 e ^ (- mx ^ 2 b / 2), dx] = - ((2 + b m x ^ 2) / (b ^ 2 * e ^ ((b m x ^ 2) / 2) * м ^ 2))

На сайте integras.wolfram.com это называется «формой ввода». Никто не любит видеть уравнение в «форме ввода». Идеальный способ визуализации этого уравнения:

введите описание изображения здесь

(Вольфрам называет это «традиционной формой»)

Для этого Codegolf напишите программу, которая будет принимать какое-то уравнение в «форме ввода» в качестве входных данных и визуализировать это уравнение в представлении ascii «традиционной формы». Итак, для этого примера мы можем получить что-то вроде этого:

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

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

Требования:

  1. Не перемешивайте, не упрощайте и не переставляйте входные данные любым способом. Визуализируйте его в той же форме, в которой оно было описано входными данными.
  2. Поддержка четырех основных математических операций (+, -, *, /). Если не умножить два соседних числа, подразумевается символ * и его следует опускать.
  3. Поддержка интеграции (как показано в примере выше) не требуется. Возможность поддержки ввода с помощью таких функций, как Integrate [...] или Sqrt [...], является бонусом.
  4. Поддерживайте силы, как показано в примере выше (n-й корень может быть смоделирован путем повышения до 1 / n-й степени).
  5. Избыточные круглые скобки (например, те, которые находятся вокруг деноменатора и числителя большой дроби в приведенном выше примере) должны быть опущены.
  6. Выражение в знаменателе и числителе дроби должно быть в центре выше и ниже горизонтальной линии деления.
  7. Вы можете выбрать, стоит ли начинать новую строку после знака равенства. В приведенном выше примере новая строка запускается.
  8. Порядок операций должен быть точно таким же на выходе, как и на входе.

Некоторые примеры ввода и связанного вывода для тестирования вашего решения:

Входные данные:

1/2 + 1/3 + 1/4

Вывод:

1   1   1
- + - + -
2   3   4

Входные данные:

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

Вывод:

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

Входные данные:

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

Вывод:

2
-
x
--
5
--
 2
4

Входные данные:

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

Вывод:

    2 1/2
(3 x )
Ami
источник
Ваш вопрос, как правило, должен иметь тег, указывающий, что это за конкурс. Я позволил себе добавить один, потому что вы сказали «Codegolf» в тексте.
dmckee --- котенок экс-модератора
3
Эта проблема слишком туманна, чтобы быть код-гольфом. Вы не говорите, какие конструкции должны поддерживаться или как они должны выглядеть. Будет ли достаточно поддержки +, -, * и /? Нужно ли поддерживать Sigma? Как насчет греческих букв? Возможные решения вопроса, как вы его задали, могут быть слишком разными по функциональности, чтобы сравнивать их по длине кода.
MtnViewMark
@MtnViewMark, я добавил некоторые «требования» ... дайте мне знать, если гольф сейчас лучше.
Ами
@ Ами - да, очень.
MtnViewMark
Я согласен с MtnViewMark, это кажется очень открытым и расплывчатым. Возможно, вам было бы лучше ограничить ввод и вывод четко определенным набором тестовых случаев для целей игры в гольф. Вы сделали эталонную реализацию?
Гнибблер

Ответы:

10

Python 2, 1666 символов

Макет на самом деле довольно прост - это разбор входных данных - это королевская боль. Я все еще не уверен, что это полностью правильно.

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])

За большой вклад в вопрос я получаю:

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

Вот еще несколько сложных тестовых случаев:

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)

Последнее неверно, какая-то ошибка приоритета в парсере.

Кит Рэндалл
источник
не должна ли базовая линия интегрального аргумента быть вертикально центрированной после интеграла? В настоящее время это больше похоже на индекс к интегралу.
Джои
Не трудно изменить, но это потратило бы немного места. В настоящее время я делаю знак интеграла достаточно большим, чтобы охватить его аргумент (минимум 3).
Кит Рэндалл
Небольшой гольф: используйте вкладки вместо двойных пробелов.
CalculatorFeline