Ví dụ 6.1: Định nghĩa các chuyển đổi và toán tử cho lớp Fraction. ----------------------------------------------------------------------------- using System; public class Fraction { public Fraction(int numerator,int denominator) { Console.WriteLine("In Fraction Constructor( int, int) "); this.numerator = numerator; this.denominator = denominator; } public Fraction(int wholeNumber) { Console.WriLine("In Fraction Constructor( int )"); numerator = wholeNumber; denominator = 1; } public static implicit operator Fraction( int theInt ) { Console.WriteLine(" In implicit conversion to Fraction"); return new Fraction( theInt ); } public static explicit operator int( Fraction theFraction ) { Console.WriteLine("In explicit conversion to int"); return theFraction.numerator / theFraction.denominator; } public static bool operator == ( Fraction lhs, Fraction rhs) { Console.WriteLine("In operator =="); if ( lhs.numerator == rhs.numerator && lhs.denominator == rhs.denominator ) { return true; } // thực hiện khi hai phân số không bằng nhau return false; } public static bool operator != ( Fraction lhs, Fraction rhs) { Console.WriteLine("In operator !="); return !( lhs == rhs ); } public override bool Equals( object o ) { Console.WriteLine("In method Equals"); if ( !(o is Fraction )) { return false; } return this == ( Fraction ) o; } public static Fraction operator+( Fraction lhs, Fraction rhs ) { Console.WriteLine("In operator +"); if (lhs.denominator == rhs.denominator ) { return new Fraction( lhs.numerator + rhs.numerator, lhs.denominator ); } //thực hiện khi hai mẫu số khộng bằng nhau int firstProduct = lhs.numerator * rhs.denominator; int secondProduct = rhs.numerator * lhs.denominator; return new Fraction( firstProduct + secondProduct, lhs.denominator * rhs.denominator); } public override string T oString() { string s = numerator.ToString() + "/" + denominator.ToString(); return s; } //biến thành viên lưu tử số và mẫu số private int numerator; private int denominator; } public class Tester { static void Main() { Fraction f1 = new Fraction( 3, 4); Console.WriteLine("f1:{0}",f1.ToString() ); Fraction f2 = new Fraction( 2, 4); Console.WriteLine("f2:{0}",f2.ToString() ); Fraction f3 = f1 + f2; Console.WriteLine("f1 + f2 = f3:{0}",f3.ToString()); Fraction f4 = f3 + 5; Console.WriteLine("f4 = f3 + 5:{0}",f4.ToString()); Fraction f5 = new Fraction( 2, 4); if( f5 == f2 ) { Console.WriteLine("f5:{0}==f2:{1}" , f5.ToString(), f2.ToString()); } } } ----------------------------------------------------------------------------- Lớp Fraction bắt đầu với hai hàm khởi dựng: một hàm lấy một tử số và mẫu số, còn hàm kia lấy chỉ lấy một số làm tử số. Tiếp sau hai bộ khởi dựng là hai toán tử chuyển đổi. Toán tử chuyển đổi đầu tiên chuyển một số nguyên sang một phân số: public static implicit operator Fraction( int theInt ) { return new Fraction( theInt); } Sự chuyển đổi này được thực hiện một cách ngầm định bởi vì bất cứ số nguyên nào cũng có thể được chuyển thành một phân số bằng cách thiết lập tử số bằng giá trị số nguyên và mẫu số có giá trị là 1. Việc thực hiện này có thể giao lại cho phương thức khởi dựng lấy một tham số. Toán tử chuyển đổi thứ hai đượ c thực hiện một cách tường minh, chuyển từ một Fraction ra một số nguyên: public static explicit operator int( Fraction theFraction ) { return theFraction.numerator / theFraction.denominator; } Bởi vì trong ví dụ này sử dụng phép chia nguyên, phép chia này sẽ cắt bỏ phần phân chỉ lấy phần nguyên. Do vậy nếu phân số có giá trị là 16/15 thì kết quả số nguyên trả về là 1. Một số các phép chuyển đổi tốt hơn bằng cách sử dụng làm tròn số. Tiếp theo sau là toán tử so sánh bằng (==) và toán tử so sánh không bằng (!=). Chúng ta nên nhớ rằng khi thực thi toán tử so sánh bằng thì cũng phải thực thi toán tử so sánh không bằng. Chúng ta đã đị nh nghĩa giá trị bằng nhau giữa hai Fraction khi tử số bằng tử số và mẫu số bằng mẫu số. Vi dụ, như hai phân số 3/4 và 6/8 thì không được so sánh là bằng nhau. Một lần nữa, một sự thực thi tốt hơn là tối giản tử số và mẫu số khi đó 6/8 sẽ đơn giản thành 3/4 và khi đó so sánh hai phân số sẽ bằng nhau. Trong lớp này chúng ta cũng thực thi phủ quyết phương thức Equals() c ủa lớp object, do đó đối tượng Fraction của chúng ta có thể được đối xử một cách đa hình với bất cứ đối tượng khác. Trong phần thực thi của phương thức chúng ta ủy thác việc so sánh lại cho toán tử so sánh bằng cách gọi toán tử (==). Lớp Fraction có thể thực thi hết tất cả các toán tử số học như cộng, trừ, nhân, chia. Tuy nhiên, trong phạm vi nhỏ hẹp của minh họa chúng ta chỉ thực thi toán t ử cộng, và thậm chí phép cộng ở đây được thực hiện đơn giản nhất. Chúng ta thử nhìn lại, nếu hai mẫu số bằng nhau thì ta cộng tử số: public static Fraction operator + ( Fraction lhs, Fraction rhs) { if ( lhs.denominator == rhs.denominator) { return new Fraction( lhs.numerator + rhs.numerator, lhs.denominator); } } Nếu mẫu số không cùng nhau, thì chúng ta thực hiện nhân chéo: int firstProduct = lhs.numerator * rhs.denominator; int secondProduct = rhs.numerator * lhs.denominator; return new Fraction( firstProduct + secondProduct, lhs.denominator * rhs.denominator); Cuối cùng là sự phủ quyết phương thức ToString() của lớp object, phương thức mới này thực hiện viết xuất ra nội dung của phân số dưới dạng : tử số / mẫu số: public override string ToString() { string s = numerator.ToString() + “/” + denominator.ToString(); return s; } Chúng ta tạo một chuỗi mới bằng cách gọi phương thức ToString() của numerator. Do numerator là một đối tượng, nên trình biên dịch sẽ ngầm định thực hiện boxing số nguyên numerator và sau đó gọi phương thức ToString(), trả về một chuỗi thể hiện giá trị của số nguyên numerator. Sau đó ta nối chuỗi với “/” và cuối cùng là chuỗi thể hiện giá trị của mẫu số. Với lớp Fraction đ ã tạo ra, chúng ta thực hiện kiểm tra lớp này. Đầu tiên chúng ta tạo ra hai phân số 3/4, và 2/4: Fraction f1 = new Fraction( 3, 4); Console.WriteLine("f1:{0}",f1.ToString() ); Fraction f2 = new Fraction( 2, 4); Console.WriteLine("f2:{0}",f2.ToString() ); Kết quả thực hiện các lệnh trên như sau: In Fraction Constructor(int, int) f1: 3/4 In Fraction Constructor(int, int) f2: 2/4 Do trong phương phức khởi dựng của lớp Fraction chúng ta có gọi hàm WriteLine() để xuất ra thông tin bộ khởi dựng nên khi tạo đối tượng (new) thì cũng các thông tin này sẽ được hịển thị. Dòng tiếp theo trong hàm Main() sẽ gọi toán tử cộ ng, đây là phương thức tĩnh. Mục đích của toán tử này là cộng hai phân số và trả về một phân số mới là tổng của hai phân số đưa vào: Fraction f3 = f1 + f2; Console.WriteLine(“f1 + f2 = f3: {0}”, f3.ToString()); Hai câu lệnh trên sẽ cho ra kết quả như sau: In operator + In Fraction Constructor( int, int) f1 + f2 = f3: 5/4 Toán tử + được gọi trước sau đó đến phương thức khởi dựng của đối tượng f3. Phương thức khởi dựng này lấy hai tham số nguyên để tạo tử số và mẫu số của phân số mới f3. Hai câu lệnh tiếp theo cộng một giá trị nguyên vào phân số f3 và gán kết quả mới về cho phân số mới f4: Fraction f4 = f3 + 5; Console.WriteLine(“f3 + 5 = f4: {0}”, f4.ToString()); Kết quả được trình bày theo thứ tự sau: In implicit conversion to Fraction In Fraction Construction(int) In operator+ In Fraction Constructor(int, int) f3 + 5 = f4: 25/4 Ghi chú: rằng toán tử chuyển đổi ngầm định được gọi khi chuyển 5 thành một phân số. Phân số được tạo ra từ toán tử chuyển đổi ngầm định này gọi phương thức khởi dựng một tham số để tạo phân số mới 5/1. Phân số mới này sẽ được chuyển thành toán hạng trong phép cộng với phân số f3 và kết quả trả về là phân số f4 là tổng của hai phân số trên. Thử nghiệm cuối cùng là tạo một phân số mới f5, rồi sau đó gọi toán tử nạp chồng so sánh bằng để kiểm tra xem hai phân số có bằng nhau hay không. Câu hỏi và trả lời Câu hỏi 1 : Có phải khi xây dựng các lớp chúng ta chỉ cần dùng nạp chồng toán tử với các chức năng tính toán ? Trả lời 1 : Đúng là như vậy, việc thực hiện nạp chồng toán tử rất tự nhiên và trực quan. Tuy nhiên một số ngôn ngữ .NET như VB.NET không hỗ trợ việc nạp chồng toán tử nên, tốt nhất nếu muốn cho lớp trong C# của chúng ta có thể được gọi từ ngôn ngữ khác không hỗ trợ nạp chồng toán tử thì nên xây dựng các phương thức tương đương để thực hiện cùng chức năng nh ư: Add, Sub, Mul, Câuhỏi 2: Những điều lưu ý nào khi sử dụng nạp chồng toán tử trong một lớp? Trả l ời 2: Nói chung là khi nào thật cần thiết và ít gây ra sự nhầm lẫn. Ví dụ như ta xây dựng lớp Employee có nhiều thuộc tính số như lương, thâm niên, tuổi . Chúng ta muốn xây dựng toán tử ++ cho lương nhưng có thể làm nhầm lẫn với việc tăng số năm công tác, hay tăng tuổi. Do vậy việc sử dụng nạp chồng toán tử cũng phải cân nhắc tránh gây nhầm lẫn. Tốt nhất là sử dụng trong l ớp có ít thuộc tính số . Câuhỏi 3: Khi xây dựng toán tử so sánh thì có phải chỉ cần dùng toán tử so sánh bằng? Trả lời 3 : Đúng là nếu cần dùng toán tử so sánh nào thì chúng ta có thể chỉ tạo ra duy nhất toán tử so sánh đó mà thôi. Tuy nhiên, tốt hơn là chúng ta cũng nên xây dựng thêm toán tử so sánh khác như: so sánh khác, so sánh nhỏ hơn, so sánh lớn hơn .Việc này sẽ làm cho lớp của chúng ta hoàn thiện hơn. Câu hỏi thêm Câu hỏi 1 : Khi nào sử dụng toán tử chuyển đổi? Thế nào là chuyển đổi tường minh và chuyển đổi ngầm định? Câu hỏi 2 : Có thể tạo ra ký hiện toán tử riêng của ta và thực thi nạp chồng toán tử đó hay không? Câuhỏi 3: Có bao nhiêu toán tử mà .NET quy định? Ký hiệu của từng toán tử? Bài tập Bài tập 1 : Hãy tiếp tục phát triển lớp Fraction trong ví dụ của chương bằng cách thêm các toán tử khác như trừ, nhân, chia, so sánh . Bài tập 2: Xây dựng lớp điểm trong không gian hai chiều, với các toán tử cộng, trừ, nhân, chia. Bài tập3 : Tương tự như bài tập 2 nhưng điểm nằm trong không gian 3 chiều. Bài tập4 : Xây dựng lớp số phúc (số ảo) với các phép toán cộng, trừ, nhân, chia.