Szalone równania chemiczne

10

Powinieneś otrzymać ciąg równania chemicznego (bez spacji, tylko litery (wielkie i małe), cyfry, nawiasy i znaki matematyczne) od użytkownika i wydrukować odpowiedź, jeśli równanie jest zrównoważone, czy nie (dowolna para odpowiedzi pozytywnych / negatywnych : Tak / Nie, prawda / fałsz, 1/0). Aby skrócić kod, możesz założyć, że ciągi wejściowe mogą zawierać tylko te elementy: Al, Ar, B, Be, C, Cl, Cr, Cu, Fe, H, He, K, N, O, S. I jeszcze jedno : mogą być -znaki. Chodzi o matematykę: +oznacza dodawanie, -oznacza odejmowanie.

Przykłady:

Wejście:

C6H5COOH-O2=7CO2+3H2O

Wynik:

No

Wejście:

2Fe(CN)6+2SO2+202=Fe2(SO4)2+6C2N2

Wynik:

Yes

Wejście:

2SO2=2SO4-2O2

Wynik:

Yes

Najkrótszy kod wygrywa.

gthacoder
źródło
Szukasz funkcji? A może program, który pobiera tekst i podaje tekst? Jeśli później, powinno to zająć i przetworzyć wiele linii? Czy tylko jedno równanie na przebieg?
MtnViewMark
Powiązane: Równania chemiczne równania
Peter Taylor
@MtnViewMark To ma być program. Powinno to zająć jedno równanie na przebieg.
gthacoder
Drugi i trzeci przykład są błędne. Myślę, że miałeś zamiar pisać 2O2zamiast 202(dwieście dwa).
r3mainer
@ squeamishossifrage Oh, yeah. Pewnie. Dziękuję Ci. Pytanie zostało zaktualizowane.
gthacoder

Odpowiedzi:

2

Mathematica 152

f=Times@@@Tr@CoefficientRules@ToExpression@(r=StringReplace)[r[
#<>")",{"="->"-(",x:_?LetterQ|")"~~y_?DigitQ:>x<>"^"<>y}],x_?UpperCaseQ:>" "<>x]⋃{}=={0}&

Wynik:

f@"C6H5COOH-O2=7CO2+3H2O"
f@"2Fe(CN)6+2SO2+2O2=Fe2(SO4)2+6C2N2"
f@"2SO2=2SO4-2O2"

Fałszywe

Prawdziwe

Prawdziwe

Traktuję wzór chemiczny jako wielomian, np

wprowadź opis zdjęcia tutaj

Następnie liczę tylko współczynniki.

Ybeltukov
źródło
Jak to działa? W twoim przykładzie współczynniki każdej „zmiennej” nie upraszczają się do zera.
MtnViewMark
@MtnViewMark Dokładniej liczę moce, Tr@CoefficientRulesa następnie mnożę je przez współczynniki z Times@@@. Za O:, 2*2+2*2=4*2za C:2*6 = 6*2 itp
ybeltukov
2

Python 2.7, 316 276 znaków

import re
r=re.sub
x='(^|[=+-])'
y=r'\1+\2'
z='(\w)([A-Z(])'
t=r('=','==',r(z,y,r(z,y,r('([A-Za-z)])(\d)',r'\1*\2',r('([=+-]|$)',r')\1',r(x+'(\d+)',r'\1\2*(',r(x+'([A-Z])',r'\1(\2',raw_input())))))))
E=re.findall('[A-Za-z]+',t)
print all(eval(t,{f:f==e for f in E})for e in E)

Wiele przekształca wyrażenia regularne, aby przekształcić równanie wejściowe w coś evalzdolnego. Następnie sprawdza równanie dla każdego elementu osobno.

Na przykład przykładowe równania przepisują na ( tzmienną):

(C*6+H*5+C+O+O+H)-(O*2)==7*(C+O*2)+3*(H*2+O)
2*(Fe+(C+N)*6)+2*(S+O*2)+2*(O*2)==(Fe*2+(S+O*4)*2)+6*(C*2+N*2)
2*(S+O*2)==2*(S+O*4)-2*(O*2)

Jestem pewien, że w regexie będzie więcej golfa.

Keith Randall
źródło
2

Haskell, 400 351 308 znaków

import Data.List
import Text.Parsec
(&)=fmap
r=return
s=string
l f b=(>>=(&b).f)
x=(=<<).replicate
m=sort&chainl1(l x(concat&many1(l(flip x)n i))n)((s"+">>r(++))<|>(s"-">>r(\\)))
i=between(s"(")(s")")m<|>(:[])&(l(:)(many lower)upper)
n=option 1$read&many1 digit
main=getContents>>=parseTest(l(==)(s"=">>m)m)

To mogło po prostu wycisnąć z niego wszystkie golfa. Nie wiem, czy jest jeszcze 100 51 trzeba zapisać 8 znaków!

& echo 'C6H5COOH-O2=7CO2+3H2O' | runhaskell Chemical.hs
False

& echo '2Fe(CN)6+2SO2+2O2=Fe2(SO4)2+6C2N2' | runhaskell Chemical.hs
True

& echo '2SO2=2SO4-2O2' | runhaskell Chemical.hs
True

Oto wersja bez golfa, na wypadek gdyby ktoś chciał pójść za nią. Jest to prosty Parsecparser oparty:

import Control.Applicative ((<$>), (<*>))
import Data.List
import Text.Parsec
import Text.Parsec.String (Parser)

type Atom = String

{- golf'd as x -}
multiple :: Int -> [Atom] -> [Atom]
multiple n = concat . replicate n

{- golf'd as m -}
chemicals :: Parser [Atom]
chemicals = sort <$> chainl1 molecules op
  where
    op :: Eq a => Parser ([a] -> [a] -> [a])
    op = (char '+' >> return (++))
     <|> (char '-' >> return (\\))

    molecules :: Parser [Atom]
    molecules = multiple <$> number <*> terms

    terms :: Parser [Atom]
    terms = concat <$> many1 term

    term :: Parser [Atom]
    term = flip multiple <$> item <*> number

{- gofl'd as i -}
item :: Parser [Atom]
item = between (char '(') (char ')') chemicals
   <|> (:[]) <$> atom
  where
    atom :: Parser Atom
    atom = (:) <$> upper <*> many lower

{- gofl'd as n -}
number :: Parser Int
number = option 1 $ read <$> many1 digit

{- gofl'd as main -}
main :: IO ()
main = getContents >>= parseTest chemEquality
  where
    chemEquality :: Parser Bool
    chemEquality = (==) <$> chemicals <*> (char '=' >> chemicals)
MtnViewMark
źródło