Ngôn Ngữ Lập Trình C#
Không cho phép khởi tạo: chúng ta không thể khởi tạo các trường thể hiện (instance
fields) trong cấu trúc, do đó đoạn mã nguồn sau sẽ không hợp lệ:
private int xVal = 20;
private int yVal = 50;
mặc dù điều này thực hiện tốt đối với lớp.
Cấu trúc được thiết kế hướng tới đơn giản và gọn nhẹ. Trong khi các dữ liệu thành viên
private hỗ trợ việc che dấu dữ liệu và sự đóng gói. Một vài người lập trình có cảm giác rằng
điều này phá hỏng cấu trúc. Họ tạo một dữ liệu thành viên public, do vậy đơn giản thực thi
một cấu trúc. Những người lập trình khác có cảm giác rằng những thuộc tính cung cấp một
giao diện rõ ràng, đơn giản và việc thực hiện lập trình tốt đòi hỏi phải che dấu dữ liệu thậm
chí với dữ liệu rất đơn giản. Chúng ta sẽ chọn cách nào, nói chung là phụ thuộc vào quan nệm
thiết kế của từng người lập trình. Dù chọn cách nào thì ngôn ngữ C# cũng hỗ trợ cả hai cách
tiếp cận.
Tạo cấu trúc
Chúng ta tạo một thể hiện của cấu trúc bằng cách sử dụng từ khóa new trong câu lệnh
gán, như khi chúng ta tạo một đối tượng của lớp. Như trong ví dụ 7.1, lớp Tester tạo một thể
hiện của Location như sau:
Location loc1 = new Location( 200, 300);
Ở đây một thể hiện mới tên là loc1 và nó được truyền hai giá trị là 200 và 300.
Cấu trúc là một kiểu giá trị
Phần định nghĩa của lớp Tester trong ví dụ 7.1 trên bao gồm một đối tượng Location là loc1
được tạo với giá trị là 200 và 300. Dòng lệnh sau sẽ gọi thực hiện bộ khởi tạo của cấu trúc
Location:
Location loc1 = new Location( 200, 300);
Sau đó phương tức WriteLine() được gọi:
Console.WriteLine(“Loc1 location: {0}”, loc1);
Dĩ nhiên là WriteLine chờ đợi một đối tượng, nhưng Location là một cấu trúc (một kiểu giá
trị). Trình biên dịch sẽ tự động boxing cấu trúc (cũng giống như trình biên dịch đã làm với
các kiểu dữ liệu giá trị khác). Một đối tượng sau khi boxing được truyền vào cho phương thức
WriteLine(). Tiếp sau đó là phương thức ToString() được gọi trên đối tượng boxing này, do
định ngầm định sẽ được trình biên dịch tạo ra. Chúng ta có thể nhìn thấy điều này nếu bỏ bộ
khởi dựng tạo ra:
/*public Location( int xCoordinate , int yCoordinate)
{
xVal = xCoordinate;
yVal = yCoordinate;
}
*/
và ta thay dòng lệnh đầu tiên trong hàm Main() tạo Location có hai tham số bằng câu lệnh tạo
không tham số như sau:
//Location loc1 = new Location( 200, 300)
Location loc1 = new Location();
Bởi vì lúc này không có phương thức khởi dựng nào khai báo, một phương thức khởi dựng
ngầm định sẽ được gọi. Kết quả khi thực hiện giống như sau:
Loc1 location 0, 0
In myFunc loc: 50, 100
Loc1 location: 0, 0
Bộ khởi tạo mặc định đã thiết lập tất cả các biến thành viên với giá trị 0.
Cấu Trúc
169
.
.
Ngôn Ngữ Lập Trình C#
Ghi chú: Đối với lập trình viên C++ lưu ý, trong ngôn ngữ C#, từ khóa new không phải
luôn luôn tạo đối tượng trên bộ nhớ heap. Các lớp thì được tạo ra trên heap, trong khi các cấu
trúc thì được tạo trên stack. Ngoài ra, khi new được bỏ qua (sẽ bàn tiếp trong phần sau), thì
bộ khởi dựng sẽ không được gọi. Do ngôn ngữ C# yêu cầu phải có phép gán trước khi sử
dụng, chúng ta phải khởi tạo tường minh tất cả các biến thành viên trước khi sử dụng chúng
trong cấu trúc.
Tạo cấu trúc không gọi new
Cấu Trúc
170
.
.
Ngôn Ngữ Lập Trình C#
set
{
xVal = value;
}
}
public int y
{
get
{
return yVal;
}
set
{
yVal = value;
}
}
public override string ToString()
{
return (string.Format(“{0} ,{1}”, xVal, yVal));
}
// biến thành viên lưu tọa độ x, y
public int xVal;
public int yVal;
}
public class Tester
thông qua thuộc tính x và thuộc tính y:
static void Main()
{
Location loc1;
// gán cho biến thành viên
loc1.xVal = 100;
loc1.yVal = 250;
// sử dụng thuộc tính
loc1.x = 300;
loc1.y = 400;
Console.WriteLine( loc1 );
}
Hãy cẩn thận với việc sử dụng các thuộc tính. Mặc dù cấu trúc cho phép chúng ta hỗ trợ đóng
gói bằng việc thiết lập thuộc tính private cho các biến thành viên. Tuy nhiên bản thân thuộc
tính thật sự là phương thức thành viên,và chúng ta không thể gọi bất cứ phương thức thành
viên nào cho đến khi chúng ta khởi tạo tất cả các biến thành viên.
Như ví dụ trên ta thiết lập thuộc tính truy cập của hai biến thành viên xVal và yVal là public
vì chúng ta phải khởi tạo giá trị của hai biến thành viên này bên ngoài của cấu trúc, trước khi
các thuộc tính được sử dụng.
Câu hỏi và trả lời
Câu hỏi 1: Có sự khác nhau giữa cấu trúc và lớp?
Trả lời 1: Đúng có một số sự khác nhau giữa cấu trúc và lớp. Như đã đề cập trong lý thuyết
thì lớp là kiểu dữ liệu tham chiếu còn cấu trúc là kiểu dữ liệu giá trị. Điều này được xem là
sự khác nhau căn bản giữa cấu trúc và lớp. Ngoài ra cấu trúc cũng không cho phép có hàm
hủy và tạo bộ khởi dựng không tham số tường minh. Cấu trúc cũng khác lớp là cấu trúc là
Cấu Trúc
172
.
.