Haskell-퀵 가이드
Haskell은 기호 계산 및 목록 처리 응용 프로그램을 처리하도록 특별히 설계된 함수형 프로그래밍 언어입니다. 함수형 프로그래밍은 수학 함수를 기반으로합니다. Haskell 외에도 함수형 프로그래밍 패러다임을 따르는 다른 인기있는 언어로는 Lisp, Python, Erlang, Racket, F #, Clojure 등이 있습니다.
에 conventional programing, 명령어는 특정 구문 또는 형식의 선언 세트로 간주되지만 functional programing, 모든 계산은 별도의 수학 함수의 조합으로 간주됩니다.
Haskell과 함께 작동하기
Haskell은 널리 사용되는 순전히 기능적인 언어입니다. 여기에서는이 언어를 Java, C, C ++, PHP 등과 같은 다른 기존 프로그래밍 언어보다 특별하게 만드는 몇 가지 사항을 나열했습니다.
Functional Language− 기존 프로그래밍 언어에서 우리는 컴파일러에게 일련의 작업을 지시합니다. 이는 컴퓨터에 "무엇을해야할지"와 "어떻게해야할까요?"를 알려주는 것뿐입니다. 그러나 Haskell에서는 컴퓨터에 "그게 무엇인지"말할 것입니다.
Laziness− Haskell은 게으른 언어입니다. 으로lazy, 우리는 Haskell이 이유없이 어떤 표현도 평가하지 않을 것임을 의미합니다. 평가 엔진이 식을 평가해야 함을 발견하면thunk data structure 특정 평가에 필요한 모든 정보와 그에 대한 포인터를 수집합니다. thunk data structure. 평가 엔진은 특정 식을 평가해야하는 경우에만 작동을 시작합니다.
Modularity− Haskell 애플리케이션은 일련의 기능 일뿐입니다. Haskell 애플리케이션은 수많은 작은 Haskell 애플리케이션의 모음이라고 말할 수 있습니다.
Statically Typed− 기존 프로그래밍 언어에서는 유형과 함께 일련의 변수를 정의해야합니다. 반대로 Haskell은 엄격하게 입력 된 언어입니다. Strictly Typed language라는 용어는 Haskell 컴파일러가 선언 된 변수의 유형을 파악할 수있을만큼 지능적이므로 사용 된 변수의 유형을 명시 적으로 언급 할 필요가 없음을 의미합니다.
Maintainability − Haskell 애플리케이션은 모듈 식이므로 유지 관리가 매우 쉽고 비용 효율적입니다.
기능적 프로그램은 더 동시 적이며 더 정확하고 더 나은 성능을 제공하기 위해 실행시 병렬 처리를 따릅니다. Haskell도 예외는 아닙니다. 처리하는 방식으로 개발되었습니다multithreading 효과적으로.
Hello World
Haskell의 역 동성을 보여주는 간단한 예입니다. 다음 코드를 살펴보십시오. 콘솔에 "Hello Word"를 인쇄하려면 한 줄만 있으면됩니다.
main = putStrLn "Hello World"
Haskell 컴파일러가 위의 코드를 발견하면 즉시 다음과 같은 출력을 생성합니다.
Hello World
Haskell의 강력 함과 단순함을 보여주기 위해이 튜토리얼 전체에서 많은 예제를 제공합니다.
우리는 온라인으로 Haskell 프로그래밍 환경을 설정했습니다. https://www.tutorialspoint.com/compile_haskell_online.php
이 온라인 편집기에는 Haskell 프로그래밍 예제를 연습 할 수있는 많은 옵션이 있습니다. 페이지의 터미널 섹션으로 이동하여"ghci". 이 명령은 자동으로 Haskell 컴파일러를로드하고 Haskell을 온라인으로 시작합니다. 사용 후 다음 출력을 받게됩니다.ghci 명령.
sh-4.3$ ghci
GHCi,version7.8.4:http://www.haskell.org/ghc/:?forhelp
Loading package ghc-prim...linking...done.
Loading packageinteger gmp...linking... done.
Loading package base...linking...done.
Prelude>
여전히 로컬 시스템에서 오프라인으로 하스켈을 사용하고 싶다면 공식 웹 페이지에서 사용 가능한 하스켈 설정을 다운로드해야합니다. https://www.haskell.org/downloads
세 가지 유형이 있습니다. installers 시장에서 사용 가능-
Minimal Installer − GHC (The Glasgow Haskell Compiler), CABAL (Common Architecture for Building Applications and Libraries) 및 Stack 도구를 제공합니다.
Stack Installer−이 설치 프로그램에서 GHC는 관리되는 유료 체인의 크로스 플랫폼으로 다운로드 할 수 있습니다. 필요할 때마다 API 도구를 업데이트 할 수 있도록 애플리케이션을 전역 적으로 설치합니다. 모든 Haskell 지향 종속성을 자동으로 해결합니다.
Haskell Platform− 이것은 Haskell을 설치하는 가장 좋은 방법입니다. 전체 플랫폼을 컴퓨터에 설치하고 특정 위치에서 설치하기 때문입니다. 이 설치 프로그램은 위의 두 설치 프로그램처럼 배포되지 않습니다.
우리는 시중에 나와있는 다양한 유형의 설치 프로그램을 보았습니다. 이제 우리 컴퓨터에서 이러한 설치 프로그램을 사용하는 방법을 살펴 보겠습니다. 이 튜토리얼에서는 Haskell 플랫폼 설치 프로그램을 사용하여 시스템에 Haskell 컴파일러를 설치할 것입니다.
Windows에서 환경 설정
Windows 컴퓨터에서 Haskell 환경을 설정하려면 공식 웹 사이트로 이동하십시오. https://www.haskell.org/platform/windows.html 사용자 정의 가능한 아키텍처에 따라 설치 프로그램을 다운로드하십시오.
시스템의 아키텍처를 확인하고 해당 설정 파일을 다운로드하여 실행하십시오. 다른 Windows 응용 프로그램처럼 설치됩니다. 시스템의 CABAL 구성을 업데이트해야 할 수도 있습니다.
MAC에서 환경 설정
MAC 시스템에서 Haskell 환경을 설정하려면 공식 웹 사이트로 이동하십시오. https://www.haskell.org/platform/mac.html Mac 설치 프로그램을 다운로드하십시오.
Linux에서 환경 설정
Linux 기반 시스템에 Haskell을 설치하려면 MAC 및 Windows처럼 쉽지 않은 명령을 실행해야합니다. 네, 지루하지만 신뢰할 수 있습니다.
아래 단계에 따라 Linux 시스템에 Haskell을 설치할 수 있습니다.
Step 1 − Linux 시스템에서 Haskell 환경을 설정하려면 공식 웹 사이트로 이동하십시오. https://www.haskell.org/platform/linux.html배포를 선택하십시오. 브라우저에 다음 화면이 표시됩니다.
Step 2− 배포를 선택하십시오. 우리의 경우 우분투를 사용하고 있습니다. 이 옵션을 선택하면 로컬 시스템에 하스켈을 설치하라는 명령이있는 다음 페이지가 화면에 표시됩니다.
Step 3 − Ctrl + Alt + T를 눌러 터미널을 엽니 다. 명령을 실행합니다. "$ sudo apt-get install haskell-platform"Enter를 누르십시오. 루트 암호로 인증 한 후 시스템에 Haskell을 자동으로 다운로드하기 시작합니다. 설치 후 확인 메시지를 받게됩니다.
Step 4− 터미널로 다시 이동하여 GHCI 명령을 실행합니다. Prelude 프롬프트가 표시되면 로컬 시스템에서 Haskell을 사용할 준비가 된 것입니다.
GHCI 프롤로그를 종료하려면 ": quit exit"명령을 사용할 수 있습니다.
Haskell은 순전히 기능적인 프로그래밍 언어이므로 다른 프로그래밍 언어보다 훨씬 더 상호 작용적이고 지능적입니다. 이 장에서는 실제로 미리 정의되어 있거나 컴퓨터 메모리에 지능적으로 디코딩되는 Haskell의 기본 데이터 모델에 대해 알아 봅니다.
이 튜토리얼 전체에서 우리 웹 사이트에서 사용 가능한 Haskell 온라인 플랫폼 (https://www.tutorialspoint.com/codingground.htm).
번호
Haskell은 어떤 숫자를 숫자로 해독 할 수있을만큼 지능적입니다. 따라서 일반적으로 다른 프로그래밍 언어의 경우처럼 유형을 외부에서 언급 할 필요가 없습니다. 예를 들어 전주곡 명령 프롬프트로 이동하여 "2 + 2"를 실행하고 Enter 키를 누르십시오.
sh-4.3$ ghci
GHCi, version 7.6.3: http://www.haskell.org/ghc/ :? for help
Loading package ghc-prim ... linking ... done.
Loading package integer-gmp ... linking ... done.
Loading package base ... linking ... done.
Prelude> 2+2
결과적으로 다음과 같은 출력을 받게됩니다.
4
위의 코드에서 우리는 유형을 미리 정의하지 않고 GHCI 컴파일러에 인수로 두 개의 숫자를 전달했지만 컴파일러는이 두 항목을 숫자로 쉽게 디코딩 할 수 있습니다.
이제 좀 더 복잡한 수학적 계산을 시도하고 지능형 컴파일러가 올바른 출력을 제공하는지 확인합니다. "15+ (5 * 5) -40"으로 시도하십시오.
Prelude> 15+(5*5)-40
위의 표현식은 예상 출력에 따라 "0"을 산출합니다.
0
캐릭터
숫자와 마찬가지로 Haskell은 입력으로 주어진 문자를 지능적으로 식별 할 수 있습니다. Haskell 명령 프롬프트로 이동하여 큰 따옴표 또는 작은 따옴표로 문자를 입력하십시오.
다음 줄을 입력으로 제공하고 출력을 확인하겠습니다.
Prelude> :t "a"
다음 출력을 생성합니다-
"a" :: [Char]
(:t) 입력을 제공하는 동안. 위의 예에서(:t)입력과 관련된 특정 유형을 포함하는 것입니다. 이 유형에 대해서는 다음 장에서 자세히 알아볼 것입니다.
잘못된 입력을 문자로 전달하는 다음 예제를 살펴보면 오류가 발생합니다.
Prelude> :t a
<interactive>:1:1: Not in scope: 'a'
Prelude> a
<interactive>:4:1: Not in scope: 'a'
"<interactive> : 4 : 1 : Not in scope :`a '"라는 오류 메시지에 의해 Haskell 컴파일러는 사용자의 입력을 인식 할 수 없음을 경고합니다. Haskell은 모든 것이 숫자로 표현되는 언어 유형입니다.
Haskell은 일반적인 ASCII 인코딩 스타일을 따릅니다. 더 많은 것을 이해하기 위해 다음 예제를 살펴 보겠습니다.
Prelude> '\97'
'a'
Prelude> '\67'
'C'
입력이 ASCII 형식으로 어떻게 디코딩되는지 살펴보십시오.
끈
ㅏ string캐릭터 모음 일뿐입니다. 문자열 사용에 대한 특정 구문은 없지만 Haskell은 큰 따옴표로 문자열을 나타내는 일반적인 스타일을 따릅니다.
"Tutorialspoint.com"문자열을 전달하는 다음 예제를 살펴보십시오.
Prelude> :t "tutorialspoint.com"
화면에 다음과 같은 출력이 생성됩니다.
"tutorialspoint.com" :: [Char]
전체 문자열이 Char의 배열로만 디코딩 된 방법을 확인하십시오. 다른 데이터 유형과 구문으로 이동하겠습니다. 실제 연습을 시작하면 모든 데이터 유형과 사용에 익숙해집니다.
부울
부울 데이터 유형은 다른 데이터 유형과 마찬가지로 매우 간단합니다. "True"또는 "False"와 같은 일부 부울 입력을 사용하여 다른 부울 연산을 사용하는 다음 예제를보십시오.
Prelude> True && True
True
Prelude> True && False
False
Prelude> True || True
True
Prelude> True || False
True
위의 예에서 "True"및 "False"가 부울 값임을 언급 할 필요가 없습니다. Haskell 자체가이를 디코딩하고 각각의 작업을 수행 할 수 있습니다. "true"또는 "false"로 입력을 수정하겠습니다.
Prelude> true
다음 출력을 생성합니다-
<interactive>:9:1: Not in scope: 'true'
위의 예에서 Haskell은 "true"와 숫자 값을 구별 할 수 없으므로 입력 "true"는 숫자가 아닙니다. 따라서 Haskell 컴파일러는 입력이 범위가 아니라는 오류를 발생시킵니다.
목록 및 목록 이해
다른 데이터 유형과 마찬가지로 List또한 Haskell에서 사용되는 매우 유용한 데이터 유형입니다. 예를 들어 [a, b, c]는 문자 목록이므로 정의에 따라 List는 쉼표로 구분 된 동일한 데이터 유형의 모음입니다.
다른 데이터 유형과 마찬가지로 List를 List로 선언 할 필요가 없습니다. Haskell은 표현식에 사용 된 구문을보고 입력을 디코딩 할 수있을만큼 지능적입니다.
Haskell이 List를 처리하는 방법을 보여주는 다음 예제를 살펴보십시오.
Prelude> [1,2,3,4,5]
다음 출력을 생성합니다-
[1,2,3,4,5]
Haskell의 목록은 본질적으로 동종이므로 다른 종류의 데이터 유형 목록을 선언 할 수 없습니다. [1,2,3,4,5, a, b, c, d, e, f]와 같은 목록은 오류를 생성합니다.
Prelude> [1,2,3,4,5,a,b,c,d,e,f]
이 코드는 다음 오류를 생성합니다-
<interactive>:17:12: Not in scope: 'a'
<interactive>:17:14: Not in scope: 'b'
<interactive>:17:16: Not in scope: 'c'
<interactive>:17:18: Not in scope: 'd'
<interactive>:17:20: Not in scope: 'e'
<interactive>:17:22: Not in scope: 'f'
목록 이해
목록 이해는 수학적 표현을 사용하여 목록을 생성하는 프로세스입니다. 다음 예제에서 [출력 | 범위, 조건].
Prelude> [x*2| x<-[1..10]]
[2,4,6,8,10,12,14,16,18,20]
Prelude> [x*2| x<-[1..5]]
[2,4,6,8,10]
Prelude> [x| x<-[1..5]]
[1,2,3,4,5]
수학적 표현을 사용하여 하나의 List를 생성하는이 방법을 List Comprehension.
튜플
Haskell은 단일 데이터 유형에서 여러 값을 선언하는 또 다른 방법을 제공합니다. 그것은Tuple. 튜플은 목록으로 간주 될 수 있지만 튜플과 목록 간에는 몇 가지 기술적 차이가 있습니다.
튜플은 변경 불가능한 데이터 유형입니다. 런타임에 요소 수를 수정할 수없는 반면 List는 변경 가능한 데이터 유형입니다.
반면에 List는 동종 데이터 유형이지만 Tuple은 내부에 다른 유형의 데이터를 포함 할 수 있기 때문에 본질적으로 이기종입니다.
튜플은 단일 괄호로 표시됩니다. Haskell이 튜플을 처리하는 방법을 보려면 다음 예제를 살펴보십시오.
Prelude> (1,1,'a')
다음 출력을 생성합니다-
(1,1,'a')
위의 예에서 우리는 두 개의 튜플을 사용했습니다. number 유형 변수 및 char 유형 변수.
이 장에서는 Haskell에서 사용되는 여러 연산자에 대해 알아 봅니다. 다른 프로그래밍 언어와 마찬가지로 Haskell은 더하기, 빼기, 곱하기 등과 같은 몇 가지 기본 연산을 지능적으로 처리합니다. 다음 장에서는 다른 연산자와 그 사용법에 대해 자세히 알아볼 것입니다.
이 장에서는 온라인 플랫폼을 사용하여 Haskell에서 다른 연산자를 사용합니다 (https://www.tutorialspoint.com/codingground.htm). 우리는integer 숫자를 입력하세요. decimal 다음 장에 숫자를 입력하십시오.
덧셈 연산자
이름에서 알 수 있듯이 더하기 (+) 연산자는 더하기 기능에 사용됩니다. 다음 샘플 코드는 Haskell에서 두 개의 정수를 추가하는 방법을 보여줍니다.
main = do
let var1 = 2
let var2 = 3
putStrLn "The addition of the two numbers is:"
print(var1 + var2)
위 파일에서 두 개의 개별 변수를 만들었습니다. var1 과 var2. 마지막으로 우리는addition운영자. 사용compile 과 execute 버튼을 클릭하여 코드를 실행하세요.
이 코드는 화면에 다음 출력을 생성합니다.
The addition of the two numbers is:
5
빼기 연산자
이름에서 알 수 있듯이이 연산자는 빼기 연산에 사용됩니다. 다음 샘플 코드는 Haskell에서 두 개의 정수를 뺄 수있는 방법을 보여줍니다.
main = do
let var1 = 10
let var2 = 6
putStrLn "The Subtraction of the two numbers is:"
print(var1 - var2)
이 예에서는 두 개의 변수를 만들었습니다. var1 과 var2. 그런 다음 빼기 (-) 연산자를 사용하여 두 값을 뺍니다.
이 코드는 화면에 다음 출력을 생성합니다.
The Subtraction of the two numbers is:
4
곱셈 연산자
이 연산자는 곱셈 연산에 사용됩니다. 다음 코드는 곱셈 연산자를 사용하여 Haskell에서 두 숫자를 곱하는 방법을 보여줍니다.
main = do
let var1 = 2
let var2 = 3
putStrLn "The Multiplication of the Two Numbers is:"
print(var1 * var2)
이 코드는 온라인 플랫폼에서 실행할 때 다음과 같은 출력을 생성합니다.
The Multiplication of the Two Numbers is:
6
부문 연산자
다음 코드를 살펴보십시오. Haskell에서 두 수를 나누는 방법을 보여줍니다.
main = do
let var1 = 12
let var2 = 3
putStrLn "The Division of the Two Numbers is:"
print(var1/var2)
다음 출력을 생성합니다-
The Division of the Two Numbers is:
4.0
시퀀스 / 범위 연산자
Sequence 또는 Range는 Haskell의 특수 연산자입니다. "(..)"로 표시됩니다. 일련의 값으로 목록을 선언하는 동안이 연산자를 사용할 수 있습니다.
1에서 10까지의 모든 값을 인쇄하려면 "[1..10]"과 같은 것을 사용할 수 있습니다. 마찬가지로 "a"에서 "z"까지 모든 알파벳을 생성하려면 다음을 입력하면됩니다."[a..z]".
다음 코드는 시퀀스 연산자를 사용하여 1에서 10까지의 모든 값을 인쇄하는 방법을 보여줍니다.
main :: IO()
main = do
print [1..10]
다음과 같은 출력이 생성됩니다.
[1,2,3,4,5,6,7,8,9,10]
의사 결정은 프로그래머가 코드 흐름에 조건을 적용 할 수있는 기능입니다. 프로그래머는 미리 정의 된 조건에 따라 일련의 명령을 실행할 수 있습니다. 다음 순서도는 Haskell의 의사 결정 구조를 보여줍니다-
Haskell은 다음과 같은 유형의 의사 결정 진술을 제공합니다.
Sr. 아니. | 성명 및 설명 |
---|---|
1 | if-else 문 하나 if 진술서 else성명서. 의 지시else 블록은 주어진 부울 조건이 충족되지 않을 때만 실행됩니다. |
2 | 중첩 된 if-else 문 배수 if 블록 뒤에 else 블록 |
Haskell은 기능적 언어이며 엄격하게 형식화되어 있으므로 전체 응용 프로그램에서 사용되는 데이터 형식이 컴파일 타임에 컴파일러에 알려집니다.
내장형 클래스
Haskell에서 모든 문장은 수학적 표현으로 간주되며이 표현의 범주는 Type. "Type"은 컴파일 타임에 사용 된 표현식의 데이터 유형이라고 말할 수 있습니다.
에 대해 자세히 알아 보려면 Type, ": t"명령을 사용합니다. 일반적인 방법으로Type 값으로 간주 할 수 있지만 Type Class유사한 유형의 집합으로 간주 될 수 있습니다. 이 장에서는 다양한 내장 유형에 대해 알아 봅니다.
Int
IntInteger 유형 데이터를 나타내는 유형 클래스입니다. 2147483647에서 -2147483647 사이의 모든 정수는Int유형 클래스. 다음 예에서 함수fType() 정의 된 유형에 따라 작동합니다.
fType :: Int -> Int -> Int
fType x y = x*x + y*y
main = print (fType 2 4)
여기에서 함수의 유형을 설정했습니다. fType() 같이 int. 이 함수는 두 가지가 필요합니다.int 값과 하나를 반환 int값. 이 코드를 컴파일하고 실행하면 다음과 같은 출력이 생성됩니다.
sh-4.3$ ghc -O2 --make *.hs -o main -threaded -rtsopts
sh-4.3$ main
20
정수
Integer 다음의 상위 집합으로 간주 될 수 있습니다. Int. 이 값은 숫자에 의해 제한되지 않으므로 Integer는 제한없이 모든 길이가 될 수 있습니다. 기본 차이점을 보려면Int 과 Integer 위의 코드를 다음과 같이 수정하겠습니다.
fType :: Int -> Int -> Int
fType x y = x*x + y*y
main = print (fType 212124454 44545454454554545445454544545)
위의 코드를 컴파일하면 다음과 같은 오류 메시지가 나타납니다.
main.hs:3:31: Warning:
Literal 44545454454554545445454544545 is out of the Int range -
9223372036854775808..9223372036854775807
Linking main ...
이 오류는 함수 fType ()이 하나의 Int 유형 값을 예상하고 실제 큰 Int 유형 값을 전달하기 때문에 발생했습니다. 이 오류를 피하기 위해 "Integer"를 사용하여 "Int"유형을 수정하고 차이점을 살펴 보겠습니다.
fType :: Integer -> Integer -> Integer
fType x y = x*x + y*y
main = print (fType 212124454 4454545445455454545445445454544545)
이제 다음과 같은 출력이 생성됩니다.
sh-4.3$ main
1984297512562793395882644631364297686099210302577374055141
흙손
다음 코드를 살펴보십시오. 그것은 Haskell에서 Float 타입이 어떻게 작동하는지 보여줍니다-
fType :: Float -> Float -> Float
fType x y = x*x + y*y
main = print (fType 2.5 3.8)
이 함수는 두 개의 float 값을 입력으로 취하고 다른 float 값을 출력으로 생성합니다. 이 코드를 컴파일하고 실행하면 다음과 같은 출력이 생성됩니다.
sh-4.3$ main
20.689999
더블
Double끝에 배정 밀도가있는 부동 소수점 숫자입니다. 다음 예를 살펴보십시오-
fType :: Double -> Double -> Double
fType x y = x*x + y*y
main = print (fType 2.56 3.81)
위의 코드를 실행하면 다음과 같은 출력이 생성됩니다.
sh-4.3$ main
21.0697
부울
Bool부울 유형입니다. True 또는 False 일 수 있습니다. 다음 코드를 실행하여 Haskell에서 Bool 유형이 어떻게 작동하는지 이해하십시오.
main = do
let x = True
if x == False
then putStrLn "X matches with Bool Type"
else putStrLn "X is not a Bool Type"
여기서는 변수 "x"를 부울로 정의하고 다른 부울 값과 비교하여 독창성을 확인합니다. 다음 출력을 생성합니다-
sh-4.3$ main
X is not a Bool Type
숯
Char는 문자를 나타냅니다. 작은 따옴표 안의 모든 것은 문자로 간주됩니다. 다음 코드에서 이전fType() Char 값을 받아들이고 Char 값을 출력으로 반환하는 함수.
fType :: Char-> Char
fType x = 'K'
main = do
let x = 'v'
print (fType x)
위의 코드는 다음을 호출합니다. fType() 기능 char'v'의 값이지만 다른 char 값, 즉 'K'를 반환합니다. 출력은 다음과 같습니다.
sh-4.3$ main
'K'
Haskell은 선언되기 전에 유형을 잡을 수있을만큼 지능적이기 때문에 이러한 유형을 명시 적으로 사용하지 않을 것입니다. 이 튜토리얼의 다음 장에서는 다양한 유형과 유형 클래스가 어떻게 Haskell을 강력한 유형의 언어로 만드는지 살펴볼 것입니다.
EQ 유형 클래스
EQ유형 클래스는 표현식의 동등성을 테스트하는 기능을 제공하는 인터페이스입니다. 표현식의 동등성을 확인하려는 모든 유형 클래스는이 EQ 유형 클래스의 일부 여야합니다.
위에서 언급 한 모든 표준 유형 클래스는 이것의 일부입니다. EQ수업. 위에서 언급 한 유형 중 하나를 사용하여 평등을 검사 할 때마다 실제로EQ 유형 클래스.
다음 예에서 우리는 EQ "=="또는 "/ ="작업을 사용하여 내부적으로 입력합니다.
main = do
if 8 /= 8
then putStrLn "The values are Equal"
else putStrLn "The values are not Equal"
다음과 같은 출력이 생성됩니다.
sh-4.3$ main
The values are not Equal
Ord 유형 클래스
Ord주문 기능을 제공하는 또 다른 인터페이스 클래스입니다. 모든types 우리가 지금까지 사용한 것은 이것의 일부입니다 Ord상호 작용. EQ 인터페이스와 마찬가지로 Ord 인터페이스는 ">", "<", "<=", "> =", "compare"를 사용하여 호출 할 수 있습니다.
이 유형 클래스의 "비교"기능을 사용한 아래 예를 찾으십시오.
main = print (4 <= 2)
여기서 Haskell 컴파일러는 4가 2보다 작거나 같은지 확인합니다. 그렇지 않기 때문에 코드는 다음과 같은 출력을 생성합니다.
sh-4.3$ main
False
보여 주다
Show인수를 문자열로 인쇄하는 기능이 있습니다. 인수가 무엇이든 항상 결과를 문자열로 인쇄합니다. 다음 예에서는이 인터페이스를 사용하여 전체 목록을 인쇄합니다. "show"를 사용하여이 인터페이스를 호출 할 수 있습니다.
main = print (show [1..10])
콘솔에 다음 출력이 생성됩니다. 여기에서 큰 따옴표는 문자열 유형 값임을 나타냅니다.
sh-4.3$ main
"[1,2,3,4,5,6,7,8,9,10]"
읽다
Readinterface는 Show와 동일한 작업을 수행하지만 결과를 String 형식으로 인쇄하지 않습니다. 다음 코드에서 우리는read 인터페이스를 사용하여 문자열 값을 읽고 동일한 값을 Int 값으로 변환합니다.
main = print (readInt "12")
readInt :: String -> Int
readInt = read
여기에서 문자열 변수 ( "12")를 readInt변환 후 차례로 12 (Int 값)를 반환하는 메서드입니다. 출력은 다음과 같습니다.
sh-4.3$ main
12
열거 형
EnumHaskell에서 순차적 또는 정렬 된 기능을 가능하게하는 또 다른 유형의 Type 클래스입니다. 이 유형 클래스는 다음과 같은 명령으로 액세스 할 수 있습니다.Succ, Pred, Bool, Char등
다음 코드는 12의 후속 값을 찾는 방법을 보여줍니다.
main = print (succ 12)
다음 출력을 생성합니다-
sh-4.3$ main
13
경계
상한과 하한이있는 모든 유형은이 유형 클래스에 속합니다. 예를 들면Int 유형 데이터의 최대 경계는 "9223372036854775807"이고 최소 경계는 "-9223372036854775808"입니다.
다음 코드는 Haskell이 Int 유형의 최대 및 최소 경계를 결정하는 방법을 보여줍니다.
main = do
print (maxBound :: Int)
print (minBound :: Int)
다음 출력을 생성합니다-
sh-4.3$ main
9223372036854775807
-9223372036854775808
이제 Char, Float 및 Bool 유형의 최대 및 최소 경계를 찾으십시오.
Num
이 유형 클래스는 숫자 연산에 사용됩니다. Int, Integer, Float 및 Double과 같은 유형은이 유형 클래스에 속합니다. 다음 코드를 살펴보십시오-
main = do
print(2 :: Int)
print(2 :: Float)
다음 출력을 생성합니다-
sh-4.3$ main
2
2.0
완전한
IntegralNum Type Class의 하위 클래스로 간주 될 수 있습니다. Num Type 클래스는 모든 유형의 숫자를 보유하는 반면 Integral 유형 클래스는 정수에만 사용됩니다. Int 및 Integer는이 Type 클래스 아래의 유형입니다.
떠 있는
Integral과 마찬가지로 Floating도 Num Type 클래스의 일부이지만 부동 소수점 숫자 만 보유합니다. 그 후,Float 과 Double 이 유형 클래스에 속합니다.
커스텀 타입 클래스
다른 프로그래밍 언어와 마찬가지로 Haskell을 사용하면 개발자가 사용자 정의 유형을 정의 할 수 있습니다. 다음 예에서는 사용자 정의 유형을 만들어 사용합니다.
data Area = Circle Float Float Float
surface :: Area -> Float
surface (Circle _ _ r) = pi * r ^ 2
main = print (surface $ Circle 10 20 10 )
여기에서 우리는 Area. 다음으로이 유형을 사용하여 원의 면적을 계산합니다. 위의 예에서 "surface"는Area 입력으로 생성하고 Float 출력으로.
여기서 "데이터"는 키워드이며 Haskell의 모든 사용자 정의 유형은 항상 대문자로 시작합니다.
다음 출력을 생성합니다-
sh-4.3$ main
314.15927
함수는 함수형 프로그래밍 언어이기 때문에 Haskell에서 중요한 역할을합니다. 다른 언어와 마찬가지로 Haskell에는 자체 기능 정의 및 선언이 있습니다.
함수 선언은 출력과 함께 함수 이름과 인수 목록으로 구성됩니다.
함수 정의는 실제로 함수를 정의하는 곳입니다.
작은 예를 들어 보겠습니다. add 이 개념을 자세히 이해하는 기능.
add :: Integer -> Integer -> Integer --function declaration
add x y = x + y --function definition
main = do
putStrLn "The addition of the two numbers is:"
print(add 2 5) --calling a function
여기서 우리는 첫 번째 줄과 두 번째 줄에서 우리의 함수를 선언했고, 우리는 두 개의 인자를 받아 하나의 정수형 출력을 생성하는 실제 함수를 작성했습니다.
대부분의 다른 언어와 마찬가지로 Haskell은 main방법. 코드는 다음과 같은 출력을 생성합니다.
The addition of the two numbers is:
7
패턴 매칭
패턴 일치는 특정 유형의 표현식을 일치시키는 프로세스입니다. 코드를 단순화하는 기술 일뿐입니다. 이 기술은 모든 유형의 Type 클래스로 구현할 수 있습니다. If-Else는 패턴 일치의 대체 옵션으로 사용할 수 있습니다.
패턴 일치는 런타임시 인수 목록에 따라 다른 메서드를 실행할 수있는 동적 다형성의 변형으로 간주 할 수 있습니다.
다음 코드 블록을 살펴보십시오. 여기서 우리는 숫자의 계승을 계산하기 위해 패턴 매칭 기술을 사용했습니다.
fact :: Int -> Int
fact 0 = 1
fact n = n * fact ( n - 1 )
main = do
putStrLn "The factorial of 5 is:"
print (fact 5)
우리는 모두 숫자의 계승을 계산하는 방법을 알고 있습니다. 컴파일러는 인수를 사용하여 "fact"라는 함수를 검색하기 시작합니다. 인수가 0이 아니면 숫자는 실제 인수보다 1이 적은 동일한 함수를 계속 호출합니다.
인수의 패턴이 0과 정확히 일치하면 "fact 0 = 1"인 패턴을 호출합니다. 우리 코드는 다음과 같은 출력을 생성합니다.
The factorial of 5 is:
120
근위 연대
Guards패턴 매칭과 매우 유사한 개념입니다. 패턴 일치에서는 일반적으로 하나 이상의 표현식과 일치하지만guards 표현식의 일부 속성을 테스트합니다.
패턴 매칭을 사용하는 것이 좋지만 guards이지만 개발자의 관점에서 보면 guards더 읽기 쉽고 간단합니다. 처음 사용하는 경우guards If-Else 문과 매우 비슷해 보이지만 기능적으로는 다릅니다.
다음 코드에서 우리는 factorial 개념을 사용하여 프로그램 guards.
fact :: Integer -> Integer
fact n | n == 0 = 1
| n /= 0 = n * fact (n-1)
main = do
putStrLn "The factorial of 5 is:"
print (fact 5)
여기에서 우리는 guards, "|"로 구분 그리고 전화fact 기능 main. 내부적으로 컴파일러는 패턴 일치의 경우와 동일한 방식으로 작동하여 다음 출력을 생성합니다.
The factorial of 5 is:
120
Where 절
Where원하는 출력을 생성하기 위해 런타임에 사용할 수있는 키워드 또는 내장 함수입니다. 함수 계산이 복잡해질 때 매우 유용 할 수 있습니다.
입력이 여러 매개 변수가있는 복잡한 표현식 인 시나리오를 고려하십시오. 이러한 경우 "where"절을 사용하여 전체 표현식을 작은 부분으로 나눌 수 있습니다.
다음 예에서는 복잡한 수학적 표현을 사용합니다. Haskell을 사용하여 다항식 [x ^ 2-8x + 6]의 근을 찾는 방법을 보여 드리겠습니다.
roots :: (Float, Float, Float) -> (Float, Float)
roots (a,b,c) = (x1, x2) where
x1 = e + sqrt d / (2 * a)
x2 = e - sqrt d / (2 * a)
d = b * b - 4 * a * c
e = - b / (2 * a)
main = do
putStrLn "The roots of our Polynomial equation are:"
print (roots(1,-8,6))
주어진 다항식 함수의 근을 계산하기위한 표현식의 복잡성에 주목하십시오. 꽤 복잡합니다. 따라서 우리는where절. 위의 코드는 다음과 같은 출력을 생성합니다.
The roots of our Polynomial equation are:
(7.1622777,0.8377223)
재귀 함수
재귀는 함수가 자신을 반복적으로 호출하는 상황입니다. Haskell은 한 번 이상 표현식을 반복하는 기능을 제공하지 않습니다. 대신 Haskell은 전체 기능을 다른 기능 모음으로 나누고 재귀 기술을 사용하여 기능을 구현하기를 원합니다.
숫자의 계승을 계산 한 패턴 일치 예제를 다시 고려해 보겠습니다. 숫자의 계승을 찾는 것은 재귀를 사용하는 고전적인 경우입니다. 여기에서 "패턴 매칭은 재귀와 어떻게 다른가요?" 이 두 가지의 차이점은 사용 방식에 있습니다. 패턴 일치는 터미널 제한 설정에서 작동하는 반면 재귀는 함수 호출입니다.
다음 예제에서는 5의 계승을 계산하기 위해 패턴 일치와 재귀를 모두 사용했습니다.
fact :: Int -> Int
fact 0 = 1
fact n = n * fact ( n - 1 )
main = do
putStrLn "The factorial of 5 is:"
print (fact 5)
다음 출력을 생성합니다-
The factorial of 5 is:
120
고차 함수
지금까지 우리가 본 것은 Haskell 함수가 type 입력으로 또 다른 생산 type다른 명령형 언어와 매우 유사합니다. 고차 함수는 함수를 입력 또는 출력 인수로 사용할 수있는 Haskell의 고유 한 기능입니다.
가상 개념이지만 실제 프로그램에서는 Haskell에서 정의하는 모든 함수가 고차 메커니즘을 사용하여 출력을 제공합니다. Haskell의 라이브러리 함수를 살펴볼 기회가 있다면 대부분의 라이브러리 함수가 고차 방식으로 작성되었음을 알 수 있습니다.
내장 된 고차 함수 맵을 가져 와서 우리의 선택에 따라 다른 고차 함수를 구현하기 위해 동일한 것을 사용하는 예를 들어 보겠습니다.
import Data.Char
import Prelude hiding (map)
map :: (a -> b) -> [a] -> [b]
map _ [] = []
map func (x : abc) = func x : map func abc
main = print $ map toUpper "tutorialspoint.com"
위의 예에서 우리는 toUpper 유형 클래스의 기능 Char입력을 대문자로 변환합니다. 여기서 "map"메소드는 함수를 인수로 취하고 필요한 출력을 반환합니다. 출력은 다음과 같습니다.
sh-4.3$ ghc -O2 --make *.hs -o main -threaded -rtsopts sh-4.3$ main
"TUTORIALSPOINT.COM"
람다 식
응용 프로그램의 전체 수명 동안 한 번만 사용할 함수를 작성해야하는 경우가 있습니다. 이러한 상황을 처리하기 위해 Haskell 개발자는 다른 익명 블록을 사용합니다.lambda expression 또는 lambda function.
정의가없는 함수를 람다 함수라고합니다. 람다 함수는 "\"문자로 표시됩니다. 함수를 생성하지 않고 입력 값을 1 씩 증가시키는 다음 예제를 살펴 보겠습니다.
main = do
putStrLn "The successor of 4 is:"
print ((\x -> x + 1) 4)
여기에서 이름이없는 익명 함수를 만들었습니다. 정수 4를 인수로 취하고 출력 값을 인쇄합니다. 우리는 기본적으로 제대로 선언하지 않고 하나의 기능을 운영하고 있습니다. 그것이 람다 표현의 아름다움입니다.
람다 표현식은 다음과 같은 출력을 생성합니다.
sh-4.3$ main
The successor of 4 is:
5
지금까지 우리는 여러 유형의 Haskell 함수에 대해 논의하고 이러한 함수를 호출하는 데 다양한 방법을 사용했습니다. 이 장에서는 특별한 Type 클래스를 가져 오지 않고도 Haskell에서 쉽게 사용할 수있는 몇 가지 기본 함수에 대해 배웁니다. 이러한 함수의 대부분은 다른 고차 함수의 일부입니다.
머리 기능
Head기능은 목록에서 작동합니다. 기본적으로 목록 인 입력 인수의 첫 번째를 반환합니다. 다음 예에서는 10 개의 값이있는 목록을 전달하고 다음을 사용하여 해당 목록의 첫 번째 요소를 생성합니다.head 함수.
main = do
let x = [1..10]
putStrLn "Our list is:"
print (x)
putStrLn "The first element of the list is:"
print (head x)
다음 출력을 생성합니다-
Our list is:
[1,2,3,4,5,6,7,8,9,10]
The first element of the list is:
1
꼬리 기능
Tail 보완하는 기능입니다 head함수. 그것은 걸립니다list입력으로 머리 부분이없는 전체 목록을 산출합니다. 즉,tail함수는 첫 번째 요소없이 전체 목록을 반환합니다. 다음 예를 살펴보십시오-
main = do
let x = [1..10]
putStrLn "Our list is:"
print (x)
putStrLn "The tail of our list is:"
print (tail x)
다음 출력을 생성합니다-
Our list is:
[1,2,3,4,5,6,7,8,9,10]
The tail of our list is:
[2,3,4,5,6,7,8,9,10]
마지막 기능
이름에서 알 수 있듯이 입력으로 제공되는 목록의 마지막 요소를 생성합니다. 다음 예를 확인하십시오.
main = do
let x = [1..10]
putStrLn "Our list is:"
print (x)
putStrLn "The last element of our list is:"
print (last x)
다음 출력을 생성합니다-
Our list is:
[1,2,3,4,5,6,7,8,9,10]
The last element of our list is:
10
초기화 기능
Init 의 반대와 똑같이 작동합니다. tail함수. 목록을 인수로 취하고 마지막 항목없이 전체 목록을 반환합니다.
main = do
let x = [1..10]
putStrLn "Our list is:"
print (x)
putStrLn "Our list without the last entry:"
print (init x)
이제 출력을 관찰하십시오.
Our list is:
[1,2,3,4,5,6,7,8,9,10]
Our list without the last entry:
[1,2,3,4,5,6,7,8,9]
Null 함수
Null 문자열에서 작동하고 반환하는 부울 검사 함수입니다. True 주어진 목록이 비어있을 때만, 그렇지 않으면 False. 다음 코드는 제공된 목록이 비어 있는지 여부를 확인합니다.
main = do
let x = [1..10]
putStrLn "Our list is:"
print (x)
putStrLn "Is our list empty?"
print (null x)
다음 출력을 생성합니다-
Our list is:
[1,2,3,4,5,6,7,8,9,10]
Is our list empty?
False
역기능
문자열 입력에서 작동하고 전체 입력을 역순으로 변환하고 결과로 하나의 출력을 제공합니다. 다음은이 함수의 코드베이스입니다.
main = do
let x = [1..10]
putStrLn "Our list is:"
print (x)
putStrLn "The list in Reverse Order is:"
print (reverse x)
다음 출력을 생성합니다-
Our list is:
[1,2,3,4,5,6,7,8,9,10]
The list in Reverse Order is:
[10,9,8,7,6,5,4,3,2,1]
길이 기능
이 함수는 길이를 계산하는 데 사용됩니다. list인수로 주어집니다. 다음 예를 살펴보십시오-
main = do
let x = [1..10]
putStrLn "Our list is:"
print (x)
putStrLn "The length of this list is:"
print (length x)
목록에 10 개의 요소가 있으므로 코드는 10을 출력으로 산출합니다.
Our list is:
[1,2,3,4,5,6,7,8,9,10]
The length of this list is:
10
기능 가져 오기
Take함수는 다른 문자열에서 하위 문자열을 만드는 데 사용됩니다. 다음 코드는 Haskell에서 take 함수를 사용하는 방법을 보여줍니다.
main = print(take 5 ([1 .. 10]))
코드는 제공된 목록에서 5 개의 요소를 포함하는 하위 문자열을 생성합니다.
[1,2,3,4,5]
드롭 기능
이 함수는 하위 문자열을 생성하는데도 사용됩니다. 그것은 반대의 역할을합니다take함수. 다음 코드를보십시오-
main = print(drop 5 ([1 .. 10]))
코드는 제공된 목록에서 처음 5 개 요소를 삭제하고 나머지 5 개 요소를 인쇄합니다. 다음 출력을 생성합니다-
[6,7,8,9,10]
최대 기능
이 함수는 제공된 목록에서 최대 값을 가진 요소를 찾는 데 사용됩니다. 실제로 사용하는 방법을 살펴 보겠습니다.
main = do
let x = [1,45,565,1245,02,2]
putStrLn "The maximum value element of the list is:"
print (maximum x)
위의 코드는 다음과 같은 출력을 생성합니다.
The maximum value element of the list is:
1245
최소 기능
이 함수는 제공된 목록에서 최소값을 가진 요소를 찾는 데 사용됩니다. 그것은 단지 반대입니다maximum 함수.
main = do
let x = [1,45,565,1245,02,2]
putStrLn "The minimum value element of the list is:"
print (minimum x)
위 코드의 출력은-
The minimum value element of the list is:
1
합계 함수
이름에서 알 수 있듯이이 함수는 제공된 목록에있는 모든 요소의 합계를 반환합니다. 다음 코드는 5 개의 요소 목록을 가져 와서 그 합계를 출력으로 반환합니다.
main = do
let x = [1..5]
putStrLn "Our list is:"
print (x)
putStrLn "The summation of the list elements is:"
print (sum x)
다음 출력을 생성합니다-
Our list is:
[1,2,3,4,5]
The summation of the list elements is:
15
제품 기능
이 함수를 사용하여 목록의 모든 요소를 곱하고 그 값을 인쇄 할 수 있습니다.
main = do
let x = [1..5]
putStrLn "Our list is:"
print (x)
putStrLn "The multiplication of the list elements is:"
print (product x)
우리 코드는 다음과 같은 출력을 생성합니다.
Our list is:
[1,2,3,4,5]
The multiplication of the list elements is:
120
Elem 기능
이 함수는 제공된 목록에 특정 요소가 포함되어 있는지 여부를 확인하는 데 사용됩니다. 따라서true 또는 false.
다음 코드는 제공된 요소 목록에 786 값이 포함되어 있는지 확인합니다.
main = do
let x = [1,45,155,1785]
putStrLn "Our list is:"
print (x)
putStrLn "Does it contain 786?"
print (elem 786 (x))
다음 출력을 생성합니다-
Our list is:
[1,45,155,1785]
Does it contain 786?
False
동일한 코드를 사용하여 제공된 목록에 1785 값이 포함되어 있는지 확인하십시오.
Function Composition한 함수의 출력을 다른 함수의 입력으로 사용하는 프로세스입니다. 뒤에있는 수학을 배우면 더 좋을 것입니다composition. 수학에서composition 로 표시됩니다 f{g(x)} 어디 g() 다른 함수의 입력으로 사용되는 함수 및 출력입니다. 즉, f().
한 함수의 출력 유형이 두 번째 함수의 입력 유형과 일치하는 경우 함수 구성은 두 함수를 사용하여 구현할 수 있습니다. 점 연산자 (.)를 사용하여 Haskell에서 함수 구성을 구현합니다.
다음 예제 코드를 살펴보십시오. 여기서는 입력 숫자가 짝수인지 홀수인지 계산하기 위해 함수 구성을 사용했습니다.
eveno :: Int -> Bool
noto :: Bool -> String
eveno x = if x `rem` 2 == 0
then True
else False
noto x = if x == True
then "This is an even Number"
else "This is an ODD number"
main = do
putStrLn "Example of Haskell Function composition"
print ((noto.eveno)(16))
여기에서 main 함수, 우리는 두 개의 함수를 호출합니다. noto 과 eveno, 동시에. 컴파일러는 먼저 함수를 호출합니다."eveno()" 와 16인수로. 그 후 컴파일러는eveno 방법의 입력으로 noto() 방법.
출력은 다음과 같습니다.
Example of Haskell Function composition
"This is an even Number"
16을 입력 (짝수)으로 제공하기 때문에 eveno() 함수 반환 true에 대한 입력이됩니다. noto() 함수를 사용하고 출력을 반환합니다. "이것은 짝수입니다".
Java에서 작업했다면 모든 클래스가 폴더에 바인딩되는 방법을 알 수 있습니다. package. 마찬가지로 Haskell은modules.
Haskell은 기능적 언어이며 모든 것이 표현식으로 표시되므로 모듈은 유사하거나 관련된 유형의 함수 모음으로 호출 될 수 있습니다.
당신은 할 수 있습니다 import한 모듈에서 다른 모듈로 함수. 다른 함수를 정의하기 전에 모든 "import"문이 먼저 와야합니다. 이 장에서 우리는 Haskell 모듈의 다양한 기능을 배울 것입니다.
목록 모듈
List 함께 작동하는 몇 가지 멋진 기능을 제공합니다. list유형 데이터. List 모듈을 가져 오면 원하는대로 다양한 기능을 사용할 수 있습니다.
다음 예에서는 목록 모듈에서 사용할 수있는 몇 가지 중요한 기능을 사용했습니다.
import Data.List
main = do
putStrLn("Different methods of List Module")
print(intersperse '.' "Tutorialspoint.com")
print(intercalate " " ["Lets","Start","with","Haskell"])
print(splitAt 7 "HaskellTutorial")
print (sort [8,5,3,2,1,6,4,2])
여기에는 정의하지 않고도 많은 기능이 있습니다. 이러한 기능은 목록 모듈에서 사용할 수 있기 때문입니다. List 모듈을 가져온 후 Haskell 컴파일러는 이러한 모든 함수를 전역 네임 스페이스에서 사용할 수 있도록했습니다. 따라서 이러한 기능을 사용할 수 있습니다.
우리 코드는 다음과 같은 출력을 생성합니다.
Different methods of List Module
"T.u.t.o.r.i.a.l.s.p.o.i.n.t...c.o.m"
"Lets Start with Haskell"
("Haskell","Tutorial")
[1,2,2,3,4,5,6,8]
Char 모듈
그만큼 Char모듈에는 문자 유형과 함께 작동하는 사전 정의 된 기능이 많이 있습니다. 다음 코드 블록을 살펴보십시오-
import Data.Char
main = do
putStrLn("Different methods of Char Module")
print(toUpper 'a')
print(words "Let us study tonight")
print(toLower 'A')
여기에서 기능 toUpper 과 toLower 이미 내부에 정의되어 있습니다. Char기준 치수. 다음 출력을 생성합니다-
Different methods of Char Module
'A'
["Let","us","study","tonight"]
'a'
지도 모듈
Map정렬되지 않은 부가 가치 쌍 유형 데이터 유형입니다. 유용한 기능이 많은 널리 사용되는 모듈입니다. 다음 예제는 Map 모듈에서 사용 가능한 미리 정의 된 함수를 사용하는 방법을 보여줍니다.
import Data.Map (Map)
import qualified Data.Map as Map --required for GHCI
myMap :: Integer -> Map Integer [Integer]
myMap n = Map.fromList (map makePair [1..n])
where makePair x = (x, [x])
main = print(myMap 3)
다음 출력을 생성합니다-
fromList [(1,[1]),(2,[2]),(3,[3])]
모듈 설정
Set 모듈에는 수학적 데이터를 조작하는 데 매우 유용한 미리 정의 된 함수가 있습니다. 집합은 이진 트리로 구현되므로 집합의 모든 요소는 고유해야합니다.
다음 예제 코드를 살펴보십시오.
import qualified Data.Set as Set
text1 = "Hey buddy"
text2 = "This tutorial is for Haskell"
main = do
let set1 = Set.fromList text1
set2 = Set.fromList text2
print(set1)
print(set2)
여기서는 문자열을 집합으로 수정합니다. 다음 출력이 생성됩니다. 출력 세트에 문자가 반복되지 않는지 확인하십시오.
fromList " Hbdeuy"
fromList " HTaefhiklorstu"
맞춤형 모듈
다른 프로그램에서 호출 할 수있는 사용자 정의 모듈을 만드는 방법을 살펴 보겠습니다. 이 사용자 정의 모듈을 구현하기 위해 우리는"custom.hs" 우리와 함께 "main.hs".
사용자 지정 모듈을 만들고 그 안에 몇 가지 기능을 정의하겠습니다.
custom.hs
module Custom (
showEven,
showBoolean
) where
showEven:: Int-> Bool
showEven x = do
if x 'rem' 2 == 0
then True
else False
showBoolean :: Bool->Int
showBoolean c = do
if c == True
then 1
else 0
Custom 모듈이 준비되었습니다. 이제 프로그램으로 가져 오겠습니다.
main.hs
import Custom
main = do
print(showEven 4)
print(showBoolean True)
코드는 다음과 같은 출력을 생성합니다.
True
1
그만큼 showEven 함수 반환 True, "4"는 짝수이므로 그만큼showBoolean 함수에 전달 된 부울 함수가 "True"이므로 함수는 "1"을 반환합니다.
지금까지 논의한 모든 예는 본질적으로 정적입니다. 이 장에서는 사용자와 동적으로 통신하는 방법을 배웁니다. Haskell에서 사용되는 다양한 입력 및 출력 기술을 배웁니다.
파일 및 스트림
지금까지 프로그램 자체의 모든 입력을 하드 코딩했습니다. 우리는 정적 변수에서 입력을 받고 있습니다. 이제 외부 파일에서 읽고 쓰는 방법을 알아 보겠습니다.
파일을 만들고 이름을 "abc.txt"로 지정하겠습니다. 다음으로이 텍스트 파일에 "Tutorialspoint에 오신 것을 환영합니다. 여기에서 Haskell을 배우기위한 최고의 리소스를 얻을 수 있습니다."라는 줄을 입력합니다.
다음으로이 파일의 내용을 콘솔에 표시하는 다음 코드를 작성합니다. 여기서는 EOF 문자를 찾을 때까지 파일을 읽는 readFile () 함수를 사용합니다.
main = do
let file = "abc.txt"
contents <- readFile file
putStrLn contents
위의 코드는 파일 끝 문자를 만날 때까지 "abc.txt"파일을 문자열로 읽습니다. 이 코드는 다음 출력을 생성합니다.
Welcome to Tutorialspoint
Here, you will get the best resource to learn Haskell.
터미널에서 인쇄되는 모든 내용이 해당 파일에 기록되어 있는지 확인하십시오.
명령 줄 인수
Haskell은 명령 프롬프트를 통해 파일을 작동하는 기능도 제공합니다. 터미널로 돌아가서"ghci". 그런 다음 다음 명령 세트를 입력하십시오-
let file = "abc.txt"
writeFile file "I am just experimenting here."
readFile file
여기에서 "abc.txt"라는 텍스트 파일을 만들었습니다. 다음으로 다음 명령을 사용하여 파일에 명령문을 삽입했습니다.writeFile. 마지막으로 다음 명령을 사용했습니다.readFile콘솔에 파일의 내용을 인쇄합니다. 우리 코드는 다음과 같은 출력을 생성합니다.
I am just experimenting here.
예외
안 exception코드에서 버그로 간주 될 수 있습니다. 컴파일러가 런타임에 예상 된 출력을 얻지 못하는 상황입니다. 다른 좋은 프로그래밍 언어와 마찬가지로 Haskell은 예외 처리를 구현하는 방법을 제공합니다.
Java에 익숙하다면 일반적으로 오류가 발생하고 오류가 발생하는 Try-Catch 블록을 알고있을 것입니다. catch블록. Haskell에는 런타임 오류를 포착하는 동일한 기능이 있습니다.
기능 정의 try"try :: Exception e => IO a-> IO (Either ea)"처럼 보입니다. 다음 예제 코드를 살펴보십시오. "Divide by Zero"예외를 포착하는 방법을 보여줍니다.
import Control.Exception
main = do
result <- try (evaluate (5 `div` 0)) :: IO (Either SomeException Int)
case result of
Left ex -> putStrLn $ "Caught exception: " ++ show ex
Right val -> putStrLn $ "The answer was: " ++ show val
위의 예에서는 내장 된 try 의 기능 Control.Exception따라서 우리는 미리 예외를 포착하고 있습니다. 위의 코드 조각은 화면에서 아래 출력을 산출합니다.
Caught exception: divide by zero
FunctorHaskell에서 매핑 할 수있는 다양한 유형의 기능적 표현입니다. 다형성을 구현하는 높은 수준의 개념입니다. Haskell 개발자에 따르면 List, Map, Tree 등과 같은 모든 유형은 Haskell Functor의 인스턴스입니다.
ㅏ Functor 다음과 같은 함수 정의가있는 내장 클래스입니다.
class Functor f where
fmap :: (a -> b) -> f a -> f b
이 정의에 따라 우리는 Functor 함수를받는 함수입니다. fmap()다른 함수를 반환합니다. 위의 예에서fmap() 함수의 일반화 된 표현입니다. map().
다음 예제에서는 Haskell Functor가 어떻게 작동하는지 볼 것입니다.
main = do
print(map (subtract 1) [2,4,8,16])
print(fmap (subtract 1) [2,4,8,16])
여기에서 우리는 map() 과 fmap()빼기 연산을위한 목록 위에. 두 명령문이 [1,3,7,15] 요소를 포함하는 목록의 동일한 결과를 생성한다는 것을 알 수 있습니다.
두 함수 모두 다른 함수를 호출했습니다. subtract() 결과를 산출합니다.
[1,3,7,15]
[1,3,7,15]
그렇다면 차이점은 무엇입니까? map 과 fmap? 차이점은 사용법에 있습니다. Functor "just"및 "Nothing"과 같은 다양한 데이터 유형으로 더 많은 기능을 구현할 수 있습니다.
main = do
print (fmap (+7)(Just 10))
print (fmap (+7) Nothing)
위의 코드는 터미널에 다음 출력을 생성합니다.
Just 17
Nothing
실용적인 Functor
Applicative Functor는 Applicative Type 클래스에서 제공하는 몇 가지 추가 기능이있는 일반 Functor입니다.
Functor를 사용하여 일반적으로 기존 함수를 내부에 정의 된 다른 함수와 매핑합니다. 그러나 Functor 내부에 정의 된 함수를 다른 Functor와 매핑하는 방법은 없습니다. 그래서 우리는Applicative Functor. 이 매핑 기능은 아래에 정의 된 Applicative Type 클래스에 의해 구현됩니다.Control기준 치수. 이 클래스는 작업 할 수있는 두 가지 방법 만 제공합니다.pure 그리고 다른 하나는 <*>.
다음은 Applicative Functor의 클래스 정의입니다.
class (Functor f) => Applicative f where
pure :: a -> f a
(<*>) :: f (a -> b) -> f a -> f b
구현에 따라 두 가지 방법을 사용하여 다른 Functor를 매핑 할 수 있습니다. "Pure" 과 "<*>". "Pure"메서드는 모든 유형의 값을 가져야하며 항상 해당 값의 Applicative Functor를 반환합니다.
다음 예제는 Applicative Functor의 작동 방식을 보여줍니다.
import Control.Applicative
f1:: Int -> Int -> Int
f1 x y = 2*x+y
main = do
print(show $ f1 <$> (Just 1) <*> (Just 2) )
여기서 우리는 함수의 함수 호출에 응용 함수를 구현했습니다. f1. 우리 프로그램은 다음과 같은 결과를 산출합니다.
"Just 4"
모노 이드
우리 모두는 Haskell이 모든 것을 함수의 형태로 정의한다는 것을 알고 있습니다. 함수에는 입력을 함수의 출력으로 얻는 옵션이 있습니다. 이것은 무엇입니까Monoid 이다.
ㅏ Monoid출력이 입력과 독립적 인 함수 및 연산자의 집합입니다. 함수 (*)와 정수 (1)를 봅시다. 이제 입력이 무엇이든 출력은 동일한 숫자로만 유지됩니다. 즉, 숫자에 1을 곱하면 같은 숫자를 얻게됩니다.
다음은 monoid의 유형 클래스 정의입니다.
class Monoid m where
mempty :: m
mappend :: m -> m -> m
mconcat :: [m] -> m
mconcat = foldr mappend mempty
Haskell에서 Monoid의 사용을 이해하려면 다음 예제를 살펴보십시오.
multi:: Int->Int
multi x = x * 1
add :: Int->Int
add x = x + 0
main = do
print(multi 9)
print (add 7)
우리 코드는 다음과 같은 출력을 생성합니다.
9
7
여기서 함수 "multi"는 입력에 "1"을 곱합니다. 마찬가지로 "add"함수는 "0"으로 입력을 추가합니다. 두 경우 모두 출력은 입력과 동일합니다. 따라서 기능{(*),1} 과 {(+),0} 모노 이드의 완벽한 예입니다.
Monads몇 가지 추가 기능이있는 Applicative Functor 유형일뿐입니다. 다음과 같은 세 가지 기본 규칙을 관리하는 유형 클래스입니다.monadic rules.
세 가지 규칙은 모두 다음과 같은 Monad 선언에 엄격하게 적용됩니다.
class Monad m where
return :: a -> m a
(>>=) :: m a -> (a -> m b) -> m b
(>>) :: m a -> m b -> m b
x >> y = x >>= \_ -> y
fail :: String -> m a
fail msg = error msg
Monad 선언에 적용되는 세 가지 기본 법률은 다음과 같습니다.
Left Identity Law − return함수는 값을 변경하지 않으며 Monad에서 아무것도 변경하지 않아야합니다. "return> => mf = mf"로 표현할 수 있습니다.
Right Identity Law − return함수는 값을 변경하지 않으며 Monad에서 아무것도 변경하지 않아야합니다. "mf> => return = mf"로 표현할 수 있습니다.
Associativity−이 법칙에 따라 Functor와 Monad 인스턴스는 모두 동일한 방식으로 작동해야합니다. 수학적으로 "(f> ==> g)> => h = f> => (g> = h)"로 표현할 수 있습니다.
처음 두 법칙은 같은 점을 반복합니다. return 양쪽에 정체성 행동이 있어야합니다. bind 운영자.
우리는 이전 예제에서 Monad라는 사실을 깨닫지 못한 채 이미 많은 Monad를 사용했습니다. 특정 목록을 생성하기 위해 List Monad를 사용하는 다음 예제를 고려하십시오.
main = do
print([1..10] >>= (\x -> if odd x then [x*2] else []))
이 코드는 다음 출력을 생성합니다.
[2,6,10,14,18]
Zippers Haskell에서는 기본적으로 데이터 구조의 특정 위치를 가리키는 포인터입니다. tree.
우리는 tree 5 개 요소 [45,7,55,120,56]완벽한 이진 트리로 표현할 수 있습니다. 이 목록의 마지막 요소를 업데이트하려면 업데이트하기 전에 마지막 요소에 도달 할 수 있도록 모든 요소를 탐색해야합니다. 권리?
그러나 만약 우리가 가진 나무가 N 요소는 [(N-1),N]. 그런 다음 원치 않는 모든 것을 통과 할 필요가 없습니다.(N-1)집단. N 번째 요소를 직접 업데이트 할 수 있습니다. 이것이 바로 지퍼의 개념입니다. 전체 트리를 순회하지 않고 해당 값을 업데이트 할 수있는 트리의 특정 위치에 초점을 맞추거나 가리 킵니다.
다음 예에서는 목록의 Zipper 개념을 구현했습니다. 같은 방법으로 Zipper를tree 또는 file 데이터 구조.
data List a = Empty | Cons a (List a) deriving (Show, Read, Eq, Ord)
type Zipper_List a = ([a],[a])
go_Forward :: Zipper_List a -> Zipper_List a
go_Forward (x:xs, bs) = (xs, x:bs)
go_Back :: Zipper_List a -> Zipper_List a
go_Back (xs, b:bs) = (b:xs, bs)
main = do
let list_Ex = [1,2,3,4]
print(go_Forward (list_Ex,[]))
print(go_Back([4],[3,2,1]))
위의 프로그램을 컴파일하고 실행하면 다음과 같은 출력이 생성됩니다.
([2,3,4],[1])
([3,4],[2,1])
여기서는 앞으로 나아갈 때 또는 뒤로 갈 때 전체 문자열의 요소에 초점을 맞추고 있습니다.