Facebook Pixel

Tự học ngôn ngữ Dart: Type Inference, Type Annotation, Dynamic Types, Constants

07 Jul, 2021

Nguyên

Author

Tự học ngôn ngữ Dart. Trong bài viết này chúng ta sẽ đi tìm hiểu về các khái niệm type inference, type annotation, dynamic types, constants.

Tự học ngôn ngữ Dart: Type Inference, Type Annotation, Dynamic Types, Constants

Mục Lục

Ngôn ngữ Dart là một ngôn ngữ strongly typed. Ngôn ngữ strongly typed là một ngôn ngữ đảm bảo rằng kiểu của một object không thay đổi đột ngột. Nó có các quy tắc và những hạn chế để đảm bảo rằng giá trị của một biến luôn khớp với static type của biến đó.

Ngôn ngữ lập trình static typed là những ngôn ngữ mà các biến không cần phải định nghĩa trước khi chúng được sử dụng.

Mặc dù kiểu (types) là bắt buộc trong Dart nhưng kiểu chỉ định (Type Annotation) có thể có hoặc không bởi vì Dart có kiểu tự suy (Type Inference).

Trước tiên hãy hiểu kiểu tự suy là như thế nào đã nhé!.

1. Kiểu tự suy (Type Inference) là gì?

Như tên gọi của nó, kiểu tự suy là một khả năng của ngôn ngữ lập trình để tự suy luận ra kiểu mỗi khi người dùng không đề cập tới.

Trong bài viết "kiểu dữ liệu và biến", chúng ta đã biết syntax để khai báo một biến như sau:

khai báo biến

Kiểu tự suy cho phép chúng ta khai báo một biến mà không cần đề cập rõ ràng về kiểu dữ liệu là gì. Trong ngôn ngữ Dart, chúng ta thay thế từ khóa var vào chỗ của dataType . Hãy xem Syntax sau đây:

Type Inference

Kiểu biến được tự suy từ giá trị ban đầu của nó nếu có. Hãy xem ví dụ đơn giản dưới đây:

Dart
main() {
  var bookTitle = "Lord of the Rings: The Fellowship of the Ring";
  var bookAuthor = "J. R. R. Tolkien";
  var bookNoOfPages = 423;

  // Driving Code
  print(bookTitle);
  print(bookAuthor);
  print(bookNoOfPages);
}

Khi chạy đoạn code trên thì chương trình cho ra output:

Lord of the Rings: The Fellowship of the Ring
J. R. R. Tolkien
423

Mặc dù chúng ta không hề khai báo bookTitle là một kiểu String. Nó đã tự suy cho chúng ta.

Kiểm tra kiểu

Trong bài viết "kiểu dữ liệu và biến", chúng ta đã nói về đối tượng và đối tượng thì có những thuộc tính đại diện cho những thông tin mà đối tượng đó biết.

Vậy đề xem những thông tin về kiểu dữ liệu của đối tượng thì chúng ta có thể dùng thuộc tính runtimeType của đối tượng đó.

Trong đoạn code bên dưới, chúng ta sẽ in ra kiểu dữ liệu của biến bookTitle bằng cách gọi thuộc tính runtimeType của nó.

Ở đoạn này bạn đừng lo lắng quá về Syntax nhé. Chúng ta sẽ được biết trong phần sau.
Dart
main() {
  var bookTitle = "Lord of the Rings: The Fellowship of the Ring";

  print(bookTitle.runtimeType);
}

Output: String.

Điều này chứng mình rằng bookTitle được tự suy là kiểu string và chúng ta không cần gán nó với bất cứ giá trị của kiểu nào khác ngoài String

2. Kiểu chỉ định (Type Annotation)

Khi chúng ta nói về kiểu biến được tự suy có nghĩa là những biến ở lần khai báo sau sẽ không được xem xét mà nó sẽ tự suy luôn. Nếu những kiểu đó rõ ràng thì hãy tự suy còn nếu nó không được như bạn mong muốn, thì bạn có thể thêm kiểu chỉ định cho nó.

Trong ví dụ bên dưới, chúng ta sẽ khai báo biến number bằng cách sử dụng từ khóa var. Chúng ta muốn một biến chứa cả kiểu intdouble.

Dart
main() {
  var number = 3;
  print(number);

  number = 3.2;
  print(number);
}

Output:
main.dart:5:12: Error: A value of type 'double' can't be assigned to a variable of type 'int'. Try changing the type of the left hand side, or casting the right hand side to 'int'. number = 3.2; ^

Output của đoạn code trên sẽ bị lỗi. Khi chúng ta gán giá trị ban đầu cho number là một số nguyên thì trình biên dịch sẽ tự suy number là kiểu của int. Cho nên khi ta tiếp tục gán một giá trị của kiểu double thì trình biên dịch sẽ báo lỗi.

Trường hợp này thì chúng ta có thể sử dụng kiểu chỉ định và khai báo biến number với kiểu dữ liệu là num. Nhớ rằng kiểu num thì đã bao gồm cả intdouble.

Dart
main() {
  num number = 3;
  print(number);

  number = 3.2;
  print(number);
}

Output:
3
3.2

3. Dynamic Types

Nếu bạn muốn một biến để chứa nhiều đối tượng của nhiều kiểu khác nhau thì bạn có thể khai báo biến đó bằng cách sử dụng từ khóa dynamic.

Dart
main() {
  dynamic dynamicVariable = 'A string'; // type String
  print(dynamicVariable);

  dynamicVariable = 5; // type int
  print(dynamicVariable);

  dynamicVariable = true; // type bool
  print(dynamicVariable);
}

Output:
A string
5
true

Trong đoạn code trên, chúng ta khai báo một biến dynamicVariable và gán cho nó giá trị ban đầu của kiểu String. Sau đó chúng ta tiếp tục gán cho nó một giá trị của kiểu int và cuối cùng là giá trị của kiểu bool. Đoạn code vẫn chạy trơn tru mà không hiển thị lỗi nào cả.

4. Xác định hằng số (Constants)

Thỉnh thoảng chúng ta sẽ tạo ra một biến và gán cho nó một giá trị cụ thể với ý định sẽ không bao giờ thay đổi nó. Để chương trình chạy thành công, điều quan trọng nhất là giá trị của biến được giữ nguyên trong suốt thời gian tồn tại của nó.

Để tạo một biến như vậy, các từ khóa finalconst nên được sử dụng. Trước khi chúng ta tìm hiểu về finalconst thực sự làm được những gì? Thì trước tiên, chúng ta cần phải biết sự khác nhau giữa thời gian biên dịch (compile-time)thời gian chạy chương trình (runtime)

4.1 Thời gian biên dịch (compile-time) và thời gian chạy chương trình (runtime)

compile-time & runtime

Thời gian biên dịch và thời gian chạy là các thuật ngữ lập trình đề cập đến các giai đoạn khác nhau trong vòng đời của chương trình. Để tạo ra một chương trình, đầu tiên bạn cần phải viết một vài source code. Source code sẽ quyết định cách mà chương trình sẽ hoạt động.

Sự biên dịch và thời gian biên dịch

Máy tính sẽ không hiểu những đoạn code mà chúng ta viết. Chúng chỉ biết một ngôn ngữ duy nhất đó chính là ngôn ngữ của máy, những đoạn mã 1 và 0. Source code cần được biên dịch sang định dạng code của máy để nó có thể thực thi chương trình. Quá trình này được gọi là sự biên dịch.

Giờ chúng ta đã biết sự biên dịch là gì rồi, dựa trên khái niệm đó chúng ta sẽ tiếp tục tìm hiểu về thời gian biên dịch thông qua ví dụ dưới đây nhé!

Chúng ta đã xác định một vài biến intdouble với những giá trị ban đầu trong chương trình của chúng ta. Những biến này luôn có cùng giá trị ban đầu bất cứ khi nào chúng ta chạy chương trình. Nó luôn được xác định tại thời điểm của sự biên dịch. Vậy nên những giá trị này là những giá trị cố định tại thời gian biên dịch.

Run và Run-time

Sau khi chương trình được biên dịch, chúng ta có thể chạy chương trình đó.

Bạn có nhớ lúc chúng ta lấy thông tin đầu vào của người dùng ở bài viết "Ứng dụng Dart đầu tiên" không? Tùy thuộc những input mà người dùng nhập vào, những giá trị sẽ thay đổi mỗi khi chúng ta chạy chương trình. Những giá trị đó không thể được xác định cho đến khi chương trình thực sự được chạy. Nó chỉ được cố định tại thời gian chạy chương trình.

4.2 Biến không bao giờ thay đổi

Để tạo ra những biến mà những giá trị của chúng không thể thay đổi, ngôn ngữ Dart cung cấp cho chúng ta 2 keyword finalconst. Thay vì khai báo một biến sử dụng var hoặc một kiểu dữ liệu khác chúng ta sẽ sử dụng finalconst.

Sử dụng final

Một biến cuối cùng (biến được khởi tạo bởi từ khóa final)  được khai báo lần đầu tiên và nó sẽ được sử dụng xuyên suốt và không thể khai báo lại. Giá trị cuối cùng của biến sẽ được biết tại thời gian chạy.

Dart
import 'dart:io';
 
main() {
  final name = stdin.readLineSync();
  print("Hello " + name);
}   

Nếu chúng ta gán lại giá trị khác cho biến name, chúng ta sẽ gặp lỗi.

Bash
import 'dart:io';
 
main() {
  final name = stdin.readLineSync();
  name = stdin.readLineSync();
  print("Hello " + name);
}   

Sử dụng const

Một biến hằng số (biến được khởi tạo bởi từ khóa const) nên được tạo ra khi bạn biết giá trị của nó tại thời gian biên dịch. Giống như biến final, một biến hằng số chỉ có thể thiết lập một lần.

Dart
main() {
  const name = "Bob";

  // Driver Code
  print(name);
}

Output: Bob

Tương tự, nếu chúng ta thử gán lại giá trị cho biến name chúng ta sẽ gặp lỗi.

Dart
main() {
  const name = "Bob";
  name = "Jack";

  // Driver Code
  print(name);
}

Output:
main.dart:3:3: Error: Setter not found: 'name'.
name = "Jack";
^^^^

Ở những ví dụ trên có thể bạn không thấy rõ được sự khác nhau giữa finalconst . Sự khác nhau của chúng chỉ có thể được thể hiện qua các chương trình phức tạp hơn. Tất cả những gì bạn cần biết lúc này là cả hai đều chỉ được thiết lập một lần. Và biến final thì có thể thiết lập tại thời gian chạy và thời gian biên dịch. Trong khi đó biến const chỉ được thiết lập tại thời gian biên dịch mà thôi.

Qua bài viết này bạn đã hiểu hơn về kiểu chỉ định, kiểu tự suy, kiểu dynamic và constants. Chúng ta sẽ tiếp tục tìm hiểu về các toán tử trong ngôn ngữ Dart ở bài viết sau.

Bài viết liên quan

Lập trình backend expressjs

xây dựng hệ thống microservices
  • Kiến trúc Hexagonal và ứng dụngal font-
  • TypeScript: OOP và nguyên lý SOLIDal font-
  • Event-Driven Architecture, Queue & PubSubal font-
  • Basic scalable System Designal font-

Đăng ký nhận thông báo

Đừng bỏ lỡ những bài viết thú vị từ 200Lab