- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
在 Node.js 中,我能够相当轻松地制作 WordPress 克隆,使用 EventEmitter 复制并在 CMS 核心中构建一个钩子(Hook)系统,然后插件可以附加到该核心。
我现在需要为我的 CMS 编写并移植到 Go 的相同级别的可扩展性和核心隔离。基本上我现在已经完成了核心,但为了使其真正灵活,我必须能够插入事件(钩子(Hook))并让插件附加到这些钩子(Hook)上并具有附加功能。
我不关心重新编译(动态/静态链接),只要您不必修改核心来加载插件 - CMS 核心永远不应该被修改。 (如 WP、Drupal 等)
我注意到有一些相当未知的项目,试图在 Go 中实现事件,看起来有点类似于 Node.js 中的 EventEmitter:
https://github.com/CHH/eventemitter
https://github.com/chuckpreslar/emission
由于上述两个项目并没有获得太多的知名度和关注度,我觉得这种思考事件的方式现在可能是我们在 Go 中应该这样做的方式?这是否意味着 Go 可能不适合这项任务?通过插件制作真正可扩展的应用程序?
Go 似乎没有在其核心中内置事件,并且 RPC 似乎不是将插件集成到核心应用程序中的有效解决方案,因为它们是原生内置的,并且好像它们是主应用程序的一部分自己。
将插件无缝集成到您的核心应用程序中的最佳方式是什么,以获得无限的扩展点(在核心中),而无需在每次需要连接新插件时操作核心?
最佳答案
一般来说,在 Go 中,如果您需要事件,您可能需要使用 channel ,但如果您需要插件,则可以采用的方法是接口(interface)。这是一个简单的插件架构的一个有点长的例子,它可以最小化需要的代码写在应用程序的主文件中以添加插件(这可以是自动化的,但不是动态的,见下文)。
我希望它是你正在寻找的方向。
1.插件接口(interface)
好吧,假设我们有两个插件,Fooer 和 Doer。我们首先定义它们的接口(interface):
// All DoerPlugins can do something when you call that method
type DoerPlugin interface {
DoSomething()
}
// All FooerPlugins can Foo() when you want them too
type FooerPlugin interface {
Foo()
}
<强>2。插件注册表
现在,我们的核心应用有一个插件注册表。我在这里做一些快速而肮脏的事情,只是为了让这个想法得到理解:
package plugin_registry
// These are are registered fooers
var Fooers = []FooerPlugin{}
// Thes are our registered doers
var Doers = []DoerPlugin{}
现在我们公开将插件添加到注册表的方法。简单的方法是为每种类型添加一个,但您可以使用更复杂的反射 Material 并具有一个功能。但通常在 Go 中,尽量保持简单:)
package plugin_registry
// Register a FooerPlugin
func RegisterFooer(f FooerPlugin) {
Fooers = append(Fooers, f)
}
// Register a DoerPlugin
func RegisterDoer(d DoerPlugin) {
Doers = append(Doers, d)
}
<强>3。实现和注册插件
现在,假设这是您的插件模块。我们创建了一个插件,它是一个执行者,并且在我们的包中init()
方法我们注册它。 init() 在每个导入包的程序启动时发生一次。
package myplugin
import (
"github.com/myframework/plugin_registry"
)
type MyPlugin struct {
//whatever
}
func (m *MyPlugin)DoSomething() {
fmt.Println("Doing something!")
}
再次,这里是自动注册包的“init magic”
func init() {
my := &MyPlugin{}
plugin_registry.RegisterDoer(my)
}
4.导入插件会自动注册它们
现在,我们唯一需要更改的是我们导入到主包中的内容。自从Go 没有动态导入或链接,这是您唯一需要编写的东西。创建一个将生成主文件的 go generate
脚本非常简单通过查看文件树或配置文件并找到您需要导入的所有插件。它不是动态的,但可以自动化。因为main导入插件是为了注册的副作用,所以导入uses the blank identifier to avoid unused import error .
package main
import (
"github.com/myframework/plugin_registry"
_ "github.com/d00dzzzzz/myplugin" //importing this will automaticall register the plugin
)
5.在应用的核心中
现在我们的核心应用无需更改任何代码即可与插件交互:
func main() {
for _, d := range plugin_registry.Doers {
d.DoSomething()
}
for _, f := range plugin_registry.Fooers {
f.Foo()
}
}
就是这样。请记住,插件注册表应该是一个单独的包应用程序的核心和插件都可以导入,因此您不会有循环导入。
当然,您可以在此组合中添加事件处理程序,但正如我所展示的,它不是必需的。
关于Golang 事件 : EventEmitter/dispatcher for plugin architecture,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28001872/
我的密码 https://gist.github.com/ButuzGOL/707d1605f63eef55e4af 因此,当我收到登录成功回调时,我想进行重定向, 重定向也可以通过调度程序进行。 我
我已经写了访问者模式如下,但我不明白什么是单次和双次分派(dispatch)。AFAIK,单分派(dispatch)是根据调用者类型调用方法,而双分派(dispatch)是根据调用者类型和参数类型调用
我有一个非 ui 线程,我需要在该线程上发送消息。 执行此操作的正常方法是在我的线程的线程过程中调用 Dispatcher.Run()。 我想修改它以使其在处理未处理的异常方面更加健壮。 我的第一个剪
我有一个具有这样功能的代码 const mapDispatchToProps = (dispatch: Dispatch) => ({ onAddProduct: ( key: str
我在使用 Window.Show 显示 WPF 窗口时遇到问题: System.InvalidOperationException was unhandled Message: An unhandle
我对何时使用 Dispatcher.Invoke 从不同线程更新 UI 上的某些内容存有疑问。 这是我的代码... public Window4() { InitializeC
我遇到了一个我无法解决的问题。我正在构建一个电子商务 react 应用程序并使用 useReducer 和 useContext 进行状态管理。客户打开产品,挑选商品数量,然后单击“添加到购物车”按钮
尽管我已经深入了解了 NEventStore 上的事务完整性,但我无法理解在连接了许多 NEventStore 实例时 NEventStore 将如何真正扩展。 总结一下我的理解,一个事件被添加到提交
我学习了 React Javascript 和 Redux,现在我遇到了这个问题。 这是一个 codesandbox 像这样尝试: 搜索书名“dep” 观察日志显示“Search url is:”,当
Dispatcher.CurrentDispatcher(在System.Windows.Threading中)和Application.Current.Dispatcher(在 >系统.Window
我得到了一些代码来处理调度程序在其构造函数中传递给 View 模型的位置。我现在想知道当我想要在 UI 线程上执行某些操作时,我是否应该使用 ObserveOn(dispatcher) 或 dispa
当我们的一个应用程序服务器内存不足时,我正在分析 Java 堆转储。我正在使用 Eclipse 内存分析器。它报告了以下内容。 One instance of "akka.dispatch.Dispa
哪一个: public static let barrier: DispatchWorkItemFlags public static let detached: DispatchWorkItem
我想使用不同于调度类型的类型提示 Action 创建者。 我已经尝试使用这两种类型对 ThunkResult 进行类型提示,但这并不理想。 // types.ts interface AppListA
我正在尝试准确地理解什么是单次分派(dispatch)和多次分派(dispatch)。 我刚刚读到这个: http://en.wikipedia.org/wiki/Multiple_dispatch
I have following api returning Flux of String我有以下返回字符串通量的接口 @GetMapping(value = "/api/getS
这是我自学前端开发一年后在Stackoverflow上的第一个问题。我已经找到了我的疑惑的答案,但由于这些问题是第三次返回,我认为是时候向 Web 提问了。 我正在尝试构建什么 我正在尝试构建一个图书
我正在使用 Kotlin 学习 Android,并且我了解到在不阻塞主线程的情况下启动协程的推荐方法是执行以下操作 MainScope().launch { withContext(Dispatc
错误本身: (alias) deleteCategory(id: number): (dispatch: Dispatch) => void import deleteCategory Argumen
我必须对抽屉进行裁剪,然后创建一个包含所有需要项的DrawerComponent,并创建一个带有NavigationActions的函数来调度我的路线,但是它不起作用。当我单击任何项目时,我都会遇
我是一名优秀的程序员,十分优秀!