Мінімізація математичних висловлювань


18

Змагання

Ви власник дивовижної послуги під назвою Coyote Beta , яка магічно відповідає на математичні запитання, які її користувачі надсилають через Інтернет.

Але виявляється, пропускна здатність дорога. У вас є два варіанти: або створити " Coyote Beta Pro", або знайти спосіб вирішити це. Зовсім недавно хтось запитав (x + 2). Не вдалося клієнту надіслати x+2, і користувач не побачив би різниці?

Завдання

Ваше завдання - «мінімізувати» математичні вирази. З огляду на вхідний вираз, ви повинні позбутися пробілів і дужок, поки воно не подасть мінімальне представлення того ж вводу. Дужки навколо асоціативних операцій не потрібно зберігати.

Єдині оператори , наведені тут +, -, *, /, і ^(зведення в ступінь), зі стандартним математичним асоціативності і пріоритеті. Єдиним пробілом, вказаним у введенні, будуть фактичні символи пробілу.

Зразок вводу / виводу

Input       | Output
------------|--------------
(2+x) + 3   | 2+x+3
((4+5))*x   | (4+5)*x
z^(x+42)    | z^(x+42)
x - ((y)+2) | x-(y+2)
(z - y) - x | z-y-x
x^(y^2)     | x^y^2
x^2 / z     | x^2/z
- (x + 5)+3 | -(x+5)+3

Оцінка балів

Для введення / виводу можна використовувати будь-який бажаний метод. Виграє найменша програма в байтах.

Точні шматочки

Експоненціація є правильним асоціативним, а також слід за стандартним математичним пріоритетом (найвищим). Допустимим числовим літералом є /[0-9]+/, а допустимим є літеральна змінна /[a-z]+/. Одинарна літеральна змінна являє собою єдине значення, навіть коли її довжина символів перевищує 1.

Мається на увазі те, що «круглі дужки навколо асоціативних операцій не потрібно зберігати» - це те, що висновок повинен складатися з виразу, що призводить до ідентичного дерева розбору, за винятком того, що асоціативні операції можуть бути переставлені.


Ідея полягає у створенні мінімального еквівалентного твердження, що призводить до того ж дерева розбору. Це так, що Coyote Beta може візуально відображати його, коли користувач робить запит.
ТНД

Якщо допустимою є змінна /[a-z]+/, це означає, що множення на поєднання як би abзаборонено?
Джо З.

1
Ви хочете, 2+(3+4)щоб вас змінили 2+3+4, правда? Це змінює дерево розбору.
feersum

2
Я приймаю питання із твердженням, що x^(y/2)=x^y/2; експоненцірованіе має більш високий пріоритет порядку, отже, x^y/2=(x^y)/2.
Conor O'Brien

1
Aww man, я збирався подати Prompt X:expr(X)в TI-BASIC, але ви не можете спростити :(
DankMemes

Відповіді:


1

C #, 523 519 504 байт

Перевірте вбудовані коментарі, щоб побачити, як це працює!


Гольф

using System;using System.Collections.Generic;namespace n{class p{static void Main(string[]a){foreach(String s in a){String r=s.Replace(" ","");List<int>l=new List<int>();for(int i=0;i<r.Length;i++){if(r[i]=='('){l.Add(i);continue;}if(r[i]==')'){switch(r[Math.Max(l[l.Count-1]-1,0)]){case'+':case'(':switch(r[Math.Min(i+1,r.Length-1)]){case'+':case'-':case')':r=r.Remove(Math.Max(l[l.Count-1],0),1);r=r.Remove(Math.Min(i,r.Length)-1,1);i-=2;break;}break;}l.RemoveAt(l.Count-1);}}Console.WriteLine(r);}}}}

Безумовно

using System;
using System.Collections.Generic;

namespace n {
    class p {
        static void Main( string[] a ) {
            // Loop every String given for the program
            foreach (String s in a) {
                // Get rid of the spaces
                String r = s.Replace( " ", "" );

                // A little helper that will have the indexes of the '('
                List<int> l = new List<int>();

                // Begin the optimizatio process
                for (int i = 0; i < r.Length; i++) {
                    // If char is an '(', add the index to the helper list and continue
                    if (r[ i ] == '(') {
                        l.Add( i );
                        continue;
                    }

                    // If the char is an ')', validate the group
                    if (r[ i ] == ')') {
                        // If the char before the last '(' is an '+' or '(' ...
                        switch (r[ Math.Max( l[ l.Count - 1 ] - 1, 0 ) ]) {
                            case '+':
                            case '(':
                                // ... and the char after the ')' we're checking now is an '+', '-' or ')' ...
                                switch (r[ Math.Min( i + 1, r.Length - 1 ) ]) {
                                    case '+':
                                    case '-':
                                    case ')':
                                        // Remove the '()' since they're most likely desnecessary.
                                        r = r.Remove( Math.Max( l[ l.Count - 1 ], 0 ), 1 );
                                        r = r.Remove( Math.Min( i, r.Length ) - 1, 1 );

                                        // Go two steps back in the loop since we removed 2 chars from the String,
                                        //   otherwise we would miss some invalid inputs
                                        i -= 2;
                                        break;
                                }

                                break;
                        }

                        // Remove the last inserted index of '(' from the list,
                        //   since we matched an ')' for it.
                        l.RemoveAt( l.Count - 1 );
                    }
                }

                // Print the result
                Console.WriteLine( r );
            }
        }
    }
}

Бічні нотатки

  1. Виправлено деякі помилки та перейменовано декілька варіантів.
  2. Вкладено комутатор, щоб позбутися зайвої змінної. Крім того, виправлено помилку, яка зробить деякі рішення недійсними, повідомив Андер Касеорг .

PS: Якщо у вас є підказка або ви знайшли помилку, будь ласка, повідомте мене в коментарях, і я спробую виправити це (тоді я додам примітку про виправлення помилки з вашим іменем;))


Гарна відповідь! : D істотних відповідей тут, як правило, краще отримати, якщо включити пояснення: P
кіт

Чи можу я це зробити у вигляді коментарів до коду?
auhmaan

Звичайно, що б не працювало c:
кіт

Тоді я це зроблю! Я також спробую десь додати резюме.
auhmaan

Ласкаво просимо до Головоломки з програмуванням та Code Golf, до речі! (хоча це не ваша перша відповідь)
кіт

0

C ++, 284 байти

Гольф

#include<iostream>
#include<algorithm>
int main(){std::string e;std::getline(std::cin,e);e.erase(std::remove_if(e.begin(),e.end(),isspace),e.end());for(int x=0;x<e.length();x++){if(e[x]=='('&&e[x+1]=='('){e.erase(x,1);}if(e[x]==')'&&e[x+1]==')'){e.erase(x,1);}}std::cout<<e;return 0;}

Безумовно

#include<iostream>
#include<algorithm>

int main()
{
    std::string e;
    std::getline(std::cin, e);
    e.erase(std::remove_if(e.begin(), e.end(), isspace), e.end());
    for(int x = 0; x < e.length(); x++) {
        if (e[x] == '(' && e[x+1] == '('){
            e.erase(x, 1);
        }
        if (e[x] == ')' && e[x+1] == ')'){
            e.erase(x, 1);
        }
    }
    std::cout<<e;
    return 0;
}

Це не має жодної логіки пріоритету і виходить з ладу у багатьох поданих тестових випадках.
Андерс Касеорг
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.