# Plugin.NET
**Repository Path**: hyjiacan/Plugin.NET
## Basic Information
- **Project Name**: Plugin.NET
- **Description**: c#插件管理器
- **Primary Language**: C#
- **License**: MIT
- **Default Branch**: master
- **Homepage**: None
- **GVP Project**: No
## Statistics
- **Stars**: 116
- **Forks**: 65
- **Created**: 2016-12-31
- **Last Updated**: 2025-06-17
## Categories & Tags
**Categories**: utils
**Tags**: None
## README
# Plugin.NET
Plugin.NET 目标是搞一个灵活的c#插件管理器(后面统称为**管理器**)。管理器使用反射(Reflect)技术,动态地加载dll类库(即插件)。在使用管理器的时候,会要求将一个`abstract`的类作为泛型参数传入,插件需要实现这个类的接口,而应用程序通过这个类里面的接口来调用插件。
> 项目目前都是下班回家后空闲时间在搞,所以进度有点慢。欢迎朋友们一起PR。 :relieved: :relieved: :relieved:
## 运行环境
项目使用 .net 4.0 编写,自己随便改改代码就能用到.net2.0和.netcore上。
## 接口定义规范
说是接口,准确来说,应该是一个虚类,即由`abstract`修饰的类。这个类里面,包含插件必须实现和可选实现的两类接口(方法)。其中,必须实现的接口使用`abstract`修饰,可选实现的方法使用`virtual`修饰。示例如下:
```csharp
///
/// 插件接口
///
public abstract class PluginInterface
{
///
/// 插件名称
///
public abstract string Name { get; }
///
/// 插件版本
///
public abstract Version Version { get; }
///
/// 插件描述
///
public virtual string Description { get; }
///
/// 加载插件后调用,使用了 abstract 修饰,所以必须实现
///
public abstract void Load();
///
/// 第一个功能方法,使用了 virtual 修饰,所以可选实现
///
///
public virtual string Method1()
{
return null;
}
///
/// 第二个功能方法,使用了 virtual 修饰,所以可选实现
///
///
///
public virtual string Method2(string from)
{
return from;
}
}
```
> 这里使用`virtual`来作为可选修饰是为了方便程序的接口升级,否则每接口有变化,插件都必须进行修改。
## 插件实现规范
插件所在项目需要引用接口所在的dll文件,在发布插件时不需要发布这个接口dll,但是如果有其它依赖的dll,需要一起发布。管理器会查找插件的类继承,插件中继承接口的类会通过无参数构造创建实例,然后交给程序使用,所以在实现插件时,必须提供一下无参的构造。
> 需要注意的是,管理器只会使用在插件中找到的第一个实现了接口的类,**其它的类将被忽略**。
## 管理器使用流程
1. 编写程序的接口类,在入口项目中引用这个接口
2. 在程序中引用**Plugin.NET.dll**
3. 初始化插件管理器
4. 绑定插件管理器的事件,各种事件提供了丰富的插件加载数据
5. 调用 `Load` 方法加载已经存在的所有插件,这个方法可以传入一个过滤器函数
6. 如果希望插件可以热加载,那么再调用 `Watch` 方法,以监视插件目录是否有新的插件放进去
7. 如果要停止热加载,那么就调用 `StopWatch` 以停止监视插件目录
## 示例
```csharp
class Program
{
static void Main(string[] args)
{
// 使用接口来实例化插件管理器
// 如果要对其它接口进行插件管理,
// 那么可以创建另一个插件管理器的实例
var pluginManager = new PluginManager();
// 处理插件管理器发出的事件
pluginManager.OnAssemblyLoading += PluginManager_OnAssemblyLoading;
pluginManager.OnAssemblyLoaded += PluginManager_OnAssemblyLoaded;
pluginManager.OnError += PluginManager_OnError;
pluginManager.OnInstanceCreating += PluginManager_OnInstanceCreating;
pluginManager.OnInstanceCreated += PluginManager_OnInstanceCreated;
// 加载插件目录下的所有插件
pluginManager.Load();
// 开始监视新放进目录的插件
pluginManager.Watch();
Console.WriteLine("正在监视插件目录变动,按`Enter`退出");
Console.ReadLine();
}
private static void PluginManager_OnAssemblyLoading(object sender, PluginAssemblyLoadingArgs e)
{
Console.WriteLine($"准备从文件\"{e.FileName}\"加载程序集");
}
private static void PluginManager_OnAssemblyLoaded(object sender, PluginAssemblyLoadedArgs e)
{
Console.WriteLine($"从文件\"{e.FileName}\"加载程序集\"{e.Assembly}\"成功");
}
private static void PluginManager_OnInstanceCreating(object sender, PluginInstanceCreatingArgs e)
{
Console.WriteLine($"准备从程序集\"{e.Assembly}\"加载类型\"{e.Class}\"");
}
private static void PluginManager_OnInstanceCreated(object sender, PluginInstanceCreatedArgs e)
{
var plugin = e.Instance;
Console.WriteLine($"从程序集\"{e.Assembly}\"加载类型\"{e.Class}\"成功");
Console.WriteLine($"插件名称: {plugin.Name}, 版本: {plugin.Version}");
e.Instance.Load();
Console.WriteLine("Method1:" + plugin.Method1());
Console.WriteLine("Method2:" + plugin.Method2("啊呀哟"));
}
private static void PluginManager_OnError(object sender, PluginErrorEventArgs e)
{
switch (e.ErrorType)
{
case PluginNET.error.PluginErrorTypes.None:
break;
case PluginNET.error.PluginErrorTypes.InvalidDll:
Console.WriteLine($"文件\"{e.FileName}\"不是有效有dll: {e.Exception}");
break;
case PluginNET.error.PluginErrorTypes.CannotLoadClassTypes:
Console.WriteLine($"无法从文件\"{e.FileName}\"中加载类型: {e.Exception}");
break;
case PluginNET.error.PluginErrorTypes.ImplementionClassNotFound:
Console.WriteLine($"在文件\"{e.FileName}\"中没有找到实现了指定接口的类");
break;
case PluginNET.error.PluginErrorTypes.IllegalClassDefinition:
Console.WriteLine($"在文件\"{e.FileName}\"中找到了实现指定接口的类,但是其声明不是class或声明为abstract或不是public");
break;
case PluginNET.error.PluginErrorTypes.DefaultConstructorNotFound:
Console.WriteLine($"在文件\"{e.FileName}\"中找到了实现指定接口的类,但未找到默认的无参构造函数");
break;
case PluginNET.error.PluginErrorTypes.InstanceCreateFailed:
Console.WriteLine($"在文件\"{e.FileName}\"中找到了实现指定接口的类\"{e.ClassType}\",但是创建实例时抛出了异常:{e.Exception}");
break;
case PluginNET.error.PluginErrorTypes.Unkown:
Console.WriteLine("未知错误");
break;
default:
break;
}
}
}
```
示例请看解决方案中的*test*目录,测试项目为*Plugin.NETTest*