使用反射、工厂调用多个dll中的的派生类
假设有一个接口ISend。ISend负责发送数据到不同的客户端。通过ISend可以把数据发送到sockent客户端,也可能是邮件服务器,或者时消息队列,或者时数据库。关键是我们开发的时候不知道有多少个客户端。但是我们的程序必须动态的加载这些客户端然后把所有的消息发送出去。
Interface ISend
{
Send(IData data);
}
由于不知道有多少个客户端,我们想到把多个ISend的实现放在dll中实现。如SockentSend,MailSend,MSGSend。通过反射可以调用dll中的这些实现。如:
……
但是我们注意到我们在每个dll中的实现都必须通过名字空间和类名称来加载class。这就不符合我们希望不知道dll中实际的类名称,也能加载dll中的实现类的目的。我们有个简单的办法解决这个问题。就是在dll中实现一个简单工厂。通过这个工厂加载dll中的实现类。如
FactorySend
{
ISend GreateInstance();
}
在每个dll中都有一个FactorySend类,负责创建一个Send对象。
例如SockentSend.dll:
FactorySend
{
ISend GreateInstance()
{
Return new SockentSend();
}
}
我们在每个dll中有了一个统一的工厂FactorySend,通过反射调用统一工厂FactorySend加载不同ISend的实现类。
现在还有一个问题,我们系统怎么动态的知道应该加载dll呢?这里我们可以使用一个.net一个监视文件系统的类
FileSystemWatcher。程序运行时,监视exe的根目录,当有dll文件复制到该目录的时候,我们就加载这个dll。
public class Watcher
{
public static void Main()
{
Run();
}
[PermissionSet(SecurityAction.Demand, Name="FullTrust")]
public static void Run()
{
string[] args = System.Environment.GetCommandLineArgs();
// If a directory is not specified, exit program.
if(args.Length != 2)
{
// Display the proper way to call the program.
Console.WriteLine("Usage: Watcher.exe (directory)");
return;
}
// Create a new FileSystemWatcher and set its properties.
FileSystemWatcher watcher = new FileSystemWatcher();
watcher.Path = args[1];
/* Watch for changes in LastAccess and LastWrite times, and
the renaming of files or directories. */
watcher.NotifyFilter = NotifyFilters.LastAccess | NotifyFilters.LastWrite
| NotifyFilters.FileName | NotifyFilters.DirectoryName;
// Only watch text files.
watcher.Filter = "*.txt";
// Add event handlers.
watcher.Changed += new FileSystemEventHandler(OnChanged);
watcher.Created += new FileSystemEventHandler(OnChanged);
watcher.Deleted += new FileSystemEventHandler(OnChanged);
watcher.Renamed += new RenamedEventHandler(OnRenamed);
// Begin watching.
watcher.EnableRaisingEvents = true;
// Wait for the user to quit the program.
Console.WriteLine("Press \'q\' to quit the sample.");
while(Console.Read()!='q');
}
// Define the event handlers.
private static void OnChanged(object source, FileSystemEventArgs e)
{
// Specify what is done when a file is changed, created, or deleted.
Console.WriteLine("File: " +e.FullPath + " " + e.ChangeType);
}
private static void OnRenamed(object source, RenamedEventArgs e)
{
// Specify what is done when a file is renamed.
Console.WriteLine("File: {0} renamed to {1}", e.OldFullPath, e.FullPath);
}
}
在系统中应该有一个ISend 实现的列表List<ISend> list。当发送数据的时候,遍历这个list,依次发送数据到相应的客户端。
分享到:
相关推荐
二是在派生类中隐藏基类的成员,尽管隐藏的成员仍然存在,但不会被自动调用。 ### 10. this关键字的含义? `this`关键字代表当前对象的引用,可以用来访问类的成员,特别是在成员变量与局部变量同名时,用于区分。...
但 abstract 不能与 override 一起使用,因为 abstract 方法已经表明需要在派生类中重写,无需再次指定 override。 13. 接口(interface)可以包含方法、属性、索引器和事件,但不能包含字段或方法实现。接口支持多...
`abstract`和`virtual`可以同时用于声明一个方法,但`abstract`方法不能与`override`一起使用,因为`abstract`方法需要在派生类中重写,而`override`用于重写基类中的虚方法。 #### 15. 接口可包含的成员类型 接口...
而`overload`是指在同一类中使用相同方法名但参数列表不同的多个方法,这是方法重载,用于提供多种调用方法的灵活性。 ### 索引指示器的概念 索引指示器是一种特殊类型的属性,允许使用索引访问类的元素,就像数组...
- **自动编译**:当编辑器检测到脚本更改后,会自动触发编译过程,编译后的代码会被打包成一个或多个DLL文件。 - **动态加载**:使用Mono提供的API可以实现在程序运行过程中动态加载或卸载DLL,从而实现脚本的即时...
本篇内容主要涉及C#中的三个关键字:`is`、`as`和`typeof`,以及反射的基本概念和`System.Type`类。 1. **is运算符**:`is`运算符用于检测一个对象是否可以被转换为指定的类型。其语法结构是`expr is type`,返回值...
sealed 用于阻止派生类进一步继承,使类成为最终类。 7. override 与 overload: override 用于重写基类的虚方法,保持方法签名不变;overload(重载)是在同一作用域内创建多个同名但参数列表不同的方法。 8. ...
抽象方法没有实现,必须在派生类中重写。 5. **internal修饰符** - internal限制访问级别,仅限于同一程序集内的类访问,提供包级别的封装。 6. **sealed修饰符** - sealed用于密封类,防止其他类继承它。密封类...
- **`abstract`成员**:必须在派生类中重写,定义了类的行为模板。 #### 5. `internal`修饰符的作用 - 允许类型或成员在同一个程序集中被其他类型访问,但在不同程序集间无法访问。 #### 6. `sealed`修饰符的功能...
11. **抽象函数能否重写基类中的虚函数**:可以,派生类中的抽象函数可以重写基类中的虚函数,这是多态性的一种体现。 12. **密封类是否可以有虚函数**:可以,即使类被`sealed`修饰,仍然可以定义虚函数。但是,...
在派生类中使用时,可能导致多态性丢失。 10. `this`关键字:`this`引用当前对象实例,常用于在方法内部访问实例成员。 11. 抽象函数和虚函数:抽象函数(`abstract`)存在于抽象类中,无实现,子类必须重写;虚...
只有当基类加入了与派生类中现有的函数名称相同的函数时,才需要使用new操作符 - **解释**:`new`操作符用于隐藏基类中的成员。 - **实践建议**:谨慎使用`new`操作符,以免造成混淆。 #### 30. 尽量使用CLS-...
12. abstract和override:抽象方法不能同时声明为abstract和override,因为抽象方法在基类中未提供实现,而override表示在派生类中提供了实现。 13. 私有构造函数:当类只声明了私有构造函数时,不允许外部直接创建...
- **new**:隐藏基类中的方法或字段,可以在派生类中使用new关键字隐藏基类中同名的方法或字段。 - **2.5 值类型、引用类型的区别** - **值类型**:存储在栈中,直接包含数据。 - **引用类型**:存储在堆中,...