Wiąz - Szybki przewodnik

Elm to funkcjonalny język programowania. Zaprojektował go Evan Czaplicki w 2012 roku.

Elm jest specjalnie używany do projektowania front-endu aplikacji internetowych.

Elm kompiluje się do JavaScript i działa w przeglądarce. Jest szybki, testowalny, łatwy w utrzymaniu i nie zawiera wyjątków w środowisku wykonawczym.

Niektóre praktyczne zastosowania platformy programistycznej Elm obejmują -

  • Games
  • Graphics
  • Aplikacje jednostronicowe

Dlaczego Elm

Elm eliminuje większość typowych problemów napotykanych przez programistów frontendowych. Obejmuje to -

Brak wyjątków czasu wykonywania

Elm to język z typowaniem statycznym. Wszystkie możliwe błędy są sprawdzane i korygowane w czasie kompilacji. Dzięki temu można nie mieć wyjątków w czasie wykonywania.

Przyjazne dla programistów komunikaty o błędach

W przeciwieństwie do innych języków programowania, kompilator Elm został zaprojektowany tak, aby dostarczać bardzo specyficzne i przyjazne dla programistów komunikaty o błędach w czasie kompilacji. Komunikaty o błędach zawierają również wskazówki, takie jak łącza do zalecanych dokumentacji projektowych.

Łatwe do przetestowania

Każdą funkcję Elm można przetestować w izolacji od wszystkich innych. Dzięki temu programy napisane w Elm są łatwo testowalne.

Automatyczne wersjonowanie semantyczne

Elm wymusza automatyczne semantyczne wersjonowanie pakietów. Gwarantuje to, że zmiana poprawki nie spowoduje awarii już uruchomionej aplikacji.

Kod wielokrotnego użytku

Funkcje Elm są z natury łatwe do ponownego wykorzystania w porównaniu z funkcjami w JavaScript, Pythonie lub TypeScript.

W tym rozdziale omówiono kroki instalacji Elm na platformach Windows, Mac i Linux.

Konfiguracja środowiska lokalnego

Rozważ poniższe kroki, aby zainstalować Elm w swoim środowisku lokalnym.

Step 1 − Install node

Ponieważ elm jest kompilowany do JavaScript, maszyna docelowa powinna mieć nodezainstalowany. Instrukcje konfiguracji można znaleźć w kursie TutorialsPoint NodeJSnode i npm

Konfiguracja węzła.

Step 2 − Install elm

Wykonaj następujące polecenie na terminalu, aby zainstalować elm. Zauważ, że stabilna wersja elm to 0.18 w momencie pisania tego kursu.

npm install -g [email protected]

Po instalacji wykonaj następujące polecenie, aby zweryfikować wersję Elm.

C:\Users\dell>elm --version
0.18.0

Step 2 − Install the Editor

Środowisko programistyczne używane tutaj to Visual Studio Code (platforma Windows).

Visual Studio Code to IDE typu open source z programu Visual Studio. Jest dostępny na platformy Mac OS X, Linux i Windows. VSCode jest dostępny pod adresem

Instalacja w systemie Windows

W tej sekcji omówimy kroki instalacji Elm w systemie Windows.

Pobieranie https://code.visualstudio.com/. dla Windowsa.

Kliknij dwukrotnie plik VSCodeSetup.exe, aby rozpocząć proces instalacji. To zajmie tylko minutę.

Możesz bezpośrednio przejść do ścieżki pliku, klikając prawym przyciskiem myszy Plik → Otwórz w wierszu poleceń. Podobnie opcja Pokaż w Eksploratorze pokazuje plik w Eksploratorze plików.

Instalacja w systemie Mac OS X

Specyficzny przewodnik instalacji programu Visual Studio Code dla systemu Mac OS X można znaleźć pod adresem Instalacja VSCode - MAC.

Instalacja w systemie Linux

Podręcznik instalacji programu Visual Studio Code dla systemu Linux można znaleźć pod adresem Instalacja VSCode - Linux.

Step 4 − Install the elm Extension

Zainstaluj rozszerzenie wiązu w VSCode, jak pokazano poniżej.

Wiąz REPL

REPL oznacza Read Eval Print Loop. Reprezentuje środowisko komputera, takie jak konsola Windows lub powłoka Unix / Linux, w którym wprowadza się polecenie, a system odpowiada, wysyłając dane wyjściowe w trybie interaktywnym.

Elm jest dostarczany w pakiecie ze środowiskiem REPL. Wykonuje następujące zadania -

  • Odczyt - czyta dane wejściowe użytkownika, analizuje dane wejściowe do struktury danych wiązu i przechowuje w pamięci.

  • Eval - pobiera i ocenia strukturę danych.

  • Drukuj - drukuje wynik.

  • Pętla - zapętla powyższe polecenie, dopóki użytkownik nie zakończy pracy. Użyj polecenia: exit, aby wyjść z REPL i powrócić do terminala.

Prosty przykład dodawania dwóch liczb w REPL pokazano poniżej -

Otwórz terminal VSCode i wpisz polecenie elm REPL.

Terminal REPL czeka na wprowadzenie danych wejściowych przez użytkownika. Wprowadź następujące wyrażenie 10 + 20. Środowisko REPL przetwarza dane wejściowe w sposób przedstawiony poniżej -

  • Odczytuje numery 10 i 20 od użytkownika.

  • Oblicza przy użyciu operatora +.

  • Wyświetla wynik jako 30.

  • Pętle do wprowadzenia następnego użytkownika. Tutaj wychodzimy z pętli.

Ten rozdział omawia, jak napisać prosty program w elm.

Step 1 − Create a directory HelloApp in VSCode

Teraz utwórz plik - Hello.elm w tym katalogu.

Powyższy diagram przedstawia folder projektu HelloApp i terminal otwarty w VSCode.

Step 2 − Install the necessary elm packages

Menedżer pakietów w elm to elm-package . Zainstaluj pakiet elm-lang / html . Ten pakiet pomoże nam wyświetlić wyjście kodu elm w przeglądarce.

Przejdź do HelloApp folder projektu, klikając prawym przyciskiem myszy Plik → Otwórz w wierszu polecenia w VSCode.

Wykonaj następujące polecenie w oknie terminala -

C:\Users\dell\Elm\HelloApp> elm-package install elm-lang/html

Następujące pliki / foldery są dodawane do katalogu projektu podczas instalacji pakietu.

  • elm-package.json (plik), przechowuje metadane projektu
  • elm-stuff (folder), przechowuje pakiety zewnętrzne

Po pomyślnym zainstalowaniu pakietu pojawi się następujący komunikat.

Step 3 − Add the following code to the Hello.elm file

-- importing Html module and the function text
import Html exposing (text)

-- create main method
main =
-- invoke text function
text "Hello Elm from TutorialsPoint"

Powyższy program wyświetli komunikat tekstowy Hello Elm from TutorialsPoint w przeglądarce.

W tym celu musimy zaimportować funkcję text w ramach Htmlmoduł. Funkcja tekstowa służy do drukowania dowolnej wartości ciągu w przeglądarce. Główną metodą jest punkt wejścia do programu. Metoda main wywołuje funkcję tekstową i przekazuje do niej wartość ciągu.

Step 4 − Compile the project

Wykonaj następujące polecenie w oknie terminala VSCode.

elm make Hello.elm

Dane wyjściowe powyższego polecenia są pokazane poniżej -

//update path to the proj folder in the command elm make
C:\Users\dell\elm\HelloApp>elm make Hello.elm
Success! Compiled 38 modules.
Successfully generated index.html

Powyższe polecenie wygeneruje plik index.htmlplik. Kompilator elm konwertuje plik .elm do JavaScript i osadza go windex.html plik.

Step 5 − Open the index.html in the browser

Otwórz plik index.html w dowolnej przeglądarce. Wynik będzie taki, jak pokazano poniżej -

Komentarze w Elm

Komentarze są sposobem na poprawę czytelności programu. Komentarze mogą służyć do dołączania dodatkowych informacji o programie, takich jak autor kodu, wskazówki dotyczące konstrukcji funkcji itp. Komentarze są ignorowane przez kompilator.

Elm obsługuje następujące typy komentarzy -

  • Komentarze jednowierszowe (-) - dowolny tekst między znakiem - a końcem wiersza jest traktowany jako komentarz.

  • Komentarze wieloliniowe ({- -}) - te komentarze mogą obejmować wiele wierszy.

Ilustracja

-- this is single line comment

{- This is a
   Multi-line comment
-}

Linie i wcięcia

Elm nie zawiera nawiasów klamrowych wskazujących bloki kodu dla definicji funkcji lub kontroli przepływu. Bloki kodu są oznaczone wcięciami linii, które są sztywno wymuszane. Wszystkie instrukcje w bloku muszą mieć wcięcie tej samej wielkości. Na przykład -

module ModuleIf exposing (..)
x = 0

function1 =
   if x > 5 then
      "x is greater"
   else
      "x is small"

Jednak następujący blok generuje błąd -

-- Create file ModuleIf.elm
module ModuleIf exposing (..)
x = 0

function1 =
   if x > 5 then
      "x is greater"
         else --Error:else indentation not at same level of if statement
      "x is small"

Zatem w Elm wszystkie ciągłe linie wcięte z taką samą liczbą spacji utworzyłyby blok.

C:\Users\admin>elm repl
---- elm-repl 0.18.0 -----------------------------------------------------------
   :help for help, :exit to exit, more at 
   <https://github.com/elm-lang/elm-repl>
   ---------------------------------------
   -----------------------------------------

> import ModuleIf exposing(..) -- importing module from ModuleIf.elm file
>function1 -- executing function from module
-- SYNTAX PROBLEM ---------------------------------------------------

I need whitespace, but got stuck on what looks like a new declaration. 
You are either missing some stuff in the declaration above or just need to add some spaces here:
7| else
   ^
I am looking for one of the following things:

   whitespace

System typów reprezentuje różne typy wartości obsługiwane przez język. System typów sprawdza poprawność podanych wartości, zanim zostaną one zapisane lub zmienione przez program. Gwarantuje to, że kod zachowuje się zgodnie z oczekiwaniami. System typów umożliwia ponadto bogatsze podpowiedzi do kodu i zautomatyzowaną dokumentację.

Elm to język z typowaniem statycznym. Wiąz ma typy podobne do tych z innych języków.

Numer

Numer Typ danych reprezentuje wartości liczbowe. System typu Elm obsługuje następujące typy liczbowe -

Sr. No. Rodzaj Przykład
1 liczba - przechowuje dowolną liczbę 7 to typ liczbowy
2 Float - przechowuje wartości ułamkowe 7/2 daje wynik 3.5 jako Float
3 Int - przechowuje wartości nieułamkowe 7 // 2 daje wynik 3 jako Int

Numer typu obejmuje zarówno wartości ułamkowe, jak i nieułamkowe. Otwórz elm REPL i wypróbuj przykłady podane poniżej -

C:\Users\admin>elm repl
---- elm-repl 0.18.0 
---------------------------------------------
--------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
------------------------------------------
--------------------------------------
> 7
7 : number
> 7/2
3.5 : Float
> 7//2
3 : Int
>

String i Char

String typ danych jest używany do reprezentowania sekwencję znaków. Char typ danych jest używany do reprezentowania pojedynczy znak. Wartości ciągów są definiowane w podwójnym cudzysłowie ”, a wartości Char są zawarte w pojedynczym cudzysłowie”.

Sr. No. Rodzaj Przykład
1 Ciąg - przechowuje sekwencję znaków „TutorialsPoint”
2 Char - przechowuje wartości ułamkowe „T”

Otwórz elm REPL i wypróbuj przykłady podane poniżej -

C:\Users\admin>elm repl
---- elm-repl 0.18.0 ---------------------------------------
--------------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
--------------------------------------
------------------------------------------
> "TutorialsPoint"
"TutorialsPoint" : String
> 'T'
'T' : Char

Bool

Typ danych Bool w Elm obsługuje tylko dwie wartości - True i False. Słowo kluczowe Bool jest używane do reprezentowania wartości logicznej.

Sr. No. Rodzaj Przykład
1 Bool - przechowuje wartości True lub False 1 == 1 zwraca True

Otwórz elm REPL i wypróbuj przykłady podane poniżej -

C:\Users\dell\elm>elm repl
---- elm-repl 0.18.0 -----------------------------------
------------------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
----------------------------------------
----------------------------------------
> True
True : Bool
> False
False : Bool
> 1==1
True : Bool
> 1==2
False : Bool
> 1 /= 2 -- not equal
True : Bool
> not True
False : Bool
> not False
True : Bool

Typy niestandardowe

Elm obsługuje tworzenie typów zdefiniowanych przez użytkownika. Na przykład rozważ wniosek o płatność. Aplikacja musi przechowywać różne sposoby płatności - kartę kredytową, kartę debetową i bankowość internetową. Można to osiągnąć poprzez zdefiniowanie niestandardowego rodzaju i ograniczenie jego wartości do trzech akceptowalnych sposobów płatności.

Poniższy przykład pokazuje, jak utworzyć typ niestandardowy.

> type PaymentMode = CreditCard|NetBanking|DebitCard
> payment1 = CreditCard
CreditCard : Repl.PaymentMode
> payment2 = DebitCard
DebitCard : Repl.PaymentMode
> payment3 = UPI
-- NAMING ERROR ---------------------------------------------- repl-temp-000.elm

Cannot find variable `UPI`

7| payment3 = UPI

W powyższym przykładzie utworzyliśmy niestandardowy typ PaymentMode. Zmienne payment1 i payment2 są przypisane do wartości PaymentMode. Jeśli wartość przypisana do zmiennej nie pasuje do żadnej z wartości zdefiniowanych przez typ PaymentMode, aplikacja zgłosi błąd składniowy.

Strukturalne typy danych

Typów danych strukturalnych można używać do przechowywania wielu wartości w formacie strukturalnym. Elm obsługuje następujące typy danych strukturalnych -

  • Tuple
  • List
  • Record
  • Record

Zostaną one szczegółowo omówione w kolejnych rozdziałach.

Zmienna z definicji to „nazwana przestrzeń w pamięci”, w której przechowywane są wartości. Innymi słowy, działa jako kontener dla wartości w programie. Zmienna pomaga programom przechowywać wartości i manipulować nimi.

Zmienne w Elm są powiązane z określonym typem danych. Typ danych określa rozmiar i układ pamięci zmiennej, zakres wartości, które mogą być przechowywane w tej pamięci oraz zestaw operacji, które można wykonać na zmiennej.

Reguły nazewnictwa zmiennych

W tej sekcji dowiemy się o regułach nazewnictwa zmiennych.

  • Nazwy zmiennych mogą składać się z liter, cyfr i znaku podkreślenia.
  • Nazwy zmiennych nie mogą zaczynać się od cyfry. Musi zaczynać się od litery lub podkreślenia.
  • Wielkie i małe litery są różne, ponieważ Elm rozróżnia wielkość liter.

Deklaracja zmiennej w Elm

Składnia typu do deklarowania zmiennej w Elm jest podana poniżej -

Składnia 1

variable_name:data_type = value

Składnia „:” (znana jako adnotacja typu) służy do powiązania zmiennej z typem danych.

Składnia 2

variable_name = value-- no type specified

Typ danych jest opcjonalny podczas deklarowania zmiennej w Elm. W tym przypadku typ danych zmiennej jest wywnioskowany na podstawie przypisanej do niej wartości.

Ilustracja

W tym przykładzie użyto edytora VSCode do napisania programu elm i wykonania go przy użyciu polecenia elm repl.

Step 1 − Create a project folder - VariablesApp. Create a Variables.elm file in the project folder.

Dodaj następującą zawartość do pliku.

module Variables exposing (..) //Define a module and expose all contents in the module
message:String -- type annotation
message = "Variables can have types in Elm"

Program definiuje moduł Zmienne. Nazwa modułu musi być taka sama, jak nazwa pliku programu elm. Składnia (..) służy do uwidocznienia wszystkich komponentów w module.

Program deklaruje zmienną wiadomość typu String .

Step 2 − Execute the program.

  • Wpisz następujące polecenie w terminalu VSCode, aby otworzyć wiązanie REPL.
elm repl
  • Wykonaj następującą instrukcję elm w terminalu REPL.
> import Variables exposing (..) --imports all components from the Variables module
> message --Reads value in the message varaible and prints it to the REPL 
"Variables can have types in Elm":String
>

Ilustracja

Użyj Elm REPL, aby wypróbować następujący przykład.

C:\Users\dell\elm>elm repl
---- elm-repl 0.18.0 ---------------------------------------
--------------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
-------------------------------------
------------------------------------------
> company = "TutorialsPoint"
"TutorialsPoint" : String
> location = "Hyderabad"
"Hyderabad" : String
> rating = 4.5
4.5 : Float

Tutaj zmienne firma i lokalizacja to zmienne typu String, a ocena to zmienna typu Float.

Wiąz REPL nie obsługuje adnotacji typu dla zmiennych. Poniższy przykład zgłasza błąd, jeśli typ danych jest uwzględniony podczas deklarowania zmiennej.

C:\Users\dell\elm>elm repl
---- elm-repl 0.18.0 -----------------------------------------
------------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
----------------------------------------
----------------------------------------
> message:String
-- SYNTAX PROBLEM -------------------------------------------- repl-temp-000.elm

A single colon is for type annotations. Maybe you want :: instead? Or maybe you
are defining a type annotation, but there is whitespace before it?

3| message:String
^

Maybe <http://elm-lang.org/docs/syntax> can help you figure it out.

Aby wstawić podział wiersza podczas korzystania z funkcji REPL elm, użyj składni \, jak pokazano poniżej -

C:\Users\dell\elm>elm repl
---- elm-repl 0.18.0 --------------------------------------
---------------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
------------------------------------------
--------------------------------------
> company \ -- firstLine
| = "TutorialsPoint" -- secondLine
"TutorialsPoint" : String

Operator definiuje jakąś funkcję, która zostanie wykonana na danych. Wartości, na których pracują operatory, nazywane są operandami. Rozważmy następujące wyrażenie

7 + 5 = 12

Tutaj wartości 7, 5 i 12 są operandami, podczas gdy + i = są operatorami.

Główni operatorzy w Elm można sklasyfikować jako -

  • Arithmetic
  • Relational
  • Logical

Operatory arytmetyczne

Załóżmy, że wartości zmiennych a i b wynoszą odpowiednio 7 i 2.

Pokaż przykłady

Sr. No. Operator Opis Przykład
1 + (Dodawanie) zwraca sumę operandów a + b wynosi 9
2 -(Odejmowanie) zwraca różnicę wartości ab wynosi 5
3 * (Mnożenie) zwraca iloczyn wartości a * b wynosi 14
4 / (Podział pływający) wykonuje operację dzielenia i zwraca iloraz zmiennoprzecinkowy a / b wynosi 3,5
5 // (Dzielenie całkowite) wykonuje operację dzielenia i zwraca iloraz liczby całkowitej a // b wynosi 3
6 % (Moduł) wykonuje operację dzielenia i zwraca resztę a% b wynosi 1

Operatorzy relacyjni

Operatory relacyjne testują lub definiują rodzaj relacji między dwiema jednostkami. Te operatory służą do porównywania dwóch lub więcej wartości. Operatory relacyjne zwracają wartość logiczną, tj. Prawda lub fałsz.

Załóżmy, że wartość a to 10, a b to 20.

Pokaż przykłady

Sr. No. Operator Opis Przykład
1 > Lepszy niż (a> b) jest fałszywe
2 < Mniej niż (a <b) jest Prawdą
3 > = Większe bądź równe (a> = b) jest fałszywe
4 <= Mniejsze lub równe (a <= b) jest Prawdą
5 == Równość (a == b) jest fałszem
6 ! = Nie równe (a! = b) jest Prawdą

Porównywalne typy

Operatory porównania, takie jak> = lub <, działają z porównywalnymi typami. Są one definiowane jako liczby, znaki, łańcuchy i listy, krotki. Porównywalne typy po obu stronach operatora muszą być takie same.

Sr. No. Porównywalny typ Przykład
1 numer 7> 2 daje Prawda
2 postać „a” == „b” daje Fałsz
3 strunowy "hello" == "hello" daje True
4 krotka (1, „One”) == (1, „One”) daje True
5 lista [1,2] == [1,2] daje True

Otwórz elm REPL i wypróbuj przykłady pokazane poniżej -

C:\Users\admin>elm repl
---- elm-repl 0.18.0 -----------------------------------------------------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
--------------------------------------------------------------------------------
> 7>2
True : Bool
> 7.0>2
True : Bool
> 7.0<2.0
False : Bool
> 'a' > 'b'
False : Bool
> 'a' < 'b'
True : Bool
> "a" < "b"
True : Bool
> (1,2) > (2,3)
False : Bool
> ['1','3'] < ['2','1']
True : Bool
>

Operatory logiczne

Operatory logiczne służą do łączenia dwóch lub więcej warunków. Operatory logiczne również zwracają wartość logiczną.

Pokaż przykłady

Sr. No. Operator Opis Przykład
1 && Operator zwraca prawdę tylko wtedy, gdy wszystkie określone wyrażenia zwracają wartość true (10> 5) && (20> 5) zwraca True
2 || Operator zwraca prawdę, jeśli co najmniej jedno z podanych wyrażeń zwróci prawdę (10 <5) || (20> 5) zwraca True
3 nie Operator zwraca odwrotność wyniku wyrażenia. Na przykład:! (> 5) zwraca fałsz. not (10 <5) zwraca True
4 xor Operator zwraca prawdę tylko wtedy, gdy dokładnie jedno wejście zwraca prawdę. Operator zwraca fałsz, jeśli oba wyrażenia zwracają prawdę. xor (10> 5) (20> 5) zwraca fałsz

Struktury decyzyjne wymagają, aby programista określił jeden lub więcej warunków, które mają być ocenione lub przetestowane przez program, wraz z instrukcją lub instrukcjami, które mają być wykonane, jeśli warunek zostanie określony jako prawdziwy, i opcjonalnie inne instrukcje do wykonania, jeśli warunek jest określany jako fałszywy.

Poniżej przedstawiono ogólną postać typowej struktury decyzyjnej występującej w większości języków programowania

Konstrukcja decyzyjna ocenia warunek przed wykonaniem instrukcji. Konstrukcje decyzyjne w Elm są klasyfikowane w następujący sposób -

Sr. No. Komunikat Opis
1 if ... then ... else oświadczenie Instrukcja if składa się z wyrażenia boolowskiego, po którym następuje wtedy, które jest wykonywane, jeśli wyrażenie zwróci prawdę, a w przeciwnym razie, które jest wykonywane, jeśli wyrażenie zwróci wartość fałsz
2 zagnieżdżona instrukcja if Możesz użyć jednego, jeśli ... to ... innego wewnątrz drugiego, jeśli.
3 oświadczenie przypadku Testuje wartość zmiennej na liście wartości.

if ... then ... else Instrukcja

Plik if…thenKonstrukcja oblicza warunek przed wykonaniem bloku kodu. Jeśli wynikiem wyrażenia logicznego jest prawda, zostanie wykonany blok kodu wewnątrz instrukcji then. Jeśli wynikiem wyrażenia logicznego jest fałsz, zostanie wykonany blok kodu wewnątrz instrukcji else.

W przeciwieństwie do innych języków programowania, w Elm musimy podać gałąź else. W przeciwnym razie Elm zgłosi błąd.

Składnia

if boolean_expression then statement1_ifTrue else statement2_ifFalse

Ilustracja

Wypróbuj następujący przykład w terminalu REPL.

> if 10>5 then "10 is bigger" else "10 is small"
"10 is bigger" : String

Zagnieżdżone jeśli

Zagnieżdżona instrukcja if jest przydatna do testowania wielu warunków. Składnia zagnieżdżonej instrukcji if jest podana poniżej -

if boolean_expression1 then statement1_ifTrue else if boolean_expression2 then statement2_ifTrue else statement3_ifFalse

Ilustracja

Wypróbuj następujący przykład w Elm REPL -

> score=80
80 : number
> if score>=80 then "Outstanding" else if score > = 70 then "good" else "average"
"Outstanding" : String

Oświadczenie Case

Instrukcja case może służyć do uproszczenia instrukcji if then else. Składnia instrukcji case jest następująca -

case variable_name of
   constant1 -> Return_some_value
   constant2 -> Return_some_value
   _ -> Return_some_value if none of the above values match

Instrukcja case sprawdza, czy wartość zmiennej odpowiada predefiniowanemu zestawowi stałych i zwraca odpowiednią wartość. Zwróć uwagę, że wartość zwracana przez każdy przypadek musi być tego samego typu. Jeśli wartość zmiennej nie pasuje do żadnej z podanych stałych, kontrola jest przekazywana do * default * (oznaczona // _) i zwracana jest odpowiednia wartość.

Ilustracja

Wypróbuj następujący przykład w Elm REPL -

> n = 10
10 : number
> case n of \
| 0 -> "n is Zero" \
| _ -> "n is not Zero"
"n is not Zero" : String

Powyższy fragment kodu sprawdza, czy wartość n wynosi zero. Kontrolka jest przekazywana do wartości domyślnej, która zwraca ciąg „n jest niezerowe”.

Elm to funkcjonalny język programowania. Elm używa koncepcji rekursji jako alternatywy dla tradycyjnych konstrukcji pętli.

W tym rozdziale omówiono koncepcję rekursji.

Rekursja

Niektóre języki programowania komputerów pozwalają modułowi lub funkcji wywoływać siebie. Ta technika jest znana jako rekurencja.

Ilustracja

W tym programie zobaczymy, jak używać rekurencji do pięciokrotnego wyświetlania powitania.

Step 1 − Create a file Loop.elm

Utwórz pętlę modułu i zdefiniuj funkcję sayHello. Funkcja sayHello przyjmuje wartość całkowitą jako dane wejściowe i zwraca wartość ciągu.

module Loop exposing(..)
//function signature
sayHello:Int ->String
//function implementation
sayHello n =
   case n of
   1 -> "Hello:1 "
   _ -> "Hello:" ++ toString (n) ++ " " ++ sayHello(n-1)

Funkcja sayHello sprawdza, czy przekazany parametr ma wartość 1. Jeśli parametr ma wartość 1, funkcja zwróci wartość, w przeciwnym razie utworzy napis Hello i wywoła tę samą funkcję.

Step 2 − Invoke sayHello from REPL

Otwórz elm REPL z bieżącego folderu projektu (lokalizacja pliku Loop.elm).

//import the module Loop
> import Loop exposing(..)
//invoke the sayHello function with parameter value as 5
> sayHello 5
"Hello:5 Hello:4 Hello:3 Hello:2 Hello:1 Hello:0 " : String
>

Ilustracja

Poniższy przykład wyświetla sumę n liczb przy użyciu rekursji.

> sumOfNos n =\
| if n==0 then 0 \
| else (n) + sumOfNos (n-1)
<function> : number -> number1

W elm REPL stworzyliśmy funkcję sumOfNos, która przyjmuje liczbę wejściową i sumuje wszystkie liczby od 0 do tej liczby.

Na przykład, jeśli przekażemy wejście jako 5, zsumuje 1 + 2 + 3 + 4 + 5, czyli 15 .

> ssumOfNos 5
15 : number

Wynik programu pokazano powyżej.

Funkcje są elementami składowymi programu Elm. Funkcja to zestaw instrukcji do wykonania określonego zadania.

Funkcje organizują program w logiczne bloki kodu. Po zdefiniowaniu funkcji można wywołać kod dostępu. Dzięki temu kod można ponownie wykorzystać. Ponadto funkcje ułatwiają odczytywanie i utrzymywanie kodu programu.

Kroki do korzystania z funkcji

Korzystanie z funkcji składa się z trzech kroków -

Deklaracja funkcji

Deklaracja funkcji informuje kompilator o nazwie funkcji, typie zwracanym i parametrach. Składnia deklarowania funkcji jest podana poniżej -

fn_name:data_type_of_the_parameters ->return_type

Deklaracja funkcji określa:

  • Nazwa funkcji.

  • Typ danych parametrów. Jest to opcjonalne, ponieważ funkcja może mieć parametry lub nie.

  • Typ danych wartości, którą funkcja zwróci. Funkcje w Elm muszą zawsze zwracać wartość, ponieważ Elm jest funkcjonalnym językiem programowania. W przeciwieństwie do funkcji w innych językach programowania, funkcje Elm nie używają słowa kluczowego return do zwracania wartości.

Definicja funkcji lub implementacja funkcji

Definicja funkcji zawiera rzeczywistą treść funkcji. Definicja funkcji określa sposób wykonania określonego zadania. Składnia definiowania funkcji jest następująca:

fn_name parameter1 parameter2 = statements

Wywołanie lub wywołanie funkcji

Funkcja musi zostać wywołana, aby ją wykonać. Składnia wywołania funkcji jest podana poniżej -

fn_name parameter1 parameter2

Ilustracja

Poniższy kod definiuje powitanie funkcji. Funkcja zwraca ciąg znaków „Hello”.

> greet = \
| if True then \
| "Hello" \
| else \
| "GoodBye"
"Hello" : String
> greet
"Hello" : String

Funkcje sparametryzowane

Parametry są mechanizmem przekazywania wartości do funkcji. Wartości parametrów są przekazywane do funkcji w momencie wywołania funkcji.

Ilustracja 1

Poniższy przykład definiuje funkcję fn_add . Funkcja przyjmuje dwie liczby jako parametry i zwraca ich sumę. Spróbuj wykonać następujące czynności w elm REPL -

> fn_add x y = x+y
<function> : number -> number -> number
> fn_add 10 20
30 : number

Ilustracja 2

Poniższy przykład definiuje funkcję sayHello. Funkcja sayHello przyjmuje i zwraca wartość typu String jako parametr i zwraca wartość typu String.

> sayHello name = "Hello "++ name
<function> : String -> String
> sayHello "Tutorialspoint"
"Hello Tutorialspoint" : String
>

Operator potoku

Aby zrozumieć operator potoku |>, rozważmy przykład, w którym mamy listę różnych ciągów znaków ["a", "b", "c"] . Teraz potrzebujemy jednego łańcucha oddzielonego znakiem -

Poniższy przykład pokazuje, jak to zrobić za pomocą String.join

> String.join "-" ["a","b","c","d","e","f"]
"a-b-c-d-e-f" : String

To samo można wykonać za pomocą operatora potoku |>. Operator potoku może służyć do łączenia wielu wywołań funkcji.

> ["a","b","c","d","e","f"] |> String.join "-"
"a-b-c-d-e-f" : String
> ["a","b","c","d","e","f"] |> List.reverse |> String.join "-"
"f-e-d-c-b-a" : String

W pierwszym przykładzie łączymy listę do metody join. W drugim przypadku ta sama lista jest przesyłana potokiem w celu odwrócenia funkcji, a następnie potokowana w celu dołączenia. Tak więc lista jest wyświetlana w odwróconej kolejności i połączona.

Sekwencja znaków Unicode jest nazywana ciągiem. W Elm ciągi znaków są umieszczone w "" podwójnych cudzysłowach . Ciąg to fragment tekstu, jak pokazano poniżej.

> "TutorialsPoint"
"TutorialsPoint" : String
> location = "Hyderabad" --variable
"Hyderabad" : String
> location
"Hyderabad" : String
>

Funkcje łańcuchowe

Poniżej podano niektóre typowe funkcje, których można używać do wykonywania zapytań lub manipulowania wartościami łańcuchów. Użyj REPL, aby wypróbować przykłady podane poniżej.

Sr. No metoda Opis
1 isEmpty: String -> Bool ciąg sprawdzający jest pusty
2 rewers: String -> String odwraca ciąg wejściowy
3 długość: String -> Int zwraca liczbę całkowitą
4 append: String -> String -> String dołącza dwa ciągi i zwraca nowy ciąg
5 append: String -> Sconcat: List String -> String dołącza listę ciągów i zwraca nowy ciąg
6 split: String -> String -> List String dzieli ciąg wejściowy za pomocą podanego separatora, zwraca listę ciągów
7 plasterek: Int -> Int -> String -> String zwraca podłańcuch z podanym początkowym, końcowym indeksem i ciągiem wejściowym
8 zawiera: Ciąg -> Ciąg -> Bool zwraca prawdę, jeśli drugi ciąg zawiera pierwszy
9 toInt: String -> Result.Result String Int analizuje String na Integer
10 toInt: String -> Result.Result String Int analizuje String na Integer
11 toFloat: String -> Result.Result String Float analizuje ciąg znaków do float
12 fromChar: Char -> String tworzy ciąg z podanego znaku.
13 toList: String -> List Char konwertuje ciąg znaków na listę znaków
14 fromList: List Char -> String konwertuje listę znaków na String
15 toUpper: String -> String konwertuje ciąg wejściowy na duże litery
16 trim: String -> String pozbywa się białych znaków po obu stronach łańcucha.
17 filter: (Char -> Bool) -> String -> String filtruje zestaw znaków z ciągu wejściowego
18 map: (Char -> Char) -> String -> String przekształca każdy znak w ciągu wejściowym

jest pusty

Ta funkcja może służyć do określenia, czy ciąg jest pusty. Ta funkcja zwraca wartość True, jeśli podany ciąg jest pusty.

Składnia

String.isEmpty String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.isEmpty
<function> : String -> Bool

Podpis funkcji pokazuje Bool jako typ zwracany, a typ wejściowy jako String -

Ilustracja

> String.isEmpty ""
True : Bool
> String.isEmpty "Tutorialspoint"
False : Bool
> location = "Hyderabad"
"Hyderabad" : String
> String.isEmpty location
False : Bool

odwrócić

Ta funkcja odwraca ciąg.

Składnia

String.reverse String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.reverse
<function> : String -> String

Podpis funkcji wyświetla ciąg jako typ zwracany, a typ wejściowy jako ciąg -

Ilustracja

> String.reverse "TutorialsPoint"
"tnioPslairotuT" : String

długość

Ta funkcja zwraca długość łańcucha.

Składnia

String.length String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.length
<function-> : String -> Int

Podpis funkcji pokazuje Int jako typ zwracany, a typ wejściowy jako String.

Ilustracja

> String.length "Mohtashim"
9 : Int

dodać

Ta funkcja zwraca nowy ciąg, dołączając dwa ciągi.

Składnia

String.append String_value1 String_value2

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.append
<function-> : String -> String -> String

Podpis zawiera dwa parametry wejściowe typu String i jeden parametr wyjściowy typu String

Ilustracja

> String.append "Tutorials" "Point"
TutorialsPoint : String

concat

Ta funkcja zwraca nowy ciąg, łącząc wiele ciągów w jeden.

Składnia

String.concat [String1,String2,String3]

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.concat
<function> : List String -> String

Signature of pokazuje listę parametrów wejściowych typu String i typu zwracanego ciągu

Ilustracja

> String.concat ["Hello","Tutorials","Point"]
HelloTutorialsPoint : String

rozdzielać

Ta funkcja dzieli ciąg przy użyciu podanego separatora.

Składnia

String.split string_seperator String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.split
<function> : String -> String -> List String

Podpis pokazuje dwa parametry wejściowe typu String i wyprowadza jako listę typu łańcuchowego.

Ilustracja

> String.split "," "Hello,Tutorials,Point"
["Hello","Tutorials","Point"] : List String

plasterek

Ta funkcja zwraca podłańcuch z podanym indeksem początkowym i końcowym. Indeksy ujemne są pobierane począwszy od końca listy. Wartość indeksu zaczyna się od zera.

Składnia

String.slice start_index end_index String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.slice
<function> : Int -> Int -> String -> String

Podpis przedstawia trzy parametry wejściowe i jeden typ zwracany.

Ilustracja

> String.slice 0 13 "TutorialsPoint"
"TutorialsPoin" : String

zawiera

Ta funkcja zwraca wartość True, jeśli drugi ciąg zawiera pierwszy.

Składnia

String.contains string1 string2

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.contains
<function> : String -> String -> Bool

Sygnatura pokazuje zwracany typ bool i dwa parametry wejściowe

Ilustracja

> String.contains "Point" "TutorialsPoint"
True : Bool

toInt

Ta funkcja konwertuje ciąg znaków na int.

Składnia

String.toInt string_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.toInt
<function> : String -> Result.Result String Int

Ponieważ toInt może zwracać błąd, zwracanym typem jest Result, czyli String lub Int.

Ilustracja

> String.toInt "20"
Ok 20 : Result.Result String Int
> String.toInt "abc"
Err "could not convert string 'abc' to an Int" : Result.Result String Int

unosić

Ta funkcja konwertuje ciąg znaków na liczbę zmiennoprzecinkową.

Składnia

String.toFloat string_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.toFloat
<function> : String -> Result.Result String Float

Ponieważ toFloat może zwracać błąd, zwracanym typem jest Result, czyli String lub Float.

Ilustracja

> String.toFloat "20.50"
Ok 20.5 : Result.Result String Float
> String.toFloat "abc"
Err "could not convert string 'abc' to a Float" : Result.Result String Float

fromChar

Ta funkcja tworzy łańcuch z podanego znaku.

Składnia

String.fromChar character_value

Aby sprawdzić podpis funkcji wpisz następujący w elm REPL -

> String.fromChar
<function> : Char -> String

Podpis przedstawia ciąg jako typ zwracany, a dane wejściowe jako typ Char

Ilustracja

> String.fromChar 'c'
"c" : String

notować

Ta funkcja konwertuje ciąg znaków na listę znaków.

Składnia

String.toList string_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.toList
<function> : String -> List Char

Sygnatury pokazują, że funkcja zwraca listę znaków i pobiera ciąg.

Ilustracja

> String.toList "tutorialspoint"
['t','u','t','o','r','i','a','l','s','p','o','i','n','t'] : List Char

fromList

Ta funkcja konwertuje listę znaków na łańcuch.

Składnia

String.fromList list_of_characters

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.fromList
<function> : List Char -> String

Sygnatury pokazują, że funkcja zwraca listę znaków i pobiera ciąg.

Ilustracja

> String.fromList ['h','e','l','l','o']
"hello" : String

toUpper

Ta funkcja konwertuje ciąg znaków na wielkie litery.

Składnia

String.toUpper String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.toUpper
<function> : String -> String

Ilustracja

> String.toUpper "hello"
"HELLO" : String

obniżyć

Ta funkcja konwertuje ciąg znaków na małe litery.

Składnia

String.toLower String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.toLower
<function> : String -> String

Ilustracja

> String.toLower "AbCd"
"abcd" : String

trym

Ta funkcja usuwa białe znaki po obu stronach łańcucha.

Składnia

String.trim String_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.trim
<function> : String -> String

Ilustracja

> String.trim "tutorialspoint "
"tutorialspoint" : String

filtr

Ta funkcja filtruje zestaw znaków z wejściowego ciągu znaków. Zachowaj tylko te postacie, które przejdą test.

Składnia

String.filter test_function string_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.filter
<function> : (Char -> Bool) -> String -> String

Podpis pokazuje, że filtr przyjmuje dwa parametry wejściowe i zwraca ciąg. Pierwszy parametr to funkcja, która ma dane wejściowe Char i zwraca wartość Bool.

Ilustracja

W tym przykładzie przekazujemy Char.isUpper jako parametr do metody filtrującej; zwraca wszystkie wielkie litery, jak pokazano poniżej.

> import Char
> String.filter Char.isUpper "abcDEF"
"DEF" : String

mapa

Ta funkcja przyjmuje ciąg znaków i przekształca każdy znak w ciągu.

Składnia

String.filter mapping_function string_value

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> String.map
<function> : (Char -> Char) -> String -> String

Ilustracja

Poniższy przykład zastępuje znak o znakiem @ -

> String.map (\c -> if c == 'o' then '@' else c) "TutorialsPoint"
"Tut@rialsP@int" : String

Struktury danych Lista, Krotki i Rekord mogą służyć do przechowywania zbioru wartości.

W tym rozdziale omówiono, jak używać listy w aplikacji Elm.

Lista to zbiór jednorodnych wartości. Wszystkie wartości na liście muszą mieć ten sam typ danych.

Weź pod uwagę następujące ograniczenia podczas używania zmiennych do przechowywania wartości -

  • Zmienne mają charakter skalarny. Innymi słowy, w momencie deklaracji zmienna może mieć tylko jedną wartość. Oznacza to, że aby zapisać w programie n wartości, potrzebnych będzie n deklaracji zmiennych. Dlatego użycie zmiennych jest niewykonalne, gdy trzeba przechowywać większy zbiór wartości.

  • Zmiennym w programie przydzielana jest pamięć w kolejności losowej, co utrudnia pobranie / odczytanie wartości w kolejności ich deklaracji.

Składnia

List_name = [value1,value2,value3.....valuen]

Ilustracja

Poniższy przykład pokazuje, jak używać List w Elm. Wypróbuj ten przykład w elm REPL -

> myList1 = [10,20,30]
[10,20,30] : List number
> myList2 = ["hello","world"]
["hello","world"] : List String

Jeśli spróbujemy dodać wartości różnych typów do listy, kompilator zgłosi błąd niezgodności typu. Jest to pokazane poniżej.

> myList = [1,"hello"]
-- TYPE MISMATCH 
--------------------------------------------- 
repl-temp-000.elm

The 1st and 2nd entries in this list are different types of values.

4| [1,"hello"]
^^^^^^^
The 1st entry has this type:
   number
But the 2nd is:
   String

Operacje na listach

Poniższa tabela przedstawia typowe operacje na liście -

Sr. No metoda Opis
1 isEmpty: List a -> Bool sprawdza, czy lista jest pusta
2 rewers: List a -> Bool odwraca listę wejść
3 długość: Lista a -> Int zwraca rozmiar listy
4 maksimum: Lista porównywalna -> Może. może być porównywalna zwraca wartość maksymalną
5 minimum: Lista porównywalna -> Może. może być porównywalna zwraca wartość minimalną
6 suma: Numer listy -> numer zwraca sumę wszystkich elementów na liście
7 produkt: Numer listy -> numer sprawdza, czy lista jest pusta
8 sort: Lista porównywalna -> Lista porównywalna sortuje listę w porządku rosnącym
9 concat: Lista (Lista a) -> Lista a scala kilka list w jedną
10 append: Lista a -> Lista a -> Lista a łączy dwie listy
11 zakres: Int -> Int -> List Int zwraca listę liczb od początku do końca
12 filter: (a -> Bool) -> Lista a -> Lista a filtruje listę wartości z listy wejściowej
13 head: Lista a -> Może. Może zwraca pierwszy element z listy
14 ogon:: Lista a -> Może.Może (Lista a) zwraca wszystkie elementy oprócz głowy

jest pusty

Ta funkcja zwraca prawdę, jeśli lista jest pusta.

Składnia

List.isEmpty list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.isEmpty
<function> : List a -> Bool

Ilustracja

> List.isEmpty
<function> : List a -> Bool

> List.isEmpty [10,20,30]
False : Bool

odwrócić

Ta funkcja odwraca listę.

Składnia

List.reverse list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.reverse
<function> : List a -> List a

Ilustracja

> List.reverse [10,20,30]
[30,20,10] : List number

długość

Ta funkcja zwraca długość listy.

Składnia

List.length list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.length
<function> : List a -> Int

Ilustracja

> List.length [10,20,30]
3 : Int

maksymalny

Ta funkcja zwraca maksymalny element z niepustej listy.

Składnia

List.maximum list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.maximum
<function> : List comparable -> Maybe.Maybe comparable

Ilustracja

> List.maximum [10,20,30]
Just 30 : Maybe.Maybe number
> List.maximum []
Nothing : Maybe.Maybe comparable

minimum

Ta funkcja zwraca minimalny element z niepustej listy.

Składnia

List.minimum list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.minimum
<function> : List comparable -> Maybe.Maybe comparable

Ilustracja

> List.minimum [10,20,30]
Just 10 : Maybe.Maybe number

suma

Ta funkcja zwraca sumę wszystkich elementów na liście.

Składnia

List.sum list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.sum
<function> : List number -> number

Ilustracja

> List.sum [10,20,30]
60 : number

produkt

Ta funkcja zwraca iloczyn wszystkich elementów na liście.

Składnia

List.product list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

<function>  : List number ->  number

Ilustracja

List.product [10,20,30]
6000 : number

sortować

Ta funkcja sortuje wartości na liście od najniższej do najwyższej.

Składnia

List.sort list_name

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.sort
<function> : List comparable -> List comparable

Ilustracja

> List.sort [10,20,30]
[10,20,30] : List number

concat

Ta funkcja łączy kilka list w jedną listę.

Składnia

List.concat [ [list_name1],[list_name2],[list_name3],.....[list_nameN] ]

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.concat
<function> : List (List a) -> List a

Ilustracja

> List.concat [[10,20], [30,40],[50,60]]
[10,20,30,40,50,60] : List number

dodać

Ta funkcja łączy dwie listy.

Składnia

List.append [list_name1] [list_name2]

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.append
<function> : List a -> List a -> List a

Ilustracja

> List.append [10,20] [30,40]
[10,20,30,40] : List number

Operator ++ może również służyć do dołączania listy do innej. Pokazuje to poniższy przykład -

> [10.1,20.2] ++ [30.3,40.4]
[10.1,20.2,30.3,40.4] : List Float

zasięg

Ta funkcja tworzy listę liczb, przy czym każdy element rośnie o jeden. Najniższa i najwyższa liczba, która powinna znajdować się na liście, jest przekazywana do funkcji.

Składnia

List.range start_range end_range

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.range
<function> : Int -> Int -> List Int

Ilustracja

> List.range 1 10
[1,2,3,4,5,6,7,8,9,10] : List Int

filtr

Ta funkcja filtruje zestaw wartości z listy wejściowej. Zachowaj tylko wartości, które przeszły test.

Składnia

List.filter test_function input_list

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.filter
<function> : (a -> Bool) -> List a -> List a

Ilustracja

Poniższy przykład filtruje wszystkie liczby parzyste z listy wejściowej

> List.filter (\n -> n%2==0) [10,20,30,55]
[10,20,30] : List Int

głowa

Ta funkcja zwraca pierwszy element z listy wejściowej.

Składnia

List.head input_list

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.head
<function> : List a -> Maybe.Maybe a

Ilustracja

> List.head [10,20,30,40]
Just 10 : Maybe.Maybe number
> List.head []
Nothing : Maybe.Maybe a

ogon

Ta funkcja zwraca wszystkie elementy po pierwszym na liście.

Składnia

List.tail input_list

Aby sprawdzić podpis funkcji, wpisz w elm REPL -

> List.tail
<function> : List a -> Maybe.Maybe (List a)

Ilustracja

> List.tail [10,20,30,40,50]
Just [20,30,40,50] : Maybe.Maybe (List number)
> List.tail [10]
Just [] : Maybe.Maybe (List number)
> List.tail []
Nothing : Maybe.Maybe (List a)

Korzystanie z operatora Cons

Operator wad (::) dodaje element na początek listy.

Ilustracja

> 10::[20,30,40,50]
[10,20,30,40,50] : List number

Nowy element do dodania i typ danych wartości na liście muszą być zgodne. Kompilator zgłasza błąd, jeśli typy danych nie są zgodne.

> [1,2,3,4]::[5,6,7,8]
-- TYPE MISMATCH ---------------------------------
------------ repl-temp-000.elm

The right side of (::) is causing a type mismatch.

3| [1,2,3,4]::[5,6,7,8]
			  ^^^^^^^^^
(::) is expecting the right side to be a:

   List (List number)

But the right side is:

   List number
Hint: With operators like (::) I always check the left side first. If it seems fine, 
I assume it is correct and check the right side. So the 
problem may be in how the left and right arguments interact.

Listy są niezmienne

Sprawdźmy, czy listy są niezmienne w Elm. Pierwsza lista myList po połączeniu z wartością 1 tworzy nową listę i jest zwracana do myListCopy . Dlatego jeśli wyświetlimy listę początkową, jej wartości nie ulegną zmianie.

> myList = [10,20,30]
[10,20,30] : List number
> myListCopy = 1::myList
[1,10,20,30] : List number
> myList
[10,20,30] : List number
>myList == myListCopy
False : Bool

Czasami może zaistnieć potrzeba przechowywania zbioru wartości różnych typów. Elm daje nam strukturę danych zwaną krotką, która służy do tego celu.

Krotka reprezentuje niejednorodny zbiór wartości. Innymi słowy, krotki umożliwiają przechowywanie wielu pól różnych typów. Krotka przechowuje stałą liczbę wartości. Krotki są przydatne, gdy chcesz zwrócić z funkcji wiele wartości różnych typów. Te struktury danych są niezmienne, podobnie jak inne typy w elm.

Składnia

(data1,data2)

Prosty przykład pokazano poniżej -

> ("TuotrialsPoint",5,True,"Hyderabad")
("TuotrialsPoint",5,True,"Hyderabad") : ( String, number, Bool, String )

W kolejnych sekcjach dowiemy się o różnych operacjach na krotkach.

pierwszy

Ta operacja wyodrębnia pierwszą wartość z krotki.

Składnia

Tuple.first tuple_name
> Tuple.first
<function> : ( a1, a2 ) -> a1

Ilustracja

> Tuple.first (10,"hello")
10 : number

druga

Plik second operacja krotki wyodrębnia drugą wartość z krotki.

Składnia

Tuple.second tuple_name
> Tuple.second
<function> : ( a1, a2 ) -> a2

Ilustracja

> Tuple.second (10,"hello")
"hello" : String

Lista krotek

Lista może przechowywać krotki. Jeśli krotki są używane wewnątrz listy, upewnij się, że wszystkie mają ten sam typ danych i taką samą liczbę parametrów.

Ilustracja

> [("hello",20),("world",30)]
[("hello",20),("world",30)] : List ( String, number )

Krotka z funkcją

Funkcja może zwracać krotki. Ponadto krotki mogą być przekazywane jako parametry do funkcji.

Ilustracja 1

Poniższy przykład definiuje funkcję fn_checkEven. Ta funkcja przyjmuje wartość całkowitą jako parametr i zwraca krotkę.

> fn_checkEven no = \
   if no%2 == 0 then \
      (True,"The no is Even")\
   else \
      (False,"No is not even")
<function> : Int -> ( Bool, String )
> fn_checkEven 10
(True,"The no is Even") : ( Bool, String )
> fn_checkEven 11
(False,"No is not even") : ( Bool, String )
>

Ilustracja 2

Poniższy kod przekazuje krotkę jako parametr do funkcji.

> fn_add (a,b) = \
| a+b
<function> : ( number, number ) -> number
> fn_add (10,20)
30 : number

Funkcja fn_add pobiera krotkę z 2 wartościami liczbowymi i zwraca ich sumę.

Destrukturyzacja

Destrukturyzacja polega na rozbiciu krotki na indywidualne wartości. Aby uzyskać dostęp do poszczególnych wartości w krotce z trzema lub więcej elementami, używamy destrukturyzacji. Tutaj przypisujemy każdą wartość w krotce do różnych zmiennych. Używając _ można zdefiniować symbole zastępcze dla wartości, które będą ignorowane lub pomijane.

Ilustracja

> (first,_,_) = (10,20,30)
10 : number
> first
10 : number

Ilustracja

W tym przykładzie do destrukcji użyjemy składni bloku let..in. Blok let zawiera zmienne, a blok in zawiera wyrażenia, które powinny zostać ocenione i wartość, która powinna zostać zwrócona.

> t1 = (10,20,30)
(10,20,30) : ( number, number1, number2 )
> let \
(a,b,c) = t1 \
in\
a + b +c
60 : number

Deklarujemy zmienne abc w klauzuli let i uzyskujemy do nich dostęp za pomocą klauzuli in.

Struktura danych rekordu w Elm może służyć do reprezentowania danych jako par klucz-wartość. Rekord może służyć do organizowania powiązanych danych, aby umożliwić łatwy dostęp i aktualizację danych. Rekordy wiązów są podobne do obiektów w JavaScript. Elementy danych w rekordzie nazywane są polami.

Definiowanie rekordu

Użyj następującej składni, aby zdefiniować rekord -

Składnia

record_name = {fieldname1 = value1, fieldname2 = value2....fieldnameN = valueN}

Rekord może przechowywać dane wielu typów. Nazwy pól w rekordzie muszą być zgodne z ogólnymi zasadami nazywania identyfikatora Elm.

Dostęp do wartości rekordów

Aby uzyskać dostęp do poszczególnych pól w rekordzie, użyj następującej składni.

Składnia

record_name.fieldname

LUB

.fieldname record_name

Ilustracja

Spróbuj wykonać następujące czynności w REPL Elm -

> company = {name="TutorialsPoint",rating=4.5}
{ name = "TutorialsPoint", rating = 4.5 } : { name : String, rating : Float }
> company.name
"TutorialsPoint" : String
> .rating company
4.5 : Float

Używanie rekordu z listą

Rekord może być przechowywany na liście. Wszystkie wartości pól rekordu powinny być tego samego typu.

Składnia

list_name = [ {field_name1 = value1},{field_name1 = value2}]

LUB

list_name = [record_name1, record_name2, record_name3....record_nameN]

Ilustracja

Spróbuj wykonać następujące czynności w Elm REPL -

> [{name = "Mohtashim"},{name = "kannan"}]
[{ name = "Mohtashim" },{ name = "kannan" }] : List { name : String }
> record1 = {name = "FirstRecord"}
{ name = "FirstRecord" } : { name : String }
> record2 = {name = "SecondRecord"}
{ name = "SecondRecord" } : { name : String }
> recordList = [record1,record2]
[{ name = "FirstRecord" },{ name = "SecondRecord" }] : List { name : String }

Zaktualizuj rekord

Zapisy są niezmienne w Elm. Kiedy rekord jest aktualizowany, zwracany jest nowy rekord ze zaktualizowanymi wartościami. Podczas aktualizacji rekordu pole może zawierać wartość innego typu.

Składnia

{record_name | field_name1 = new_value1, field_name2 = new_value2,field_name3 = new_value3....field_nameN = new_valueN}

Ilustracja

Spróbuj wykonać następujące czynności w Elm REPL -

> record1 = {name="FirstRecord"}
{ name = "FirstRecord" } : { name : String }
> record1_updated = {record1 | name = "FirstRecordUpdate"}
{ name = "FirstRecordUpdate" } : { name : String }
> record1
{ name = "FirstRecord" } : { name : String }
> record1 == record1_updated
False : Bool

Ilustracja

Poniższy przykład aktualizuje wiele pól rekordu. Spróbuj wykonać następujące czynności w Elm REPL -

> record3 = {a = 1,b = 2,c = 3,d = 4,e = 5}
{ a = 1, b = 2, c = 3, d = 4, e = 5 }
: { a : number, b : number1, c : number2, d : number3, e : number4 }
> record4 = {record3 | d=400 ,e=500}
{ a = 1, b = 2, c = 3, d = 400, e = 500 }
: { a : number2, b : number3, c : number4, d : number, e : number1 }
>

Aliasy typów

Alias ​​typu definiuje schemat rekordu. Innymi słowy, alias typu określa, które pola mogą być przechowywane w rekordach, oraz typ wartości, które te pola mogą przechowywać. Dlatego programista nie popełni błędu polegającego na pominięciu określonego atrybutu podczas przypisywania wartości.

Składnia

type alias alias_name = {field_name1:data_type,field_name2:data_type,....field_nameN:data_type}

Ilustracja

Wykonaj następujące czynności w Elm REPL -

> type alias Developer = { name:String,location:String,age:Int}
> dev1 = Developer "kannan" "Mumbai" 20
{ name = "kannan", location = "Mumbai", age = 20 } : Repl.Developer
> dev2 = Developer "mohtashim" "hyderabad" 20
{ name = "mohtashim", location = "hyderabad", age = 20 } : Repl.Developer
>

Teraz, jeśli zapomnisz wpisać lokalizację i wiek, instrukcja zwraca funkcję, która ma parametry wejściowe dla pól lokalizacji i wieku.

> dev3 = Developer "Bhagavati"
<function> : String -> Int -> Repl.Developer
We can invoke the function as shown below and pass to it the values for location and age fields.
> dev3 "Pune" 25
{ name = "Bhagavati", location = "Pune", age = 25 } : Repl.Developer

Błąd to nieoczekiwany stan w programie. Błędy mogą wystąpić w czasie kompilacji lub w czasie wykonywania. Błędy czasu kompilacji występują podczas kompilacji programu (na przykład błąd w składni programu), podczas gdy podczas wykonywania programu występują błędy czasu wykonywania. W przeciwieństwie do innych języków programowania, Elm nie generuje błędów w czasie wykonywania.

Rozważ aplikację, która akceptuje wiek użytkownika. Aplikacja powinna zgłosić błąd, jeśli wiek jest zerowy lub ujemny. W takim przypadku aplikacja Elm może użyć koncepcji obsługi błędów, aby jawnie zgłosić błąd w czasie wykonywania, jeśli użytkownik wprowadzi zero lub wartość ujemną jako wiek. Obsługa błędów określa sposób postępowania, jeśli podczas wykonywania programu wydarzy się coś nieoczekiwanego.

Język programowania Elm obsługuje błędy w następujący sposób -

  • MayBe
  • Result

Może

Rozważ funkcję wyszukiwania w aplikacji. Funkcja wyszukiwania zwraca powiązane dane, jeśli znaleziono słowo kluczowe wyszukiwania, w przeciwnym razie nic nie zwraca. Ten przypadek użycia można zaimplementować w Elm przy użyciu typu MayBe.

Składnia

variable_name:MayBe data_type

Zmienna typu MayBe może zawierać jedną z następujących wartości -

  • Just some_Value - jest używana, jeśli istnieją prawidłowe dane.

  • Nic - jest używane, jeśli wartość jest nieobecna lub nieznana. W innych językach programowania nic nie jest równoważne z wartością null.

Ilustracja

Poniższy przykład pokazuje, jak używać typu MayBe ze zmiennymi i funkcją.

Step 1 - Utwórz plik MayBeDemo.elm plik i dodaj do niego następujący kod

-- MayBeDemo.elm
module MayBeDemo exposing(..)
import Maybe

--declaring a MayBe variable and assigning value to it
userName : Maybe String
userName = Just "Mohtashim"

--declaring a MayBe variable and assigning value to it
userAge :Maybe Int
userAge = Just 20

--declaring a MayBe variable and assigning value to it
userSalary:Maybe Float
userSalary = Nothing

--declaring a custom type
type Country = India | China | SriLanka

--defining a function that takes a String parameter as input and returns a value of type MayBe

getCountryFromString : String -> Maybe Country
getCountryFromString p =
case p of
   "India"
      -> Just India
   "China"
      -> Just China
   "SriLanka"
      -> Just SriLanka
   _
      -> Nothing

Step 2 - Zaimportuj moduł w elm repl i wykonaj jak podano poniżej

E:\ElmWorks\ErroApp> elm repl
---- elm-repl 0.18.0 -----------------------------------------------------------
:help for help, :exit to exit, more at 
      
        -------------------------------------------------------------------------------- > import MayBeDemo exposing(..) > userName Just "Mohtashim" : Maybe.Maybe String > userAge Just 20 : Maybe.Maybe Int > userSalary Nothing : Maybe.Maybe Float > getCountryFromString "India" Just India : Maybe.Maybe MayBeDemo.Country > getCountryFromString "india" Nothing : Maybe.Maybe MayBeDemo.Country 
      

Funkcja sprawdza, czy wartość przekazana do funkcji to Indie, Chiny czy SriLanka. Jeśli wartość parametru nie pasuje do żadnego z nich, nic nie zwraca.

Wynik

Rozważmy przykład, w którym aplikacja musi zweryfikować jakiś warunek i zgłosić błąd, jeśli warunek nie jest spełniony. Aby to osiągnąć, można użyć typu Wynik. Typ Wynik powinien być używany, jeśli aplikacja chce jawnie zgłosić błąd i zwrócić szczegółowe informacje o tym, co poszło nie tak.

Składnia

Deklaracja typu Result przyjmuje dwa parametry - typ danych błędu (zwykle String) i typ danych wyniku, który ma zostać zwrócony, jeśli wszystko pójdzie dobrze.

type Result error_type data_value_type
= Ok data_value
| Err error_message

Typ Wynik zwraca jedną z następujących wartości -

  • Ok some_value - reprezentuje wynik do zwrócenia

  • Err - reprezentuje komunikat o błędzie, który ma zostać zwrócony, jeśli oczekiwane warunki nie są spełnione.

Ilustracja 1

Wypróbuj następujący przykład w Elm REPL -

> String.toInt
<function> : String -> Result.Result String Int
-- successful result
> String.toInt "10"
Ok 10 : Result.Result String Int
-- unsuccessful result , Error
> String.toInt "a"
Err "could not convert string 'a' to an Int" : Result.Result String Int

Funkcja String.toInt zwraca wartość całkowitą, jeśli przekazany parametr jest prawidłowy. Jeśli parametr nie jest liczbą, funkcja zwraca błąd.

Ilustracja 2

Poniższy przykład akceptuje wiek jako parametr. Funkcja zwraca wiek, jeśli jest między 0 a 135, w przeciwnym razie zwraca odpowiedni komunikat o błędzie.

Step 1 - Utwórz plik ResultDemo.elm i dodaj do niego następujący kod.

--ResultDemo.elm
module ResultDemo exposing(..)

userId : Result String Int
userId = Ok 10

emailId : Result String Int
emailId = Err "Not valid emailId"

isReasonableAge : String -> Result String Int
isReasonableAge input =
   case String.toInt input of
      Err r ->
         Err "That is not a age!"

   Ok age ->
      if age < 0 then
         Err "Please try again ,age can't be negative"
      else if age > 135 then
         Err "Please try agian,age can't be this big.."

   else
      Ok age

Step 2 - Zaimportuj moduł w pakiecie elm i wykonaj jak podano poniżej

E:\ElmWorks\ElmRepo\15_ErrorHandling\15_Code> elm repl
---- elm-repl 0.18.0 -----------------------------------------------------------
:help for help, :exit to exit, more at <https://github.com/elm-lang/elm-repl>
--------------------------------------------------------------------------------
> import ResultDemo exposing (..)
> userId
Ok 10 : Result.Result String Int
> emailId
Err "Not valid emailId" : Result.Result String Int
> isReasonableAge "10"
Ok 10 : Result.Result String Int
> isReasonableAge "abc"
Err "That is not a age!" : Result.Result String Int

W tym rozdziale omówimy standardowy sposób tworzenia aplikacji na platformie Elm. Elm używa wzorca architektonicznego podobnego do wzorca Model-View-Controller.

Poniżej znajdują się cztery główne części Architektury Wiązów.

  • Model
  • View
  • Message
  • Update

Jak działa architektura Elm

Plik modelzawiera stan aplikacji. Na przykład, jeśli aplikacja wyświetla listę klientów, stan będzie zawierał dane każdego klienta. Aby wyświetlić stan w reprezentacyjny sposób, aview/ html musi zostać wygenerowany. Gdy użytkownik wejdzie w interakcję z widokiem, naciskając przycisk lub wpisując dane w formularzu, widok generuje sygnały o nazwiemessages. Wiadomości są przekazywane doupdatemetoda, która ocenia komunikaty i podejmuje odpowiednie działania. Dlatego metoda aktualizacji wygeneruje nowy model.

Nowy model generuje nowy widok. Widok doprowadzi do nowych interakcji od użytkownika do komunikatów sygnalizacyjnych, które trafią do funkcji aktualizacji. Ponadto funkcja tworzy nowy model. Tak więc cykl powtarza się, jak pokazano na powyższym schemacie.

Model

Model dotyczy stanu aplikacji. Składnia do definiowania modelu jest podana poniżej -

-- Model syntax

type alias Model = {
   property1:datatype,
   proptery2:datatype
...
}

Aby stworzyć model, musimy najpierw stworzyć szablon ze wszystkimi wymaganymi właściwościami. Każda właściwość określa stan aplikacji.

Widok

Widok to wizualna reprezentacja stanu aplikacji. The View wie, jak pobrać dane i wygenerować z nich stronę internetową. Gdy użytkownik wchodzi w interakcję z widokiem, może manipulować stanem, generując komunikaty. Składnia definiowania widoku jest podana poniżej -

--View Syntax
view model =some_implementation

Wiadomość

Komunikat to żądanie od użytkownika zmiany stanu aplikacji. Komunikaty są przekazywane jako parametr do funkcji aktualizacji.

--Message Syntax
type Message = Message1 |Message2 ...

Składnia przedstawia typ Message. Aplikacja elm zmieni stan na podstawie przekazanych do niej wiadomości. Decyzje te podejmowane są w metodzie aktualizacji.

Aktualizacja

Funkcja aktualizacji interpretuje komunikaty, które są do niej przekazywane jako parametr, i aktualizuje model.

--Update Syntax
update Message_type model =
   some_implementation

Funkcja aktualizacji trwa Message i Modeluj jako parametry.

Menedżer pakietów to narzędzie wiersza poleceń, które automatyzuje proces instalowania, uaktualniania, konfigurowania i usuwania pakietów w aplikacji.

Podobnie jak JavaScript ma menedżera pakietów o nazwie npm, elm ma menedżera pakietów o nazwie elm-package .

Menedżer pakietów wykonuje następujące trzy zadania -

  • Instaluje wszystkie zależności, których potrzebuje aplikacja elm
  • Publikuje pakiety niestandardowe
  • Określa wersję pakietu, gdy jesteś gotowy do publikacji i aktualizacji.

Polecenia programu Elm Package Manager

Poniższa tabela zawiera listę różnych poleceń menedżera pakietów Elm -

Sr. No. Komenda Składnia Opis
1 zainstalować instalacja pakietu elm Instaluje pakiety do użytku lokalnego
2 publikować publikacja pakietu elm Publikuje twój pakiet w katalogu centralnym
3 guz wypukłość pakietu elm Zwiększa numery wersji w oparciu o zmiany interfejsu API
4 różn Różnica w pakiecie elm Pobiera różnice między dwoma interfejsami API

Aby opublikować swój pakiet, musisz hostować kod źródłowy na GitHub i mieć odpowiednią wersję oznaczoną tagiem git. Poniższa ilustracja pokazuje, jak używać menedżera pakietów elm do ściągania zależności zewnętrznych.

Ilustracja - Instalowanie pakietu svg

W tym przykładzie zobaczymy, jak zintegrować skalowalną grafikę wektorową (SVG) z aplikacją elm.

Step 1 - Utwórz folder elmSvgApp

Step 2 - Zainstaluj pakiet svg za pomocą następującego polecenia -

elm-package install elm-lang/svg

Step 3- Zainstaluj Utwórz plik SvgDemo.elm i wpisz zawartość podaną poniżej. Importujemy moduł Svg, aby narysować prostokąt o wymiarach 100x100 i wypełnić kolorem czerwonym.

import Svg exposing (..)
import Svg.Attributes exposing (..)

main =
   svg
   [ width "120"
   , height "120"
   , viewBox "0 0 120 120"
   ]
   [ rect
      [ x "10"
      , y "10"
      , width "100"
      , height "100"
      , rx "15"
      , ry "15"
      ,fill "red"
      ]
      []
   ]

Step 4- Teraz skompiluj projekt używając elm make. \ SvgDemo.elm. Spowoduje to wygenerowanie pliku index.html, jak pokazano poniżej -

Wiadomość jest składnikiem architektury Elm. Te komponenty są generowane przez View w odpowiedzi na interakcję użytkownika z interfejsem aplikacji. Wiadomości reprezentują żądania użytkowników dotyczące zmiany stanu aplikacji.

Składnia

--Message Syntax
type Message = some_message1 |some_message2 ...|some_messageN

Ilustracja

Poniższy przykład to prosta aplikacja licznika. Aplikacja zwiększa i zmniejsza wartość zmiennej o 1, gdy użytkownik kliknie odpowiednio przyciski Dodaj i Odejmij.

Aplikacja będzie miała 4 komponenty. Komponenty opisano poniżej -

Wiadomość

Wiadomości w tym przykładzie będą wyglądać następująco:

type Message = Add | Subtract

Model

Model reprezentuje stan aplikacji. W aplikacji licznika definicja modelu jest podana poniżej; początkowy stan licznika będzie wynosił zero.

model = 0

Widok

Widok reprezentuje wizualne elementy aplikacji. Widok zawiera dwa przyciski (+) i (-). Komunikaty Dodaj i Odejmij są generowane przez Widok, gdy użytkownik kliknie odpowiednio przyciski + i -. Zmodyfikowana wartość modelu jest następnie wyświetlana w widoku.

view model =
-- invoke text function
h1[]
[   div[] [text "CounterApp from TutorialsPoint" ]
   ,button[onClick Subtract] [text "-"]
   ,div[][text (toString model)]
   ,button[onClick Add] [text "+"]
]

Aktualizacja

Ten komponent zawiera kod, który powinien zostać wykonany dla każdej wiadomości wygenerowanej przez widok. Pokazuje to poniższy przykład -

update msg model =
case msg of
Add -> model+1
Subtract -> model-1

Kładąc wszystko razem

Step 1- Utwórz folder MessagesApp i plik MessagesDemo.elm

Step 2 - Dodaj następujący kod w pliku elm -

import Html exposing (..)
import Html.Events exposing(onClick)

model = 0 -- Defining the Model

--Defining the View

view model =
   h1[]
   [  div[] [text "CounterApp from TutorialsPoint" ]
      ,button[onClick Subtract] [text "-"]
      ,div[][text (toString model)]
      ,button[onClick Add] [text "+"]
   ]

--Defining the Messages

type Message = Add | Subtract

--Defining Update

update msg model =
case msg of
   Add -> model+1
   Subtract -> model-1

-- Define the main method
main =
   beginnerProgram
   {
      model=model
      ,view=view
      ,update=update
   }

Step 3 - Wykonaj elm make commandw terminalu. Plikelm make command kompiluje kod i generuje plik HTML z pliku .elm utworzonego powyżej.

C:\Users\dell\elm\MessagesApp> elm make .\MessageDemo.elm
Some new packages are needed. Here is the upgrade plan.

   Install:
      elm-lang/core 5.1.1
      elm-lang/html 2.0.0
      elm-lang/virtual-dom 2.0.4

Do you approve of this plan? [Y/n] y
Starting downloads...

   ΓùÅ elm-lang/html 2.0.0
   ΓùÅ elm-lang/virtual-dom 2.0.4

ΓùÅ elm-lang/core 5.1.1
Packages configured successfully!
Success! Compiled 38 modules.
Successfully generated index.html

Step 4 - Otwórz index.html i sprawdź działanie, jak pokazano poniżej -

Wiąz - Polecenia

W poprzednich rozdziałach omówiliśmy różne elementy architektury wiązu i ich funkcje. Użytkownik i aplikacja komunikują się ze sobą za pomocą Wiadomości.

Rozważmy przykład, w którym aplikacja musi komunikować się z innymi składnikami, takimi jak serwer zewnętrzny, interfejsy API, mikrousługa itp., Aby obsłużyć żądanie użytkownika. Można to osiągnąć za pomocą poleceń w wiązie. Wiadomości i polecenia nie są synonimami. Wiadomości reprezentują komunikację między użytkownikiem końcowym a aplikacją, podczas gdy polecenia reprezentują sposób, w jaki aplikacja Elm komunikuje się z innymi jednostkami. Polecenie jest wyzwalane w odpowiedzi na wiadomość.

Poniższy rysunek przedstawia przepływ pracy złożonej aplikacji Elm -

Użytkownik wchodzi w interakcję z widokiem. Widok generuje odpowiedni komunikat na podstawie akcji użytkownika. Komponent aktualizacji otrzymuje ten komunikat i uruchamia polecenie.

Składnia

Składnia definiowania polecenia jest następująca:

type Cmd msg

Wiadomość wygenerowana przez widok jest przekazywana do polecenia.

Ilustracja

Poniższy przykład wysyła żądanie do interfejsu API i wyświetla wynik z interfejsu API.

Aplikacja przyjmuje numer od użytkownika, przekazuje go do Numbers API. Ten interfejs API zwraca fakty związane z liczbą.

Różne elementy aplikacji są następujące -

Moduł HTTP

Moduł HTTP firmy Elm służy do tworzenia i wysyłania żądań HTTP. Ten moduł nie jest częścią podstawowego modułu. Do zainstalowania tego pakietu użyjemy menedżera pakietów elm.

API

W tym przykładzie aplikacja będzie komunikować się z interfejsem API Numbers - „http://numbersapi.com/#42”.

Widok

Widok aplikacji zawiera pole tekstowe i przycisk.

view : Model -> Html Msg
view model =
   div []
      [ h2 [] [text model.heading]
      ,input [onInput Input, value model.input] []
      , button [ onClick ShowFacts ] [ text "show facts" ]
      , br [] []
      , h3 [][text model.factText]
      ]

Model

Model reprezentuje wartość wprowadzoną przez użytkownika i wynik, który zostanie zwrócony przez interfejs API.

type alias Model =
   { heading : String
   , factText : String
   , input :String
   }

Wiadomość

Aplikacja zawiera następujące trzy komunikaty -

  • ShowFacts
  • Input
  • NewFactArrived

Po kliknięciu Pokaż Fakty przycisku ShowFacts wiadomość jest przekazywana do metody aktualizacji. Gdy użytkownik wpisze jakąś wartość w polu tekstowym, komunikat wejściowy jest przekazywany do metody aktualizacji. Na koniec, po odebraniu odpowiedzi serwera Http, wiadomość NewFactArrived zostanie przekazana do aktualizacji.

type Msg
   = ShowFacts
   |Input String
   | NewFactArrived (Result Http.Error String)

Aktualizacja

Metoda update zwraca krotkę, która zawiera model i obiekty poleceń. Gdy użytkownik kliknie przycisk Pokaż fakty, komunikat jest przekazywany do aktualizacji, która następnie wywołuje interfejs NumbersAPI.

update : Msg -> Model -> (Model, Cmd Msg)
update msg model =
   case msg of
      Input newInput ->
      (Model "NumbersApi typing.." "" newInput ,Cmd.none)
      ShowFacts ->
         (model, getRadmonNumberFromAPI model.input)

      NewFactArrived (Ok newFact) ->
         (Model "DataArrived" newFact "", Cmd.none)

      NewFactArrived (Err _) ->
      (model, Cmd.none)

Funkcja pomocnika

Funkcja pomocnicza getRandomNumberFromAPI wywołuje NumbersAPI i przekazuje do niego liczbę wprowadzoną przez użytkownika. Wynik zwracany przez API jest używany do aktualizacji modelu.

getRadmonNumberFromAPI : String->Cmd Msg
getRadmonNumberFromAPI newNo =
   let
      url =
         "http://numbersapi.com/"++newNo
   in
      Http.send NewFactArrived (Http.getString url)
Sr. No. metoda Podpis Opis
1 Http.getString getString: String -> Ciąg żądania Utwórz żądanie GET i zinterpretuj treść odpowiedzi jako ciąg.
2 Http.send send: (Błąd wyniku a -> msg) -> Żądanie a -> Cmd msg Wyślij żądanie HTTP.

Główny

To jest punkt wyjścia do projektu Elm.

main =
   Html.program
      { init = init
      , view = view
      , update = update
      , subscriptions = subscriptions
      }

Kładąc wszystko razem

Step 1 - Utwórz folder CommandApp i plik CommandDemo.elm.

Step 2- Zainstaluj moduł http za pomocą polecenia pakiet elm install elm-lang / http .

Step 2 - Wpisz zawartość CommandDemo.elm, jak pokazano poniżej -

import Html exposing (..)
import Html.Attributes exposing (..)
import Html.Events exposing (..)
import Http

main =
   Html.program
      { init = init
      , view = view
      , update = update
      , subscriptions = subscriptions
      }

-- MODEL
type alias Model =
   { heading : String
   , factText : String
   , input :String
   }

init : (Model, Cmd Msg)
init =
   ( Model "NumbersAPI" "NoFacts" "42"-- set model two fields
   , Cmd.none -- not to invoke api initially
   )

-- UPDATE

type Msg
   = ShowFacts
   |Input String
   | NewFactArrived (Result Http.Error String)

update : Msg -> Model -> (Model, Cmd Msg)
update msg model =
   case msg of
      Input newInput ->
      (Model "NumbersApi typing.." "" newInput ,Cmd.none)
      ShowFacts ->
         (model, getRadmonNumberFromAPI model.input)

      NewFactArrived (Ok newFact) ->
         (Model "DataArrived" newFact "", Cmd.none)

      NewFactArrived (Err _) ->
         (model, Cmd.none)

- VIEW

view : Model -> Html Msg
view model =
   div []
      [ h2 [] [text model.heading]
      ,input [onInput Input, value model.input] []
      , button [ onClick ShowFacts ] [ text "show facts" ]
      , br [] []
      , h3 [][text model.factText]
      ]

-- SUBSCRIPTIONS

subscriptions : Model -> Sub Msg
subscriptions model =
   Sub.none

-- HTTP

getRadmonNumberFromAPI : String->Cmd Msg
getRadmonNumberFromAPI newNo =
   let
      url =
      "http://numbersapi.com/"++newNo
   in
      Http.send NewFactArrived (Http.getString url)

Step 4 - Wystrzel komendę.

C:\Users\dell\elm\CommandApp> elm make .\CommandDemo.elm

Spowoduje to wygenerowanie pliku html, jak pokazano poniżej.

Elm - Subskrypcje

W poprzednim rozdziale omówiliśmy, że widok współdziała z innymi składnikami za pomocą poleceń. Podobnie składnik (np. WebSocket) może komunikować się z widokiem przy użyciu subskrypcji. Subskrypcje to sposób, w jaki aplikacja Elm może odbierać zewnętrzne dane wejściowe, takie jak zdarzenia klawiatury, zdarzenia timera i zdarzenia WebSocket.

Poniższy rysunek wyjaśnia rolę subskrypcji w aplikacji Elm. Użytkownik współdziała z aplikacją Elm za pośrednictwem wiadomości. Podana aplikacja korzysta z WebSocket i ma dwa tryby działania -

  • Wyślij dane po stronie klienta do serwera gniazda za pomocą polecenia
  • Odbieraj dane w dowolnym momencie z serwera gniazda za pośrednictwem subskrypcji

Składnia

Składnię definiowania subskrypcji podano poniżej -

type Sub msg

Ilustracja

Zrozummy subskrypcje na prostym przykładzie.

W poniższym przykładzie aplikacja wysyła wiadomość do serwera. Serwer jest serwerem echa, który odpowiada klientowi tą samą wiadomością. Wszystkie przychodzące wiadomości są później wyświetlane na liście. Będziemy używać WebSocket (protokół wss), aby móc stale nasłuchiwać wiadomości z serwera. WebSocket wyśle ​​dane wejściowe użytkownika do serwera za pomocą poleceń, podczas gdy będzie używać subskrypcji do odbierania wiadomości z serwera.

Poniżej przedstawiono różne elementy aplikacji -

Serwer echa

Dostęp do serwera echo można uzyskać za pomocą protokołu wss. Serwer echa odsyła dane wejściowe użytkownika do aplikacji. Kod definiujący serwer echa jest podany poniżej -

echoServer : String
echoServer =
"wss://echo.websocket.org"

Model

Model reprezentuje dane wejściowe użytkownika i listę komunikatów przychodzących z serwera gniazda. Kod do definiowania Modelu jest podany poniżej -

type alias Model =
   { input : String
   , messages : List String
   }

Wiadomości

Typ wiadomości będzie zawierał dane wejściowe do pobierania tekstu od użytkownika. Wiadomość Wyślij zostanie wygenerowana, gdy użytkownik kliknie przycisk wysyłania wiadomości do serwera WebSocket. NewMessage jest używany, gdy wiadomość przychodzi z serwera echa.

type Msg
   = Input String
   | Send
   | NewMessage String

Widok

Widok aplikacji zawiera pole tekstowe i przycisk przesyłania służący do wysyłania danych wejściowych użytkownika na serwer. Odpowiedź z serwera jest wyświetlana w widoku przy użyciu znacznika DIV .

view : Model -> Html Msg
view model =
   div []
      [ input [onInput Input, value model.input] []
      , button [onClick Send] [text "Send"]
      , div [] (List.map viewMessage (List.reverse model.messages))
      ]

viewMessage : String -> Html msg
viewMessage msg =
   div [] [ text msg ]

Aktualizacja

Funkcja aktualizacji przejmuje wiadomość i komponenty modelu. Aktualizuje model na podstawie typu wiadomości.

update : Msg -> Model -> (Model, Cmd Msg)
update msg {input, messages} =
   case msg of
      Input newInput ->
         (Model newInput messages, Cmd.none)

   Send ->
      (Model "" messages, WebSocket.send echoServer input)

   NewMessage str ->
      (Model input (str :: messages), Cmd.none)
Sr. No. metoda Podpis Opis
1 WebSocket.listen słuchaj: String -> (String -> msg) -> Sub msg Subskrybuje wszystkie przychodzące wiadomości w sieci Web.
2 WebSocket.send send: String -> String -> Cmd msg Wysyła żądanie wss na adres serwera. Ważne jest, aby subskrybować również ten adres z odsłuchiwaniem. Jeśli nie, zostanie utworzone gniazdo sieciowe do wysyłania jednej wiadomości, a następnie zamknięte.

Subskrypcja

Funkcja subskrypcji przyjmuje obiekt modelu. Aby otrzymać wiadomości z serwera WebSocket, wywołujemy WebSocket.listen przekazując wiadomość jako NewMessage . Gdy nowa wiadomość pochodzi z serwera, wywoływana jest metoda aktualizacji.

subscriptions : Model -> Sub Msg
subscriptions model =
WebSocket.listen echoServer NewMessage

Główny

Główną funkcją jest punkt wejścia do aplikacji elm, jak pokazano poniżej.

main =
   Html.program
      { init = init
      , view = view
      , update = update
      , subscriptions = subscriptions
      }

Kładąc wszystko razem

Step 1 - Utwórz katalog, SubscriptionApp i dodaj do niego plik SubscriptionDemo.elm.

Step 2 - Dodaj następującą zawartość do pliku SubscriptionDemo.elm -

import Html exposing (..)
import Html.Attributes exposing (..)
import Html.Events exposing (..)
import WebSocket

main =
   Html.program
      { init = init
      , view = view
      , update = update
      , subscriptions = subscriptions
      }

echoServer : String
echoServer =
   "wss://echo.websocket.org"

-- MODEL

type alias Model =
   { input : String
   , messages : List String
   }

init : (Model, Cmd Msg)
init =
   (Model "" [], Cmd.none)

-- UPDATE
type Msg
   = Input String
   | Send
   | NewMessage String

update : Msg -> Model -> (Model, Cmd Msg)
update msg {input, messages} =
   case msg of
      Input newInput ->
      (Model newInput messages, Cmd.none)

   Send ->
      (Model "" messages, WebSocket.send echoServer input)

   NewMessage str ->
      (Model input (str :: messages), Cmd.none)

-- SUBSCRIPTIONS
subscriptions : Model -> Sub Msg
subscriptions model =
   WebSocket.listen echoServer NewMessage

-- VIEW
view : Model -> Html Msg
view model =
   div []
      [ input [onInput Input, value model.input] []
      , button [onClick Send] [text "Send"]
      , div [] (List.map viewMessage (List.reverse model.messages))
      ]

viewMessage : String -> Html msg
viewMessage msg =
div [] [ text msg ]

Step 3 - Zainstaluj pakiet websockets za pomocą menedżera pakietów elm.

C:\Users\dell\elm\SubscriptionApp> elm-package install elm-lang/websocket

Step 4 - Zbuduj i wygeneruj plik index.html, jak pokazano poniżej.

C:\Users\dell\elm\SubscriptionApp> elm make .\SubscriptionDemo.elm

Step 5 - Po wykonaniu wygenerowane zostanie następujące wyjście -