- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我对 OSGi 世界有些陌生。有些概念仍然让我难以理解。
我正在尝试使用 Swing、Equinox 和 Declarative Services 创建图形化 OSGi 应用程序。目标是简化应用程序插件和扩展的创建。
我遇到了一个设计问题,由于我是从头开始做的,所以我想尽我所能使用所有最佳实践。
我确实有一个包含 API 的包,并且只公开要作为服务实现的接口(interface)。
public class SomeClass {
}
public interface Manager<T> {
void add(T obj);
void update(T obj);
void remove(T obj);
}
public interface SomeClassManager extends Manager<SomeClass> {
}
public interface Listener<T> {
void added(T obj);
void updated(T obj);
void removed(T obj);
}
public interface SomeClassListener extends Listener<SomeClass> {
}
假设我有一个 bundle (Core),它提供的服务是某些类型对象的管理器(它基本上包含一个内部列表并添加、删除和更新它)。
public class SomeClassCoreManager implements SomeClassManager {
private ArrayList<SomeClass> list = new ArrayList<SomeClass>();
private ArrayList<SomeListener> listeners = new ArrayList<SomeListener>();
protected void bindListener(SomeListener listener) {
listeners.add(listener);
}
protected void undindListener(SomeListener listener) {
listeners.remove(listener);
}
public void add(SomeClass obj) {
// Adds the object to the list
// Fires all the listeners with "added(obj)"
}
public void update(SomeClass obj) {
// Updates the object in the list.
// Fires all the listeners with "updated(obj)"
}
public void remove(SomeClass obj) {
// Removes the object from the list.
// Fires all the listeners with "removed(obj)"
}
}
我还有第二个包 (UI) 负责主 UI。它不应该“关心”管理自身的对象,但应该在添加、删除或更改对象以更新 JTree 时收到通知。为此,我使用了白板模式:UI 包实现了一项服务,核心包使用该服务来触发对象更改事件。
public class MainWindow extends JFrame {
private JTree tree = new JTree();
private SomeClassManager manager;
protected void activate() {
// Adds the tree and sets its model and creates the rest of the UI.
}
protected void bindManager(SomeClassManager manager) {
this.manager = manager;
}
protected unbindManager(SomeClassManager manager) {
this.manager = null;
}
}
public class SomeClassUIListener implements SomeClassListener {
public void added(SomeClass obj) {
// Should add the object to the JTree.
}
public void updated(SomeClass obj) {
// Should update the existing object in the JTree.
}
public void removed(SomeClass obj) {
// Should remove the existing object from the JTree.
}
}
我的问题如下:
MainWindow 是一个 DS 组件。我正在使用它的激活器来启动整个用户界面。实例创建由 OSGi 处理。
为了从管理器获取更新,我将 SomeClassUIListener 作为声明式服务公开。它的实例也由 OSGi 处理。
我应该如何从 SomeClassUIListener 访问 JTree 模型的实例?
我想出了几个选项,但我不确定该使用哪个:
选项 1:为 UI 包使用某种内部 DI 系统(如 Guice 或 Pico),并将其放入一个具有静态方法的类中以获取它并在整个包中使用它。
有些人似乎不赞成这种方法。
选项 2:通过 OSGi 在 SomeClassUIListener 中注入(inject)对 MainWindow 的引用(通过将其转换为服务),然后从那里开始。这是可能的还是可取的?在我看来,这是更简单的解决方案。但是,另一方面,随着 UI 变得越来越复杂,这不会使组件配置文件变得困惑吗?
选项 3:仅为监听器创建一个单独的包,并使用 OSGi 更新 MainWindow。这在我看来有点极端,因为随着 UI 复杂性的增加,我将不得不创建大量的包。
选项 4:使用 MainWindow 类来实现 Listener。但是,主 UI 包中的服务越多,MainWindow 类就越大。我认为这不是一个好的选择。
我想不出更多的选择。以上任何一条路要走吗?或者有其他选择吗?
提前谢谢你。
编辑:
只是澄清一下,因为 Peter Kriens 对这个问题有一些疑问。
我的目标是将用户界面与管理器分离。 Manager 是指一种存储库,我在其中存储某种类型的对象(例如,如果您考虑位于 http://docs.oracle.com/javase/tutorial/uiswing/components/tree.html 的 Oracle 的 JTree 教程,则管理器将包含 Books 的实例)。
Manager 可以被任何其他 bundle 使用,但根据我目前的计划,它会通知在其中注册的任何监听器。监听器可能是主要的 UI 包,但也可能是选择监听更新的任何其他包。
最佳答案
我不确定我是否完全理解您的建议,感觉您正在创建一整套基础设施。在 OSGi 中,这通常不是必需的,所以为什么不从小处着手呢。
您的基本模型是一个管理器和一个扩展。这是域模型,我会尝试在此处进行处理:
@Component(immediate)
public class ManagerImpl { // No API == immediate
List<Extension> extensions = new CopyOnWriteArrayList<Extension>();
JFrame frame = new JFrame();
@Reference(cardinality=MULTIPLE)
void addExtension( Extension e ) {
addComponent(frame, e.getName(), e.getComponent());
extensions.add(e);
}
void removeExtension( Extension e) {
if ( extensions.remove(e) ) {
removeComponent(frame, e.getName());
}
}
@Component
public class MyFirstExtension implements Extension {
public String getName() { return "My First Extension";}
public Component getComponent() { return new MyFirstExtensionComponent(this); }
}
这不是您要找的吗?小心不要创建各种类型的监听器,通常您会在 OSGi 注册表中找到这些事件。
关于java - OSGi UI 应用程序中的最佳实践,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20218967/
我在 trying to share the Struts packages 时遇到了这个问题在 OSGi 容器内的多个包之间。我想避免在包内重复依赖项并在它们之间引入新的依赖项(通过让一个包导出其内
我正在考虑在我们的项目中管理版本控制的最佳方式。目前,我们 bundle 中的每个包都是导出的(这将在以后改进,所以不要因为这个失礼而挂断电话)。我们正在使用 maven-bundle-plugin,
我正在进行一个项目,我们将迁移基于大量定制技术的主要软件系统,使其基于 OSGi 服务。为此,我们可能需要某种与 OSGi 服务配合良好的消息总线。 同步和异步传送 仅点对点 保证交付 - 最好通过文
我的项目有一组自定义定义的注释,它们可以出现在 OSGi 4.3 框架中部署的任何包中。我想在类路径中找到任何带有这些注释的类。我尝试使用 BundleWiring.listResources(...
我创建了一个片段包来访问一些添加到第二方 jar 的功能。我的片段应该注册一个服务来公开这个新功能。它似乎不起作用。在我深入进行故障排除之前,我想知道这是否被允许?也就是说,Felix SCR 的 b
OSGi 是 Java 的动态模块化系统。好的,但是基线主题是什么,为什么要开发 OSGi?使用 OSGi 有什么好处?开发 OSGi 的主要故事是什么?它为什么存在? 最佳答案 如果你仔细观察,Ja
当一个包被更新(比如修复一个错误)时,当前正在使用正在更新的包的其他包会发生什么? 假设有两个捆绑包 service 和 dao。假设当我发出更新 dao 层的命令时,服务包中的类正在使用 dao 包
在 OSGi 下,组件与服务之间的主要区别是什么?据我了解,所有服务都必须是组件,但并非所有组件都必须是服务。 在示例用例中使用其中一种比另一种有什么好处? 最佳答案 “组件”的定义不如服务正式。 服
这合法吗?org.fragment1 的 MANIFEST.MF(org.host 是普通包,不是片段): Bundle-SymbolicName: org.fragment1 Fragment-Ho
在我当前的应用程序中,我在几个地方遇到了这种模式:我在一个 bundle 中有两个服务接口(interface),它们执行不同但相关的工作。 interface Service1 { ... } in
我的 OSGi 应用程序需要一个 jar(sample;version=A),并且我必须将相同的 jar(sample;version=B) 用于我开发的新包。 示例 jar 有一些增强功能,因此我不
osgi> install file:D:\f1\*.jar osgi> install --start file:D:\f1\*.jar 以上命令在 WSO2 OSGi 控制台中是非法的。如何从文件
和有什么不一样和 在spring DM的xml配置文件中。 最佳答案 可用于获取 对现有 OSGi 服务的引用,以便您的 bean 可以使用它。 可用于导出将 bean 作为 OSGi 服务,以
我有一个应用程序暂时使用 Equinox 作为 osgi 框架。直到现在我都使用系统属性 osgi.install.area 来指定我的包在哪里 ${osgi.install.area}/ plu
背景 在过去一年左右的时间里,我设计了许多工具,旨在帮助我为 XPage 编程。这些工具主要包括帮助程序 java 类、扩展日志记录(使用 OpenLogger 和我自己的东西),以及我个人觉得我不能
我有一个 OSGi 组件 MyComponent . 该组件引用了服务 MyService .现在MyService有几个实现 MyServiceImpl1和 MyServiceImpl2 . MyC
我有几个 OSGi 包,每个都可以从 OSGi 包存储库更新。 当我启动我的 OSGi 框架 (Apache Felix) 时,我希望第一个包启动并检查所有已安装包的更新。如果有可用更新,它应该更新每
当我们在 Apache Felix Web OSGi 控制台的配置选项卡中更新组件的任何配置时,这些配置设置保存在哪里?这是针对 AEM 6.0 或更高版本。 最佳答案 手动保存的配置设置存储在 cr
我有简单的 OSGI 事件监听器类 @Component(immediate = true) @Service(value = { EventHandler.class, JobConsumer.cl
我们正在开发一个网络应用程序(我们称之为图像库),我们已经确定了以下需求: 该应用程序迎合由一组用户组成的客户。 可以动态创建新客户并由客户管理其用户 客户有不同的功能集,可以动态更改 客户可以开发自
我是一名优秀的程序员,十分优秀!