Dart 다트언어에 대하여 알아보기
다트 Dart 다트 언어에 대하여 알아보기
다트 언어에 대하여 알아보도록 하겠습니다.
아래의 다트 Document 원문에 대한 내용입니다.
https://dart.dev/guides/language/language-tour#class-variables-and-methods
목차
왜 다트를 해야할까 ?
다트는 구글이 자바스크립트를 대체하기 위해 2011년 개발한 웹 프로그래밍용 언어이지만 2018년이 될 때까지 배울 필요가 없는 프로그래밍 언어 중 하나로 여겨졌다. 하지만 다트는 새로운 프로그래밍 언어로, 나날이 큰 인기를 얻고 있습니다
현재 Dart는 다양한 VM과 컴파일러 지원을 통해 웹 프런트엔드부터 데스크톱, 서버, 모바일 애플리케이션까지 개발할 수 있다. Dart로 구축한 애플리케이션은 VM 위에서 동작합니다. Dart 프로젝트의 목적은 구조적이지만 유연하고, 프로그래머들에게 자연스럽고, 다양한 종류의 기기에서 돌아가게 하는 것입니다. 현재 구글의 크로스 플랫폼 앱 프레임워크인 Flutter가 이 언어를 사용하고 있습니다.
다트의 중요한 개념
- 변수에 할당할 수 있는 모든 것은 객체 이며 모든 객체는 클래스 의 인스턴스입니다.함수 및 null 도 모두 객체입니다. 모든 객체는 Object 클래스 에서 상속됩니다.
- Dart는 강력하게 타입적이지만. 유형이 예상되지 않는다고 할 때는 dynamic을 사용하여 타입에 lenient 하게 사용할 수도 있습니다.
- Dart는 List<int>(정수 목록) 또는 List<dynamic>(모든 유형의 개체 목록)과 같은 일반 유형을 지원합니다.
- Dart는 main() 함수 같은 탑 레벨 함수, 클래스 또는 객체에 연결된 함수, 함수 내의 함수 작성 등을 지원합니다.
- 마찬가지로 Dart는 최상위 변수 뿐만 아니라 클래스 또는 객체에 연결된 변수를 지원합니다. 인스턴스 변수는 필드 또는 속성이라고도 합니다.
- 자바와는 달리, 다트 키워드가 없습니다 public, protected하고 private. 식별자가 밑줄 (_)로 시작하면 라이브러리 전용입니다. 자세한 내용은 라이브러리 및 가시성을 참조하십시오.
- 식별자는 문자 나 밑줄 (_)로 시작하고 그 뒤에 문자와 숫자를 조합하여 사용할 수 있습니다.
- 다트에는 표현식 (런타임 값이 있음)과 명령문 (없음)이 있습니다. 예를 들어 조건식 condition ? expr1 : expr2의 값은 expr1또는 expr2입니다. 값이없는 if-else 문과 비교하십시오 . 명령문은 종종 하나 이상의 표현식을 포함하지만 표현식은 명령문을 직접 포함할 수 없습니다.
-다트 도구는 경고 와 오류 의 두 가지 문제를 보고 할 수 있습니다 . 경고는 코드가 작동하지 않을 수 있음을 나타내지 만 프로그램 실행을 방해하지는 않습니다. 오류는 컴파일 타임 또는 런타임 일 수 있습니다. 컴파일 타임 오류는 코드가 전혀 실행되지 못하게 합니다.런타임 오류로 인해 코드가 실행되는 동안 예외 가 발생합니다.
다트 언어의 키워드
keyword | description |
abstract | 추상클래스 정의 키워드 |
Implicit interfaces |
암시적 인터페이스 interface 키워드가 없음으로 일반 class로 정의된 내용을 암시적으로 구현하게 할 수 있습니다. |
dynamic | 특정 타입을 지정하지 않고 변수를 선언할 수있는 키워드입니다. |
show | 라이브러리의 일부만 선택적으로 가져올 수 있는 키워드입니다. |
hide | 라이브러리의 일부분을 제외하고 가져올수 있는 키워드입니다. |
as | Typecase에 사용됩니다. |
is | 객체유형을 비교하고 bool을 반환합니다. |
is! | 객체유형을 비교하고 bool을 반환합니다. |
static |
전역 변수 및 전역 메서드 등을 선언할 수 있습니다. 상수 선언은 camelCase를 따릅니다. |
assert |
assert의 bool조건을 확인하여 false라면 에러를 발생시킵니다. - flutter에서는 디버그 모드에서 assertion을 활성화합니다. - dartdevc와 같은 개발 전용 도구는 기본적으로 assetion을 활성화합니다. - 프로덕션 코드에서는 assertion이 무시됩니다. |
enum |
열거형으로 고정된 상수의 값을 나타냅니다. - 각 열거형 값에는 index getter가 있습니다. - 모든 값의 목록을 얻으려면 values 상수를 사용할 수 있습니다. |
in | List와 Set과 같은 반복가능한 Collection에서 이 키워드를 사용하여 반복할 수 있습니다. |
super | extents를 통해 상속받은 클래스에서 부모클래스를 참조할때 사용합니다. |
async |
비동기 함수를 선언할 때 사용합니다. - Future <String> lookUpVersion() async =>'1.0.0'; - Future checkVersion() async { var version = await lookUpVersion(); } - async 함수는 시간이 오래 걸리는 작업을 수행할 수 있지만 해당 작업을 기다리지 않습니다. await를 만나면 실행하며 , await가 완료된 후에 실행을 재개하여 Future 객체를 반환합니다. - async 함수의 에러 핸들은 try.catch 및 finally를 사용합니다. |
await | 비동기 함수의 결과를 기다리는데 사용합니다. |
sync |
동기 Generator함수를 구현할 때 sync*로 선언하고, yield를 이용하여 값을 전달합니다.
비동기 Generator함수를 구현할 때는 async* 를 사용하여 선언하고 yield를 이용하여 값을 전달합니다.
|
Iterable | 동기 Generator |
Stream | 비동기 Generator |
switch_case | switch_case 문 키워드 |
factory | 팩토리 생성자 |
mixin |
- 믹스인은 다른 클래스 코드를 재사용하는 방법입니다. - mixin으로 선언한 클래스는 다른 클래스에서 with 키워드를 이용하여 재사용할 수 있습니다. - 일반적으로 abstract class 대신 사용합니다 - instantiated(인스턴트화) 할 수 없습니다
|
with |
mixin으로 선언한 클래스는 다른 클래스에서 with 키워드를 이용하여 재사용 할 수 있습니다. |
on |
- mixin으로 선언한 클래스에서 사용할 상위 클래스 on키워드를 사용하여 선언합니다 - mixin A on B {}이라고 선언된 클래스가 있다면 - A를 with으로 사용하는 클래스는 반드시 B를 구현하거나 상속받아야 합니다. - on 키워드를 사용하려면 mixin 키워드를 사용해서 믹스인을 선언해야 합니다. |
throw |
이 키워드를 이용하여 예외를 던질 수 있습니다. |
try, on , catch , finally |
- try { } 블록에서 에러가 발생할 수도 있는 구문을 작성합니다. - on [SomeException]{ }블럭에서 SomeException 타입의 에러를 지정하여 처리할 수 있습니다. - catch(e){ } 블록에서는 여러 유형의 예외를 처리하거나 on에서 지정한 타입의 excption을 매개변수로 받을 수 있습니다. - catch(e, s){}로 선언하면 e로 excption을 , s는 stacktrace입니다. - catch나 on블록 안에서 rethrow;를 사용하면 하위 블록으로 에러를 다시 던질 수 있습니다. - finally : 예외 발생 여부와 상관없이 일부 코드가 반드시 실행되도록 합니다. |
final , const |
final은 최종 변수로 값을 변경할 수 없습니다 const는 상수 변수입니다. |
null |
초기화되지 않은 변수의 초기 값은 null입니다. 숫자 형식의 변수(int)조차도 처음에는 null입니다. 숫자는 다트의 다른 모든 것과 마찬가지로 개체이기 때문입니다. |
typedefs |
타입 정의 또는 기능 형 별칭 , 함수 유형을 필드 및 반환 형식을 선언할 때 사용할 수 있는 이름을 제공합니다. |
covariant |
재정의 된 메서드에서 매개 변수는 슈퍼 클래스에있는 해당 매개 변수와 동일한 형식 또는 슈퍼 유형입니다. - convariant를 사용하여 유형을 원래 매개 변수의 하위 유형으로 바꿀 수 있습니다.
- 일반적으로 수퍼 클래스 메서드에 적용하는 것이 가장 적합합니다. - covariant키워드는 단일 매개 변수에 적용되며 또한 세터가 필드에서 지원됩니다. |
다트 언어의 연산자
Operator | Description |
expr++ , expr-- , () , [] , ?. | 단항 접미사 |
-expr ,!expr , ~expr , ++expr , --expr ,await, expr | 단항 접두사 |
* , / , % , ~/ | 곱셈 |
+ , - | 덧 뺄샘 |
<< , >> , >>> | 시프트 |
& | 비트 AND |
^ | 비트 XOR |
| | 비트 OR |
>= > <= < as is is! | 부호 , 유형 |
== , != | 부등호 |
&& | 논리 AND |
|| | 논리 OR |
?? | null의 경우 |
expr1 ? expr2 : expr3 | 가정 ? true : false |
= *= /= += -= &= ^= | 할당 |
UseCase -keyword
abstract
abstract 키워드는 추상 클래스를 정의합니다.
abstract class AbstractContainer{
void updateChildren();
}
class Container extends AbstractContainer{
@override
void updateChildren() {
}
}
as, is
is : 타입 추론 연산자로 키워드 왼쪽 객체가 오른쪽의 타입인지 bool값으로 반환합니다.
as : 캐스팅 연산자로 오른쪽의 타입으로 캐스팅합니다.
main() {
final emp = 1;
print(emp is String);
print(emp is! String);
print(emp as int);
}
assert
assert의 bool조건을 확인하여 false라면 에러를 발생시킵니다.- flutter에서는 디버그 모드에서 assertion을 활성화합니다.
- dartdevc와 같은 개발 전용 도구는 기본적으로 assetion을 활성화합니다.
- 프로덕션 코드에서는 assertion이 무시됩니다.
main() {
int text;
text ??= 0;
final number = 99;
final urlString = "http";
// Make sure the variable has a non-null value.
assert(text != null);
// Make sure the value is less than 100.
assert(number < 100);
// Make sure this is an https URL.
assert(urlString.startsWith('https'),
'URL ($urlString) should start with "https".');
}
async, await
async 키워드는 비동기 함수를 선언할 때 사용합니다.
- async 함수는 시간이 오래 걸리는 작업을 수행할 수 있지만 해당 작업을 기다리지 않습니다. await를 만나면 실행하며 , await가 완료된 후에 실행을 재개하여 Future 객체를 반환합니다.
- async 함수의 에러 핸들은 try.catch 및 finally를 사용합니다.
await 키워드는 비동기 함수의 결과를 기다리는 데 사용합니다.
//String lookUpVersion() => "1.0.0";
Future<String> lookUpVersion() async => "1.0.0";
Future checkVersion() async {
var version = await lookUpVersion();
}
Future checkVersionErrorHandle() async {
try {
var version = await lookUpVersion();
} catch (e) {
// React to inability to look up the version
}
}
Future checkVersionManyOfAsyncFunction() async {
var version = await lookUpVersion();
await lookUpVersion();
await lookUpVersion();
}
Future<int> sumStream(Stream<int> stream) async {
var sum = 0;
await for (var value in stream) {
sum += value;
}
return sum;
}
Future handle(Stream<String> requestServer) async {
await for (var request in requestServer) {
handleRequest(request);
}
}
void handleRequest(String request) {
print(request);
}
main() {
checkVersion();
checkVersionErrorHandle();
checkVersionManyOfAsyncFunction();
final elements = [1, 2, 3];
sumStream(Stream.fromIterable(elements));
handle(Stream.fromIterable(["r1", "r2", "r3"]));
}
constructor(this)
클래스와 이름이 같은 함수를 생성하여 생성자를 선언합니다 (선택적으로 명명된 생성자에 설명된 추가 식별자 )
가장 일반적인 형태의 생성자 인 생성자 생성자는 클래스의 새 인스턴스를 만듭니다.
this 키워드는 현재 인스턴스를 나타냅니다.
class Point {
num x, y;
Point(num x, num y) {
// There's a better way to do this, stay tuned.
this.x = x;
this.y = y;
}
}
convariant
convariant키워드는 매개 변수 유형을 다른 하위 유형으로 대체하여 사용할 수 있습니다.
class Animal {
void chase(Animal x) {
// .....
}
}
class Mouse extends Animal {}
class Cat extends Animal {
void chase(covariant Mouse x) {
//...
}
}
위 예에서는 covariant하위 유형에서의 사용을 보여 주지만 covariant키워드는 슈퍼 클래스 또는 하위 클래스 메서드에 배치 될 수 있습니다. 일반적으로 수퍼 클래스 메서드가 가장 적합합니다. covariant키워드는 단일 매개 변수에 적용되며 또한 세터과 필드에서 지원됩니다.
dynamic
명시적으로 타입이 예상되지 않도록 선언할 수 있습니다.
void log(dynamic object){
print(object.toString());
}
bool convertToBool(dynamic arg){
if(arg is bool) return arg;
if(arg is String )return arg =='true';
throw ArgumentError('Cannnot convert $arg to a bool');
}
main(){
log(true);
log(0.123124);
log(1);
log("sss");
}
enum
Enum클래스는 은 고정된 수의 상수 값을 나타내는 데 사용되는 특수한 종류의 클래스입니다.
열거 형을 사용할 수 있으며 모든 열거 형 값을 처리하지 않으면 경고가 표시됩니다.
열거된 유형에는 다음과 같은 제한이 있습니다.
- 열거 형을 하위 클래스로 만들거나 혼합하거나 구현할 수 없습니다.
- 열거 형을 명시 적으로 인스턴스화 할 수 없습니다.
enum Color { red, green, blue }
main() {
assert(Color.red.index == 0);
assert(Color.green.index == 1);
assert(Color.blue.index == 2);
// -------------------------
List<Color> colors = Color.values;
assert(colors[2] == Color.blue);
// -------------------------
var aColor = Color.blue;
switch (aColor) {
case Color.red:
print('Red as roses!');
break;
case Color.green:
print('Green as grass!');
break;
default: // Without this, you see a WARNING.
print(aColor); // 'Color.blue'
}
}
factory
factory키워드는
항상 클래스의 새 인스턴스를 생성하지 않는 생성자를 구현할 때 키워드를 사용합니다.
예를 들어 팩토리 생성자는 캐시에서 인스턴스를 반환하거나 하위 유형의 인스턴스를 반환할 수 있습니다.
다음 예제는 캐시에서 객체를 반환하는 팩토리 생성자를 보여줍니다.
캐시에 이미 로거가 있다면 생성하지 않고, 없다면 생성합니다.
class Logger {
final String name;
bool mute = false;
// _cache is library-private, thanks to
// the _ in front of its name.
static final Map<String, Logger> _cache = <String, Logger>{};
factory Logger(String name) {
return _cache.putIfAbsent(name, () => Logger._internal(name));
}
Logger._internal(this.name);
void log(String msg) {
if (!mute) print(msg);
}
}
main() {
var logger = Logger('UI');
logger.log('Button cxlicked');
}
final
immutable 한 변수를 선언하는 키워드입니다.
main() {
final name = 'Bob'; // Without a type annotation
final String nickname = 'Bobby';
///컴파일 타임 상수const 가 될 변수에 사용하십시오 .
const bar = 1000000; // Unit of pressure (dynes/cm2)
const double atm = 1.01325 * bar; // Standard atmosphere
///const키워드는 상수 변수를 선언하기위한 것이 아닙니다.
///상수 값 을 생성하고 상수 값 을 생성하는 생성자를 선언하는 데 에도 사용할 수 있습니다
///. 모든 변수는 상수 값을 가질 수 있습니다.
var foo = const [];
final bars = const [];
const baz = []; // Equivalent to `const []`
// name = 'Alice'; // Error: a final variable can only be set once.
// Valid compile-time constants as of Dart 2.5.
const Object i = 3; // Where i is a const Object with an int value...
const list = [i as int]; // Use a typecast.
const map = {if (i is int) i: "int"}; // Use is and collection if.
const set = {if (list is List<int>) ...list}; // ...and a spread.
}
for
main() {
var message = StringBuffer('Dart is fun');
for (var i = 0; i < 5; i++) {
message.write('!');
}
print(message);
//------------------
var callbacks = [];
for (var i = 0; i < 2; i++) {
callbacks.add(() => print(i));
}
callbacks.forEach((c) => c());
//------------------
var collection = [0, 1, 2];
for (var x in collection) {
print(x);
}
}
if, else
main() {
if (isRaining())
print("bringRainCoat");
else if (isSnowing())
print("wearJacket");
else
print("putToDown");
}
bool isRaining() => true;
bool isSnowing() => true;
mixin , on , with
Animal이라는 슈퍼클래스를 상속하는 Mammal , Brid , Fish가 있습니다.
그리고 Mammal , Brid , Fish를 상속하는 하위 클래스들이 있습니다.
그 하위 클래스들은 각각의 행위 fly , swim , walk 등의 행동을 할 수 있습니다.
하위 클래스들이 fly , swim , walk 등의 행동을 가질 때
하위 클래스들이 하나의 슈퍼 클래스만 가진다면 구현을 쉽게 할 수 있습니다.
위 클래스들을 적절히 상속만 시켜주면은 가능합니다.
하지만 Dart에서는 모든 클래스에는 ( Object 제외) 하나의 슈퍼클래스만 가질 수 있습니다.
Walker , Swim , Fly 행동 클래스를 상속하는 대신 인터페이스처럼 구현할 수 도 있지만
여러 클래스에서 동작을 구현해하므로 좋은 해결책이 아닙니다.
여러 클래스 계층에서 클래스코 드를 재사용할 방법으로 mixin을 사용합니다,
mixin usecase 1
class A {}
mixin X on A {}
//class Y on A{}//에러
class p extends A with X {}
//class Q extends X {} //에러
on 키워드는 믹스인의 사용을 선언된 클래스를 확장하거나 구현하는 클래스로만 제한하는 데 사용됩니다.
on 키워드를 사용하려면 mixin 키워드를 사용해서 믹스인을 선언해야 합니다
mixin usecase 2
class A {
String getMessage() => 'A';
}
class B {
String getMessage() => 'B';
}
class P {
String getMessage() => 'P';
}
class AB extends P with A, B {}
class BA extends P with B, A {}
main(List<String> args) {
callSequnce();
typeCheck();
}
void typeCheck() {
AB ab = AB();
print(ab is P);
print(ab is A);
print(ab is B);
BA ba = BA();
print(ba is P);
print(ba is A);
print(ba is B);
}
void callSequnce() {
String result = '';
AB ab = AB();
result += ab.getMessage();
BA ba = BA();
result += ba.getMessage();
print(result);
// 가장 마지막에 선언된 것이 우선됨
}
슈퍼클래스와 with으로 선언한 클래스들이 모두 같은 메소드를 가진다면
가장 마지막에 선언한 클래스의 메소드를 우선적으로 처리합니다.
mixin usecase 3
abstract class Super {
void method() {
print("Super");
}
}
class MySuper implements Super {
void method() {
print("MySuper");
}
}
mixin Mixin on Super {
void method() {
super.method();
print("Sub");
}
}
class Client extends MySuper with Mixin {}
void main() {
Client().method();
}
13 행에서 18 행까지의 mixin 선언은 Super에 대한 수퍼 클래스 제한을 나타냅니다.
믹스 인이 슈퍼에서 제공하는 기능을 사용하기 때문에 이 믹스 인을 클래스에 적용하려면이 클래스가 슈퍼 클래스를 확장하거나 구현해야 합니다..
mixin usecase animal
abstract class Animal {}
abstract class Mammal extends Animal {}
abstract class Bird extends Animal {}
abstract class Fish extends Animal {}
abstract class Walker {
// This class is intended to be used as a mixin, and should not be
// extended directly.
factory Walker._() => null;
void walk() {
print("I'm walking");
}
}
abstract class Swimmer {
// This class is intended to be used as a mixin, and should not be
// extended directly.
factory Swimmer._() => null;
void swim() {
print("I'm swimming");
}
}
abstract class Flyer {
// This class is intended to be used as a mixin, and should not be
// extended directly.
factory Flyer._() => null;
void fly() {
print("I'm flying");
}
}
class Dolphin extends Mammal with Swimmer {}
class Bat extends Mammal with Walker, Flyer {}
class Cat extends Mammal with Walker {}
class Dove extends Bird with Walker, Flyer {}
class Duck extends Bird with Walker, Swimmer, Flyer {}
class Shark extends Fish with Swimmer {}
class FlyingFish extends Fish with Swimmer, Flyer {}
원문 - https://medium.com/flutter-community/dart-what-are-mixins-3a72344011f3
show , hide
show 키워드는 show 키워드로 선언한 클래스만 import 합니다.
hide 키워드는 hide 키워드로 선언한 클래스를 제외한 나머지 클래스를 import 합니다.
// Import only foo.
import 'lib1.dart' show foo;
// Import all names EXCEPT foo.
import 'lib2.dart' hide foo;
main() {
foo();
}
static
import 'dart:math';
class Queue {
static const initialCapacity = 16;
}
class Point {
num x, y;
Point(this.x, this.y);
static num distanceBetween(Point a, Point b) {
var dx = a.x - b.x;
var dy = a.y - b.y;
return sqrt(dx * dx + dy * dy);
}
}
void main() {
assert(Queue.initialCapacity == 16);
var a = Point(2, 2);
var b = Point(4, 4);
var distance = Point.distanceBetween(a, b);
assert(2.8 < distance && distance < 2.9);
print(distance);
}
super
서브클래스에서 슈퍼클래스를 참조하는 키워드입니다.
class Television {
void turnOn() {
print('is super turnOn Method');
}
}
class SmartTelevision extends Television {
void turnOn() {
super.turnOn();
print('is sub turnOn Method');
}
}
main() {
final smartTelevision = SmartTelevision();
smartTelevision.turnOn();
}
switch-case
void case1() {
var command = 'OPEN';
switch (command) {
case 'CLOSED':
print('CLOSED');
break;
case 'PENDING':
print('PENDING');
break;
case 'APPROVED':
print('APPROVED');
break;
case 'DENIED':
print('DENIED');
break;
case 'OPEN':
print('OPEN');
break;
default:
print('DEFAULT');
}
}
sync* , async* , yield
동기 Generator(ex Iterator) 함수를 구현할 때 sync*로 선언하고,
yield를 이용하여 값을 전달합니다.
비동기 Generator함수(ex Stream)를 구현할 때는 async* 를 사용하여 선언하고
yield를 이용하여 값을 전달합니다.
Iterable<int> naturalsTo(int n) sync* {
int k = 0;
while (k < n) yield k++;
}
Stream<int> asynchronousNaturalsTo(int n) async* {
int k = 0;
while (k < n) yield k++;
}
Iterable<int> naturalsDownFrom(int n) sync* {
//생성기가 재귀 적 인 경우 다음을 사용하여 성능을 향상시킬 수 있습니다 yield*.
if (n > 0) {
yield n;
yield* naturalsDownFrom(n - 1);
}
}
main() {
print(naturalsTo(5));
asynchronousNaturalsTo(10).forEach(print);
print(naturalsDownFrom(5));
}
함수가 호출되면 generator함수를 통해 스트림이 만들어집니다. 그리고 스트림이 청취되기 시작하면 함수의 본문이 시작됩니다. 함수가 리턴되면 스트림이 닫힙니다. 스트림이 닫힐 때까지 yeild나 yeild*를 통해서 스트림으로 이벤트가 내보내집니다.
yield*는 다른 비동기생성기로 위임하고 다른 생성기가 값 생성을 중지 한 후에 자체 값 생성을 다시 시작합니다.
throw
import 'dart:html';
//예외를 던지는 것은 표현식이므로 => 문과 표현식을 허용하는 다른 곳에서 예외를 throw 할 수 있습니다.
void distanceTo(Point other) => throw UnimplementedError();
main(){
throw FormatException('Expected at least 1 section');
throw 'Out of llamas!';
}
try_catch_on_finally
- try { } 블록에서 에러가 발생할 수도 있는 구문을 작성합니다.
- on [SomeException]{ }블럭에서 SomeException 타입의 에러를 지정하여 처리할 수 있습니다.
- catch(e){ } 블록에서는 여러 유형의 예외를 처리하거나 on에서 지정한 타입의 excption을 매개변수로 받을 수 있습니다.
- catch(e, s){}로 선언하면 e로 excption을 , s는 stacktrace입니다.
- catch나 on블록 안에서 rethrow;를 사용하면 하위 블록으로 에러를 다시 던질 수 있습니다.
- finally : 예외 발생 여부와 상관없이 일부 코드가 반드시 실행되도록 합니다.
void someErrorOccur() {
print("excute...");
// throw OutOfMemoryError();
throw Exception();
// throw 'Exception';
}
main() {
try {
someErrorOccur();
} on OutOfMemoryError {
// A specific exception
print("is OutOfMemory Exception");
} on Exception catch (e) {
// Anything else that is an exception
print('Unknown exception: $e');
rethrow;
} catch (e, s) {
// No specified type, handles all
print('Something really unknown.. Details: $e');
print('Stack trace : $s');
} finally {
print('end excute..');
}
}
typedefs
타입 정의 또는 기능 형 별칭 , 함수 유형을 필드 및 반환 형식을 선언할 때 사용할 수 있는 이름을 제공합니다.
typedef Compare = int Function(Object a, Object b);
class SortedCollection {
Compare compare;
SortedCollection(this.compare);
}
// Initial, broken implementation.
int sort(Object a, Object b) => 0;
void main() {
SortedCollection coll = SortedCollection(sort);
assert(coll.compare is Function);
assert(coll.compare is Compare);
}
다트 Detail Example
anonymous function
다트의 익명 함수는 아래와 같이 생성할 수 있습니다.
([[Type] param1[, …]]) {
codeBlock;
};
예
main() {
var list = ['apples', 'bananas', 'oranges'];
list.forEach((item) {
print('${list.indexOf(item)}: $item');
});
//화살표 표기법을 이용하여 단축 가능
list.forEach((item) => print('${list.indexOf(item)}: $item'));
}
collection
Set
main() {
//Set을 아래와 같이 정의 할 수 있습니다.
var halogens = {
'fluorine',
'chlorine',
'bromine',
'iodine',
'astatine',
'fluorine'
};
assert(halogens.length == 5);
// String 유형의 Set 정의
// Set<String> names = {}; // This works, too.
// var names = {}; // Creates a map, not a set.
var names = <String>{};
//Set에 데이터 추가
names.add("ss");
names.addAll(halogens);
//상수 Set
// constantSet.add('helium'); // Uncommenting this causes an error.
final constantSet = const {
'fluorine',
'chlorine',
'bromine',
'iodine',
'astatine',
};
}
Map
main(){
//선언하며 데이터 추가
var gifts = {
// Key: Value
'first': 'partridge',
'second': 'turtledoves',
'fifth': 'golden rings'
};
var nobleGases = {
2: 'helium',
10: 'neon',
18: 'argon',
};
//선언 후 데이터 추가
var gifts2 = Map();
gifts['first'] = 'partridge';
gifts['second'] = 'turtledoves';
gifts['fifth'] = 'golden rings';
var nobleGases2 = Map();
nobleGases[2] = 'helium';
nobleGases[10] = 'neon';
nobleGases[18] = 'argon';
//상수 맵 선언
// constantMap[2] = 'Helium'; // Uncommenting this causes an error.
final constantMap = const {
2: 'helium',
10: 'neon',
18: 'argon',
};
}
List
void main() {
List<int> list = [2, 1, 3];
list.add(4);
print(list);
print(list[0]);
}
일급 객체
다트는 언어는 매개변수로 함수를 전달할 수 있고
변수에 함수를 할당할 수 있는 일급 객체를 지원합니다.
void printElement(int element) {
print(element);
}
var list = [1, 2, 3];
main() {
//매개변수로 함수 전달
list.forEach(printElement);
//변수에 함수 할당
var loudify = (msg) => '!!! ${msg.toUpperCase()} !!!';
assert(loudify('hello') == '!!! HELLO !!!');
}
암시적 인터페이스(implic_interface)
다트에서는 interface 키워드가 없습니다
일반 class로 정의 한 후 구현하고자 하는 클래스에서 implements를 사용하여
정의된 함수나 변수를 모두 구현하게 할 수 있습니다.
class Person {
final _name;
Person(this._name);
String greet(String who) => "Hello, $who. I am $_name";
}
class Impostor implements Person{
@override
get _name => '';
@override
String greet(String who)=> "Hi, $who. Do you know who I am ?";
}
String greetBob(Person person)=>person.greet("Bob");
main(){
print(greetBob(Person("kathy")));
print(greetBob(Impostor()));
}
어휘 폐쇄(lexcial_closures)
폐쇄 기능은 원래의 범위 밖에서 사용되는 경우에도, 그 어휘 범위의 변수에 액세스 하는 기능 개체입니다.
함수는 주변 범위에 정의된 변수를 닫을 수 있습니다. 다음 예제에서는 makeAdder() 변수를 캡처합니다
addBy. 반환된 함수가 어디를 가든 기억 addBy 합니다.
Function makeAdder(num addBy) {
return (num i) => addBy + i;
}
void main() {
// Create a function that adds 2.
var add2 = makeAdder(2);
// Create a function that adds 4.
var add4 = makeAdder(4);
assert(add2(3) == 5);
assert(add4(3) == 7);
}
중첩 함수
함수 안에 중첩하여 함수를 선언할 수 있습니다.
bool topLevel = true;
void main() {
var insideMain = true;
void myFunction() {
var insideFunction = true;
void nestedFunction() {
var insideNestedFunction = true;
assert(topLevel);
assert(insideMain);
assert(insideFunction);
assert(insideNestedFunction);
}
}
}
스프레드 연산자(spread operator)와 컬렉션
스프레드 연산자 (...)와 널 인식 스프레드 연산자 (...?)를 도입하여 여러 요소를 컬렉션에 간결하게 삽입할 수 있습니다.
예를 들어, 스프레드 연산자 (...)를 사용하여 목록의 모든 요소를 다른 목록에 삽입할 수 있습니다.
스프레드 연산자의 오른쪽에 있는 표현식이 벌인 경우 널 인식 스프레드 연산자 (...?)를 사용하여 예외를 피할 수 있습니다.
또한 컬렉션에 if 및 for가 도입되어 조건부 ( if) 및 반복 ( for)을 사용하여 컬렉션을 작성하는 데 사용할 수 있습니다.
main() {
var list = [1, 2, 3];
var list2 = [0, ...list];
assert(list2.length == 4);
var list3;
var list4 = [0, ...?list3];
assert(list4.length == 1);
//collection if
bool promoActive = true;
var nav = ['Home', 'Furniture', 'Plants', if (promoActive) 'Outlet'];
assert(nav.length == 4);
//collection for
var listOfInts = [1, 2, 3];
var listOfStrings = ['#0', for (var i in listOfInts) '#$i'];
assert(listOfStrings[1] == '#1');
assert(listOfStrings.length == 4);
}
널 인식 연산자 (null_aware_operators)
??= 을 사용하여 해당 값이 null이라면 초기화를 진행합니다.
nul이 아니라면 값이 변경되지 않습니다.
//널 인식 연산자
int a; // The initial value of a is null.
a ??= 3;
print(a); // <-- Prints 3.
a ??= 5;
print(a); // <-- Still prins 3.
print(1 ?? 3); // <-- Prints 1.
print(null ?? 12); // <-- Prints 12.
문자열 보간(string Interpolation)
"" 안에 $또는 ${}으로 값을 부여할 수 있습니다.
print('${3 + 2}');
print('${"word".toUpperCase()}');
조건부 프로퍼티 접근(conditionalPropertyAccess)
null 일 수 있는 객체의 속성이나 메서드에 대한 접근을 보호하려면 점 (.) 앞에 물음표 (?)를 넣으십시오
null을 체크하면서 프로퍼티나 함수 호출이 가능합니다.
var myObject = new MyObject();
myObject
?.myProperty; //null 일 수있는 객체의 속성이나 메서드에 대한 액세스를 보호하려면 ?점 ( .) 앞에 물음표 ( )를 넣으십시오 .
myObject?.myProperty?.someMethod(); //null 체크를 하며 , 연속 호출 가능
화살표 문법(arrow syntax)
=> 다트 코드에서 기호를 보았을 것입니다. 이 화살표 구문은 표현식을 오른쪽으로 실행하고 값을 반환하는 함수를 정의하는 방법입니다.
final aListOfStrings = ['one', 'two', 'three'];
bool hasEmpty = aListOfStrings.any((s) {
return s.isEmpty;
});
bool hasEmpty2 = aListOfStrings.any((s) => s.isEmpty);
//위의 두 방법은 모두 같은 방법입니다.
cascade
동일한 객체에서 일련의 작업을 수행하려면 cascade (..)을 사용하십시오.
우리는 모두 다음과 같은 표현을 보았습니다.
자바의 Builder Pattern처럼 사용 가능합니다.
알아야 할 것은 리턴 타입이 자기 자신이 아니며 참조라는 것입니다.
void cascade(){
final myObject = new MyObject();
myObject..someMethod()..someMethod()..someMethod();
var bigObject = fillBigObject(new BigObject());
print(bigObject.anInt);
print(bigObject.aString);
print(bigObject.aList);
}
BigObject fillBigObject(BigObject obj) {
// Create a single statement that will update and return obj:
return obj
..anInt = 1
..aString = 'String!'
..aList = [3, 0]
..allDone();
}
//cascade not used
var button = querySelector('#confirm');
button.text = 'Confirm';
button.classes.add('important');
button.onClick.listen((e) => window.alert('Confirmed!'));
//cascade used
querySelector('#confirm')
..text = 'Confirm'
..classes.add('important')
..onClick.listen((e) => window.alert('Confirmed!'));
Getter , Setter
class MyClass {
int _aProperty = 0;
int get aProperty => _aProperty;
set aProperty(int value) {
if (value >= 0) {
_aProperty = value;
}
}
List<int> _values = [];
void addValue(int value) {
_values.add(value);
}
// A computed property.
int get count {
return _values.length;
}
int get countSum => _values.fold(0, (acc, v) => acc + v);
}
final myClass = new MyClass();
myClass.aProperty = -1;
print(myClass.aProperty);
myClass.aProperty = 10;
print(myClass.aProperty);
myClass.addValue(1);
myClass.addValue(2);
myClass.addValue(3);
print(myClass.count);
print(myClass.countSum);
}
선택적 파라미터(OptionalPositionalParameters)
매개변수 선언에 []를 사용하여 매개변수를 선택적으로 사용할 수 있습니다.
int sumUp(int a, int b, int c) => a + b + c;
int sumUpToFive(int a, [int b, int c, int d, int e]) {
int sum = a;
if (b != null) sum += b;
if (c != null) sum += c;
if (d != null) sum += d;
if (e != null) sum += e;
return sum;
}
//Dart를 사용하면 매개 변수를 대괄호로 묶어 선택적으로 지정할 수 있습니다.
int total;
total ??= 0;
total = sumUp(1, 2, 3);
print(total);
total = sumUpToFive(1, 2);
print(total);
total = sumUpToFive(1, 2, 3, 4, 5);
print(total);
명명 생성자, 생성자에 this 사용하기
1.this. 를 사용한 생성자 자
class MyColor {
int red;
int green;
int blue;
MyColor(this.red, this.green, this.blue); // this.를 사용한 생성자 자
}
final color2 = MyColor(80,80,128);
2. 명명 매개변수 생성자
class MyColor {
int red;
int green;
int blue;
MyColor({this.red, this.green, this.blue}); // 명명된 매개변수 생성자
}
final color2 = MyColor(red:80,blue:80,green:128);
3. 명명 매개변수 생성자와 기본값 0
class MyColor {
int red;
int green;
int blue;
MyColor({this.red=0,this.green=0,this.blue=0}) ; //명명 매개변수 생성자와 기본값 0
}
final color2 = MyColor(red: 80, green: 80, blue: 128);
4. 선택적 매개변수와 기본값 0
class MyColor {
int red;
int green;
int blue;
MyColor([this.red=0,this.green=0,this.blue=0]) ; //선택적 매개변수와 기본값 0
}
생성자와 assert
생성자를 이용해 객채를 생성할 때 ,
생성자의 입력값을 assert로 확인할 수 있습니다.
class NonNegativePoint {
int x;
int y;
NonNegativePoint(this.x, this.y)
: assert(x >= 0),
assert(y >= 0) {
print('I just made a NonNegativePoint: ($x, $y)');
}
}
명명 생성자
생성자에 다른 이름을 부여할 수 있습니다.
class Points {
num x, y;
Points(this.x, this.y);
Points.origin() {
x = 0;
y = 0;
}
}
void namedConstructors() {
final myPoints = Points.origin();
print("${myPoints.x},${myPoints.y}");
}
팩토리 생성자 ( factory constructor)
팩토리 생성자를 이용하여 여러 타입의 객체를 생성할 수 있습니다.
abstract class Shape {
num get area;
factory Shape(String type) {
if (type == 'circle')
return Circle(2);
else if (type == 'square')
return Square(2);
else
return null;
}
}
void factoryConstructor(){
final circle = Shape('circle');
final square = Shape('square');
final nullShape= Shape('aa');
print(circle.area);
print(square.area);
print(nullShape);
}
리디렉션 생성자(redirectingConstructor)
this 키워드를 이용하여 다른 생성자를 호출할 수 있습니다.
class Automobile{
String make;
String model;
int mpg;
// The main constructor for this class.
Automobile(this.make,this.model,this.mpg);
// Delegates to the main constructor.
Automobile.hybrid(String make,String model):this(make,model,60);
// Delegates to a named constructor
Automobile.fancyHybrid() : this.hybrid('Futurecar', 'Mark 2');
}
상수 생성자(ConstConstructors)
클래스가 절대 변경되지 않는 객체를 생성하는 경우 이러한 객체를 컴파일 타임 상수로 만들 수 있습니다.
이렇게 하려면 const생성자를 정의하고 모든 인스턴스 변수가 최종적인지 확인하십시오.
class ImmutablePoint {
const ImmutablePoint(this.x, this.y);
final int x;
final int y;
static const ImmutablePoint origin =
ImmutablePoint(0, 0);
}
void constConstructors(){
final immutablePoints =ImmutablePoint.origin;
print("immutablePoints ${immutablePoints.x},${immutablePoints.y}");
}
functional dart
String scream(int length) => "A${'a' * length}h!";
main() {
//origin
final values = [1, 2, 3, 5, 10, 50];
for (var length in values) {
print(scream(length));
}
//functional
values.map(scream).forEach(print);
//complex functional
values.skip(1).take(3).map(scream).forEach(print);
}
전체 샘플 코드 보러 가기
https://github.com/qjatjr1108/Playground_Dart
Reference
https://dart.dev/guides/language/language-tour#class-variables-and-methods
https://medium.com/flutter-community/dart-what-are-mixins-3a72344011f3
-