Візуалізатор рівнянь мистецтва Ascii


10

Справа з рівняннями за відсутності хорошого редактора рівнянь є безладним і неприємним. Наприклад, якби я хотів висловити інтеграл та його рішення, це може виглядати приблизно так:

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

На integrals.wolfram.com це називається "форма введення". Ніхто не любить бачити рівняння у "формі введення". Ідеальним способом візуалізації цього рівняння було б:

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

(Вольфрам називає це "традиційною формою")

Для цього кодогольфа напишіть програму, яка візьме деяке рівняння у "формі введення" як вхідне та візуалізує це рівняння в уявленні 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 )

Як правило, у вашому запитанні має бути тег, який виражає, що це за конкурс. Я взяв на себе сміливість додати один, бо ви сказали "кодовий гольф" у тексті.
dmckee --- кошеня колишнього модератора

3
Ця проблема занадто туманна, щоб бути кодовим гольфом. Ви не кажете, які конструкції повинні бути підтримані, або як вони мають виглядати. Буде достатньо підтримувати +, -, * та /? Чи потрібно підтримувати Sigma? Що з грецькими літерами? Можливі варіанти вирішення питання, поставленого вами, можуть бути занадто різними у функціональності, щоб порівнювати їх за довжиною коду.
MtnViewMark

@MtnViewMark, я додав деякі "вимоги" ... дайте мені знати, чи краще зараз гольф.
Амі

@Ami - так, багато.
MtnViewMark

Я погоджуюся з MtnViewMark, це здається дуже відкритим і неясним. Можливо, вам буде краще обмежити введення та вихід чітко визначеним набором тестів для цілей гольфу. Ви зробили еталонну реалізацію?
гніблер

Відповіді:


10

Пітон 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
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.