intTypePromotion=1
zunia.vn Tuyển sinh 2024 dành cho Gen-Z zunia.vn zunia.vn
ADSENSE

Các giải pháp lập trình CSharp- P11

Chia sẻ: Cong Thanh | Ngày: | Loại File: PDF | Số trang:10

131
lượt xem
28
download
 
  Download Vui lòng tải xuống để xem tài liệu đầy đủ

Các giải pháp lập trình CSharp- P11: Các giải pháp lập trình C# khảo sát chiều rộng của thư viện lớp .NET Framework và cung cấp giải pháp cụ thể cho các vấn đề thường gặp. Mỗi giải pháp được trình bày theo dạng “vấn đề/giải pháp” một cách ngắn gọn và kèm theo là các ví dụ mẫu.

Chủ đề:
Lưu

Nội dung Text: Các giải pháp lập trình CSharp- P11

  1. 101 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu // Tạo một ListDictionary chứa các plug-in. ListDictionary pluginList = new ListDictionary(); pluginList["SimplePlugin"] = "CreateInstanceExample"; // Tạo một PluginManager trong miền ứng dụng mới // và chỉ định danh sách các plug-in. PluginManager manager2 = (PluginManager)domain1.CreateInstanceAndUnwrap( "CreateInstanceExample", "PluginManager", true, 0, null, new object[] {pluginList}, null, null, null); // Hiển thị các plug-in đã được nạp vào NewAppDomain2. Console.WriteLine("Plugins in NewAppDomain2:"); foreach (string s in manager2.GetPluginList()) { Console.WriteLine(" - " + s); } // Nhấn Enter để thoát. Console.ReadLine(); } } 8. Truyền dữ liệu giữa các miền ứng dụng  Bạn cần một cơ chế đơn giản để truyền dữ liệu trạng thái hay cấu hình giữa các miền ứng dụng.  Dùng các phương thức SetData và GetData của lớp AppDomain. Dữ liệu có thể được truyền qua các miền ứng dụng như đối số hay trị trả về khi bạn cho gọi các thành viên của các đối tượng hiện có trong các miền ứng dụng. Việc truyền dữ liệu qua các miền ứng dụng được thực hiện dễ dàng giống như truyền dữ liệu trong cùng một miền ứng dụng. Mọi miền ứng dụng đều duy trì một vung đệm dữ liệu (data cache) chứa một tập các cặp “tên/giá trị”. Hầu hết nội dung của vung đệm dữ liệu phản ánh các thiết lập cấu hình của miền ứng dụng, như các giá trị từ đối tượng AppDomainSetup được cung cấp trong quá trình tạo miền ứng dụng (xem mục 3.1). Vung đệm dữ liệu này có thể được sử dụng để trao đổi dữ liệu giữa các miền ứng dụng hay lưu trữ các giá trị tạm thời dùng trong cùng một miền ứng dụng. Phương thức SetData thực hiện việc kết hợp một khóa dạng chuỗi với một đối tượng và lưu trữ nó vào vùng đệm dữ liệu của miền ứng dụng. Phương thức GetData thực hiện công việc ngược lại là lấy lại đối tượng từ vùng đệm dữ liệu thông qua khóa. Nếu mã lệnh trong một
  2. 102 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu miền ứng dụng gọi phương thức SetData hay GetData để truy xuất vung đệm dữ liệu của miền ứng dụng khác, thì đối tượng dữ liệu phải hỗ trợ ngữ nghĩa marshal-by-value hay marshal-by- reference, nếu không thì ngoại lệ System.Runtime.Serialization.SerializationException sẽ bị ném (xem mục 3.3 để biết thêm chi tiết về cách truyền đối tượng qua các miền ứng dụng). Đoạn mã sau trình bày cách sử dụng phương thức SetData và GetData để truyền một System.Collections.ArrayList giữa hai miền ứng dụng. using System; using System.Reflection; using System.Collections; public class ListModifier { public ListModifier () { // Nhận danh sách từ đệm dữ liệu. ArrayList list = (ArrayList)AppDomain.CurrentDomain.GetData("People"); // Thay đổi danh sách. list.Add("Tam"); } } public class PassDataExample { public static void Main() { // Tạo một miền ứng dụng mới. AppDomain domain = AppDomain.CreateDomain("Test"); // Tạo một ArrayList và thêm thông tin vào. ArrayList list = new ArrayList(); list.Add("Phuong"); list.Add("Phong"); list.Add("Nam"); // Đặt danh sách vào vùng đệm dữ liệu của miền ứng dụng mới. domain.SetData("People", list);
  3. 103 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu // Tạo một ListModifier trong miền ứng dụng mới // sẽ thay đổi nội dung của list trong vùng đệm dữ liệu. domain.CreateInstance("03-08", "ListModifier"); // Nhận lại danh sách và hiển thị nội dung của nó. foreach (string s in (ArrayList)domain.GetData("People")) { Console.WriteLine(s); } // Nhấn Enter để thoát. Console.ReadLine(); } } 9. Giải phóng assembly và miền ứng dụng  Bạn cần giải phóng các assembly hay các miền ứng dụng lúc thực thi.  Không có cách nào để giải phóng các assembly riêng lẻ. Bạn có thể giải phóng toàn bộ một miền ứng dụng bằng phương thức tĩnh AppDomain.Unload, đồng thời với việc giải phóng miền ứng dụng là tất cả các assembly đã được nạp vào miền ứng dụng đó cũng được giải phóng. Cách duy nhất để giải phóng một assembly là giải phóng cả miền ứng dụng mà nó đã được nạp vào. Đáng tiếc, việc giải phóng một miền ứng dụng cũng sẽ giải phóng luôn tất cả các assembly đã được nạp vào đó. Đây là một giới hạn yêu cầu bạn phải tổ chức và quản lý tốt cấu trúc miền ứng dụng và assembly. Khi giải phóng một miền ứng dụng bằng phương thức tĩnh AppDomain.Unload, bạn cần truyền cho nó một tham chiếu AppDomain đến miền ứng dụng cần giải phóng. Bạn không thể giải phóng miền ứng dụng mặc định do CLR tạo lúc startup. Đoạn mã dưới đây trình bày cách sử dụng phương thức Unload. // Tạo một miền ứng dụng mới. AppDomain newDomain = AppDomain.CreateDomain("New Domain"); // Nạp assembly vào miền ứng dụng mày. § // Giải phóng miền ứng dụng. AppDomain.Unload(newDomain); Phương thức Unload chặn các tiểu trình mới đi vào miền ứng dụng được chỉ định và gọi phương thức Thread.Abort trên tất cả các tiểu trình hiện đang chạy trong miền ứng dụng này. Nếu tiểu trình gọi phương thức Unload hiện đang chạy trong miền ứng dụng cần giải phóng thì một tiểu trình khác sẽ được khởi chạy để thực hiện quá trình giải phóng. Nếu có vấn đề trong
  4. 104 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu việc giải phóng miền ứng dụng, ngoại lệ System.CannotUnloadAppDomainException sẽ bị ném bởi tiểu trình thực hiện quá trình giải phóng. Trong khi miền ứng dụng đang được giải phóng, CLR gọi thực thi các phương thức giải phóng của tất cả các đối tượng trong miền ứng dụng. Tùy thuộc vào số lượng đối tượng và bản chất của các phương thức giải phóng mà quá trình này có thể mất một khoảng thời gian nào đó. Phương thức AppDomain.IsFinalizingForUnload trả về true nếu miền ứng dụng đang được giải phóng và CLR đã bắt đầu giải phóng các đối tượng trong đó; ngược lại, trả về false. 10. Truy xuất thông tin Type  Bạn muốn thu lấy đối tượng System.Type mô tả một kiểu dữ liệu nhất định.  Sử dụng một trong các cách sau: • Toán tử typeof • Phương thức tĩnh GetType của lớp System.Type • Phương thức GetType thuộc một thể hiện của kiểu • Phương thức GetNestedType hay GetNestedTypes của lớp Type • Phương thức GetType hay GetTypes của lớp Assembly • Phương thức GetType, GetTypes, hay FindTypes của lớp System.Reflection. Module Đối tượng Type cung cấp một điểm khởi đầu để làm việc với các kiểu dữ liệu bằng cơ chế phản chiếu. Một đối tượng Type cho phép bạn kiểm tra siêu dữ liệu của kiểu, thu lấy các thành viên của kiểu, và tạo các đối tượng của kiểu. Do tầm quan trọng của nó, .NET Framework cung cấp nhiều cơ chế để lấy tham chiếu đến các đối tượng Type. Phương pháp hiệu quả nhất để thu lấy đối tượng Type cho một kiểu cụ thể là sử dụng toán tử typeof: System.Type t1 = typeof(System.Text.StringBuilder); Tên kiểu không được đặt trong dấu nháy kép và phai kha phân giai đôi với trình biên dịch. Vì tham chiếu được phân giải lúc biên dịch nên assembly chứa kiểu này trở thành phần phụ thuộc tĩnh cua assembly và sẽ được liệt kê như thế trong assembly manifest của bạn. Một cách khác là sử dụng phương thức tĩnh Type.GetType, nhận vào một chuỗi chứa tên kiểu. Vì sử dụng chuỗi để chỉ định kiểu nên bạn có thể thay đổi nó lúc thực thi, điều này mở ra cánh cửa đến với thế giới lập trình động bằng cơ chế phản chiếu (xem mục 3.12). Nếu bạn chỉ định tên kiểu, bộ thực thi phải tìm kiểu này trong một assembly đã được nạp. Bạn cũng có thể chỉ định một tên kiểu theo tiêu chuân assembly (tham khảo tài liệu .NET Framework SDK về phương thức Type.GetType để biết cách kết cấu tên kiểu theo tiêu chuân assembly). Các lệnh sau trình bày cách sử dụng phương thức GetType: // Có phân biệt chữ hoa-thường, trả về null nếu không tìm thấy. Type t2 = Type.GetType("System.String");
  5. 105 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu // Có phân biệt chữ hoa-thường, // ném ngoại lệ TypeLoadException nếu không tìm thấy. Type t3 = Type.GetType("System.String", true); // Không phân biệt chữ hoa-thường, // ném ngoại lệ TypeLoadException nếu không tìm thấy. Type t4 = Type.GetType("system.string", true, true); // Tên kiểu theo tiêu chuân assembly. Type t5 = Type.GetType("System.Data.DataSet,System.Data," + "Version=1.0.5000.0,Culture=neutral, PublicKeyToken=b77a5c561934e089"); Để thu lấy đối tượng Type mô tả kiểu của một đối tượng hiện có, hãy sử dụng phương thức GetType, được hiện thực bởi Object và được thừa kế bởi tất cả các kiểu dữ liệu. Dưới đây là một ví dụ: System.Text.StringBuilder sb = new System.Text.StringBuilder(); Type t6 = sb.GetType(); Bảng 3.2 tóm tắt các phương thức khác cũng cung cấp khả năng truy xuất đối tượng Type. Bảng 3.2 Các phương thức trả về đối tượng Type Phương thức Mô tả Lấy đối tượng Type mô tả một kiểu lồng bên trong đối tượng Type.GetNestedType Type hiện có Lấy một mảng các đối tượng Type mô tả các kiểu lồng bên trong Type.GetNestedTypes đối tượng Type hiện có Lấy đối tượng Type mô tả một kiểu được khai báo bên trong Assembly.GetType assembly Lấy một mảng các đối tượng Type mô tả các kiểu được khai báo Assembly.GetTypes bên trong assembly Lấy đối tượng Type mô tả một kiểu được khai báo bên trong Module.GetType module Lấy một mảng các đối tượng Type mô tả các kiểu được khai báo Module.GetTypes bên trong module Lấy một mảng đã được lọc, chứa các đối tượng Type mô tả các kiểu được khai báo bên trong module—các kiểu này được lọc Module.FindTypes bằng một delegate (xác định xem mỗi Type có xuất hiện trong mảng đích hay không)
  6. 106 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu 11. Kiểm tra kiểu của một đối tượng  Bạn muốn kiểm tra kiểu của một đối tượng.  Sử dụng phương thức thừa kế Object.GetType để thu lấy Type cho đối tượng này. Trong vài trường hợp, bạn cũng có thể sử dụng toán tử is và as để kiểm tra kiểu của một đối tượng. Tất cả các kiểu dữ liệu đều thừa kế phương thức GetType từ lớp cơ sở Object. Như đã được thảo luận trong mục 3.10, phương thức này trả về một tham chiếu Type mô tả kiểu của đối tượng. Bộ thực thi duy trì một đối tượng Type cho mỗi kiểu được nạp và tất cả các tham chiếu cho kiểu này cùng chỉ đến đối tượng này. Điều này nghĩa là bạn có thể so sánh hai tham chiếu kiểu một cách hiệu quả. Ví dụ dưới đây trình bày cách kiểm tra một đối tượng có phải là System.IO.StringReader hay không: // Tạo một StringReader để thử nghiệm. Object someObject = new StringReader("This is a StringReader"); // Kiểm tra xem someObject có phải là một StringReader hay không // bằng cách thu lấy và so sánh tham chiếu Type (sử dụng toán tử typeof). if (typeof(System.IO.StringReader) == someObject.GetType()) { // Làm gì đó. § } C# cung cấp toán tử is để thực hiện nhanh việc kiểm tra như trên. Ngoài ra, is sẽ trả về true nếu đối tượng cần kiểm tra dẫn xuất từ lớp được chỉ định. Đoạn mã dưới đây kiểm tra xem someObject là một thể hiện của System.IO.TextReader, hay một lớp dẫn xuất từ TextReader (như StringReader): // Kiểm tra xem someObject là TextReader, // hay dẫn xuất từ TextReader bằng toán tử is. if (someObject is System.IO.TextReader) { // Làm gì đó. § } Cả hai cách này đều đòi hỏi kiểu dùng với toán tử typeof và is phải là kiểu đã biết và khả phân giải lúc biên dịch. Một cách khác linh hoạt hơn (nhưng chậm hơn) là sử dụng phương thức Type.GetType để trả về một tham chiếu Type cho kiểu được chỉ định. Tham chiếu Type không được phân giải cho đến khi thực thi, việc này ảnh hưởng đến hiệu năng, nhưng cho phép bạn thay đổi phép so sánh kiểu lúc thực thi dựa trên giá trị của một chuỗi. Phương thức IsType dưới đây sẽ trả về true nếu đối tượng thuộc kiểu được chỉ định và sử dụng phương
  7. 107 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu thức Type.IsSubclassOf để kiểm tra đối tượng này có phải là một lớp con của kiểu được chỉ định hay không. public static bool IsType(object obj, string type) { Type t = Type.GetType(type, true, true); return t == obj.GetType() || obj.GetType().IsSubclassOf(t); } Cuối cùng, bạn có thể sử dụng toán tử as để ép bất kỳ đối tượng nào sang kiểu được chỉ định. Nếu đối tượng không thể bị ép sang kiểu được chỉ định, toán tử as sẽ trả về null. Điều này cho phép bạn thực hiện các phép ép kiểu an toàn (safe cast), nhưng kiểu được so sánh phải là khả phân giải lúc thực thi. Dưới đây là một ví dụ: // Sử dụng toán tử as để thực hiện một phép ép kiểu an toàn. StringReader reader = someObject as System.IO.StringReader; if (reader != null) { // Làm gì đó. § }  Phương thức tĩnh GetUnderlyingType của lớp System.Enum cho phép bạn thu lấy kiểu thưc sư của một kiểu liệt kê. 12. Tạo một đối tượng bằng cơ chế phản chiếu  Bạn cần tạo một đối tượng bằng cơ chế phản chiếu.  Thu lấy đối tượng Type mô tả kiểu của đối tượng cần tạo, gọi phương thức GetConstructor của nó để có được đối tượng System.Reflection.ConstructorInfo mô tả phương thức khởi dựng cần dùng, sau đó thực thi phương thức ConstructorInfo.Invoke. Bước đầu tiên trong việc tạo một đối tượng bằng cơ chế phản chiếu là thu lấy đối tượng Type mô tả kiểu của đối tượng cần tạo (xem mục 3.10 để biết thêm chi tiết). Khi có được đối tượng Type, hãy gọi phương thức GetConstructor để thu lấy đối tượng ConstructorInfo mô tả một trong các phương thức khởi dựng của kiểu này. Dạng thức thông dụng nhất của phương thức GetConstructor là nhận một mảng Type làm đối số, và trả về đối tượng ConstructorInfo mô tả phương thức khởi dựng nhận các đối số được chỉ định trong mảng Type. Để thu lấy đối tượng ConstructorInfo mô tả phương thức khởi dựng mặc định (không có đối số), bạn hãy truyền cho phương thức GetConstructor một mảng Type rỗng (sử dụng trường tĩnh Type.EmptyTypes); đừng sử dụng null, nếu không GetConstructor sẽ ném ngoại lệ System.ArgumentNullException. Nếu GetConstructor không tìm thấy phương thức khởi dựng nào có chữ ký phù hợp với các đối số được chỉ định thì nó sẽ trả về null.
  8. 108 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu Môt khi đã có đối tượng ConstructorInfo như mong muốn, hãy gọi phương thức Invoke của nó. Bạn phải cung cấp một mảng chứa các đối số mà bạn muốn truyền cho phương thức khởi dựng. Phương thức Invoke sẽ tạo đối tượng mới và trả về một tham chiếu đến đối tượng đó (bạn phải ép về kiểu phù hợp). Dưới đây là đoạn mã trình bày cách tạo một đối tượng System.Text.StringBuilder, chỉ định nội dung ban đầu cho StringBuilder và sức chứa của nó. // Thu lấy đối tượng Type cho lớp StringBuilder. Type type = typeof(System.Text.StringBuilder); // Tạo Type[] chứa các đối tượng Type cho mỗi đối số // của phương thức khởi dựng (một chuỗi và một số nguyên). Type[] argTypes = new Type[] {typeof(System.String), typeof(System.Int32)}; // Thu lấy đối tượng ConstructorInfo. ConstructorInfo cInfo = type.GetConstructor(argTypes); // Tạo object[] chứa các đối số cho phương thức khởi dựng. object[] argVals = new object[] {"Some string", 30}; // Tạo đối tượng và ép nó về kiểu StringBuilder. StringBuilder sb = (StringBuilder)cInfo.Invoke(argVals); Chức năng phản chiếu thường được sử dụng để hiện thực các factory. Trong đó, bạn sử dụng cơ chế phản chiếu để thể hiện hóa các lớp thừa kế một lớp cơ sở phổ biến hay hiện thực một giao diện phổ biến. Thông thường, cả lớp cơ sở chung và giao diện chung đều được sử dụng. Lớp cơ sở trừu tượng sẽ hiện thực giao diện và bất kỳ chức năng chung nào, sau đó mỗi hiện thực cụ thê sẽ thừa kế lớp cơ sở. Không có cơ chế nào để khai báo rằng mỗi lớp cụ thê phải hiện thực các phương thức khởi dựng với các chữ ký cụ thể. Nếu muốn người khác (hãng thứ ba) hiện thực các lớp cụ thê thì bạn phải chỉ rõ (trong tài liệu hướng dẫn) chữ ký của phương thức khởi dựng mà factory của bạn gọi. Cách thông thường để tránh vấn đề này là sử dụng phương thức khởi dựng mặc định (không có đối số), sau đó cấu hình đối tượng bằng phương thức và thuộc tính. Ví dụ dưới đây sẽ hiện thực một factory dùng để tạo các đối tượng có hiện thực giao diện IPlugin. using System; using System.Reflection; // Giao diện chung mà tất cả các plug-in phải hiện thực. public interface IPlugin { string Description { get; set; }
  9. 109 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu void Start(); void Stop(); } // Lớp cơ sở trừu tượng mà tất cả các plug-in phải dẫn xuất từ đây. public abstract class AbstractPlugin : IPlugin { // Chuỗi chứa lời mô tả plug-in. private string description = ""; // Thuộc tính dùng để lấy lời mô tả plug-in. public string Description { get { return description; } set { description = value; } } // Khai báo các thành viên của giao diện IPlugin. public abstract void Start(); public abstract void Stop(); } // Một hiện thực đơn giản cho giao diện IPlugin // để minh họa lớp PluginFactory. public class SimplePlugin : AbstractPlugin { // Hiện thực phương thức Start. public override void Start() { Console.WriteLine(Description + ": Starting..."); } // Hiện thực phương thức Stop. public override void Stop() { Console.WriteLine(Description + ": Stopping..."); } } // Factory dùng để tạo các đối tượng của IPlugin. public sealed class PluginFactory { public static IPlugin CreatePlugin(string assembly,
  10. 110 Chương 3: Miền ứng dụng, cơ chế phản chiếu, và siêu dữ liệu string pluginName, string description) { // Thu lấy đối tượng Type cho plug-in được chỉ định. Type type = Type.GetType(pluginName + ", " + assembly); // Thu lấy đối tượng ConstructorInfo. ConstructorInfo cInfo = type.GetConstructor(Type.EmptyTypes); // Tạo đối tượng và ép nó về kiểu StringBuilder. IPlugin plugin = (IPlugin)cInfo.Invoke(null); // Cấu hình IPlugin mới. plugin.Description = description; return plugin; } } Câu lệnh sau đây sẽ tạo đối tượng SimplePlugin bằng lớp PluginFactory: IPlugin plugin = PluginFactory.CreatePlugin( "CreateObjectExample", // Tên assembly "SimplePlugin", // Tên lớp plug-in "A Simple Plugin" // Lời mô tả plug-in );  Lớp System.Activator cung cấp hai phương thức tĩnh CreateInstance và CreateInstanceFrom dùng để tạo các đối tượng dựa trên đối tượng Type hay chuỗi chứa tên kiểu. Xem tài liệu .NET Framework SDK để biết thêm chi tiết. 13. Tạo một đặc tính tùy biến  Bạn cần tạo ra một đặc tính theo ý bạn.  Tạo một lớp dẫn xuất từ lớp cơ sở trừu tượng System.Attribute. Hiện thực các phương thức khởi dựng, các trường, và các thuộc tính để cho phép người dùng cấu hình đặc tính. Sử dụng System.AttributeUsageAttribute để định nghĩa: • Những phần tử chương trình nào là đích của đặc tính • Bạn có thể áp dụng nhiều thể hiện của đặc tính cho một phần tử chương trình hay không
ADSENSE

CÓ THỂ BẠN MUỐN DOWNLOAD

 

Đồng bộ tài khoản
2=>2