osgi 模块化_OSGI –模块化您的应用程序
osgi 模塊化
由于我是模塊化,低耦合,高凝聚力等的大力擁護(hù)者,所以……我相信這項(xiàng)技術(shù)是我們使用Java平臺(tái)創(chuàng)建應(yīng)用程序的突破。 使用OSGi,創(chuàng)建高度可擴(kuò)展的應(yīng)用程序非常簡(jiǎn)單,例如參見(jiàn)Eclipse IDE。 我的目的不是要深入展示該技術(shù)的工作原理,而是要舉例說(shuō)明其某些優(yōu)勢(shì)。 該示例包含一個(gè)用于發(fā)送消息的系統(tǒng)。 用戶在TextField中鍵入一條消息,然后可以通過(guò)多種方式發(fā)送該消息,例如Email或SMS。 但是,在此示例中,我們有四個(gè)模塊。 圖形用戶界面,域,電子郵件的發(fā)件人以及通過(guò)SMS的發(fā)件人。
遵循OSGi的術(shù)語(yǔ),每個(gè)模塊都是一個(gè)捆綁包。 捆綁包不過(guò)是一個(gè)“罐子”,其中包含MANIFEST.MF的一些附加信息。 此信息由OSGi框架使用。 像Java中的幾乎所有內(nèi)容一樣,OSGi技術(shù)是一種規(guī)范,因此有不同的實(shí)現(xiàn)方式可供選擇。 其中最著名的是Equinox(Eclipse項(xiàng)目),Felix(Apache)和Knopflerfish。 在本文中,我們將使用Equinox。
下載Equinox。 對(duì)于本文,我們只需要罐子即可。 運(yùn)行jar以訪問(wèn)Equinox的控制臺(tái)。
C:\osgi>java -jar org.eclipse.osgi_3.5.1.R35x_v20090827.jar –console要查看已安裝的捆綁包,只需鍵入命令ss。
C:\osgi>java -jar org.eclipse.osgi_3.5.1.R35x_v20090827.jar – console osgi> ss框架啟動(dòng)。
id State Bundle< 0 ACTIVE org.eclipse.osgi_3.5.1.R35x_v20090827 osgi> _ 正如我們現(xiàn)在所看到的,我們僅安裝了一個(gè)捆綁軟件。 春分包。
現(xiàn)在,我們將創(chuàng)建捆綁包并將其添加到Equinox。 創(chuàng)建捆綁包非常簡(jiǎn)單。
使用以下類創(chuàng)建一個(gè)簡(jiǎn)單的項(xiàng)目:
此類是我們捆綁軟件的激活器。 OSGi框架使用激活器來(lái)啟動(dòng)或停止捆綁軟件。 在第一個(gè)示例中,激活器僅在啟動(dòng)和停止時(shí)才打印消息。 現(xiàn)在,我們需要修改jar的清單以使其成為OSGi捆綁包。
Manifest-Version: 1.0 Bundle-ManifestVersion: 2 Bundle-Name: LuisCM Plug-in Bundle-SymbolicName: br.com.luiscm.helloworld Bundle-Version: 1.0.0 Bundle-Activator: br.com.luiscm.helloworld.Activator Bundle-ActivationPolicy: lazy Bundle-RequiredExcutionEnvironment: JavaSE-1.6 Import-Package: org.osgi.framework;version=”1.3.0?請(qǐng)參閱傳遞給OSGi捆綁軟件的清單,以了解我們的一些信息。 其中的束名稱(SymbolicName)和Activator類。 現(xiàn)在讓我們?cè)贓quinox中安裝此捆綁包。 生成項(xiàng)目的jar并將其安裝在Equinox中很簡(jiǎn)單:
install file:.jar osgi> install file:bundle.jar Bundle id is 1 osgi>要驗(yàn)證是否正確安裝了捆綁軟件,只需運(yùn)行命令ss:
osgi> ss Framework is launched. id State Bundle 0 ACTIVE org.eclipse.osgi_3.5.1.R35x_v20090827 1 INSTALLED br.com.luiscm.helloworld_1.0.0 osgi> _該捆綁包已正確安裝,您現(xiàn)在就可以啟動(dòng)它:
start osgi> start 1 Hello World! osgi>停止捆綁包:
osgi> stop 1 Goodbye World! osgi>現(xiàn)在我們知道如何創(chuàng)建捆綁包,讓我們開(kāi)始示例。 在示例中,我們有四個(gè)捆綁包。
*域:顧名思義,它在我們的示例中存儲(chǔ)域類。 我們將有兩個(gè)類:Message和IMessageSender。
* SenderSMS:通過(guò)短信發(fā)送消息的IMessageSender實(shí)現(xiàn)。
* SenderEmail:實(shí)現(xiàn)通過(guò)電子郵件發(fā)送消息的IMessageSender。 * UI:GUI示例
捆綁用戶界面
我們將從UI捆綁包開(kāi)始。 激活器將為用戶輸入消息建立框架。
package br.com.luiscm.helloworld;import java.awt.BorderLayout; import java.awt.event.ActionEvent; import java.awt.event.ActionListener;import javax.swing.JButton; import javax.swing.JFrame; import javax.swing.JTextField;import org.osgi.framework.BundleActivator; import org.osgi.framework.BundleContext;import br.com.luiscm.helloworld.core.service.Message;public class Activator implements BundleActivator {private Message message;private JFrame frame;/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext)*/public void start(BundleContext context) throws Exception {buildInterface();}/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)*/public void stop(BundleContext context) throws Exception {destroyInterface();}private void destroyInterface() {frame.setVisible(false);frame.dispose();}private void buildInterface() {frame = new JFrame("Hello");frame.setSize(200, 80);frame.getContentPane().setLayout(new BorderLayout());final JTextField textField = new JTextField();final JButton button = new JButton("Send");button.addActionListener(new ActionListener() {@Overridepublic void actionPerformed(ActionEvent event) {message.send(textField.getText());}});frame.getContentPane().add(textField, BorderLayout.NORTH);frame.getContentPane().add(button, BorderLayout.SOUTH);frame.setVisible(true);} }請(qǐng)注意,該捆綁包取決于一個(gè)稱為Message的類。 此類是我們的領(lǐng)域,因此不屬于此捆綁包。 這是OSGi的另一個(gè)細(xì)節(jié)。 通信是通過(guò)捆綁服務(wù)完成的。 我們可以將此模型視為VM中的SOA。 服務(wù)包UI將使用包核心。 讓我們看一下MANIFEST包UI。
Manifest-Version: 1.0 Bundle-ManifestVersion: 2 Bundle-Name: UI Plug-in Bundle-SymbolicName: br.com.luiscm.helloworld.ui< Bundle-Version: 1.0.0 Bundle-Activator: br.com.luiscm.helloworld.ui.Activator Bundle-ActivationPolicy: lazy Bundle-RequiredExecutionEnvironment: JavaSE-1.6 Import-Package: br.com.luiscm.helloworld.core.service, javax.swing, org.osgi.framework;version=”1.3.0?, org.osgi.util.tracker;version=”1.3.6?請(qǐng)參閱Import-Package語(yǔ)句。 我們正在導(dǎo)入軟件包捆綁包核心。 此程序包中包含我們領(lǐng)域提供的服務(wù)。 還要導(dǎo)入javax.swing包。
現(xiàn)在我們需要?jiǎng)?chuàng)建服務(wù)。
捆綁核心
核心捆綁包有兩個(gè)域類。 發(fā)件人的界面和“消息”字段。
接口:
package br.com.luiscm.helloworld.core.service;public interface IMessageSender {void send(String message); }域:
package br.com.luiscm.helloworld.core.service;import java.util.ArrayList; import java.util.List;public class Message {private final List services = new ArrayList();public void addService(final IMessageSender messageSender) {services.add(messageSender);}public void removeService(final IMessageSender messageSender) {services.remove(messageSender);}public void send(final String message) {for (final IMessageSender messageSender : services) {messageSender.send(message);}} }請(qǐng)參閱Message類,其中包含服務(wù)列表。 這些服務(wù)是要使用的消息的發(fā)送者。 請(qǐng)注意,send方法僅在郵件列表消息上交互。 到目前為止,一切都非常簡(jiǎn)單。 現(xiàn)在,我們需要將Message類導(dǎo)出為核心服務(wù)包。 UI模塊將直接與此服務(wù)交互以發(fā)送消息。
首先,我們需要告訴它將OSGi捆綁包導(dǎo)出到其他捆綁包。 參見(jiàn)清單:
Manifest-Version: 1.0 Bundle-ManifestVersion: 2 Bundle-Name: Helloworld Plugin Bundle-SymbolicName: br.com.luiscm.helloworld.core Bundle-Version: 1.0.0 Bundle-Activator: br.com.luiscm.helloworld.core.Activator Bundle-ActivationPolicy: lazy Bundle-RequiredExecutionEnvironment: JavaSE-1.6 Import-Package: org.osgi.framework;version=”1.3.0?, org.osgi.util.tracker;version=”1.3.6? Export-Package: br.com.luiscm.helloworld.core.service請(qǐng)參閱信息導(dǎo)出包。 為了使一個(gè)類對(duì)于另一個(gè)捆綁包可見(jiàn),必須將其導(dǎo)出到包中。 在我們的例子中,UI需要捆綁Message類,因此我們需要將包導(dǎo)出到該類所在的位置。 請(qǐng)記住,UI導(dǎo)入了捆綁包。
消息要將組件注冊(cè)為服務(wù),我們需要直接與OSGi API進(jìn)行交互。 啟動(dòng)核心捆綁包后,我們將在OSGi上下文中注冊(cè)服務(wù)。 代碼很簡(jiǎn)單:
package br.com.luiscm.helloworld.core;import org.osgi.framework.BundleActivator; import org.osgi.framework.BundleContext;public class Activator implements BundleActivator {/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext)*/public void start(BundleContext context) throws Exception {context.registerService(Message.class.getName(), messageService, null);}/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)*/public void stop(BundleContext context) throws Exception {messageService = null;} }方法registerService期望參數(shù)為服務(wù)名稱(建議使用類名稱),服務(wù)本身以及一些其他設(shè)置。
現(xiàn)在,我們需要更改UI以使用捆綁消息服務(wù)。 在bundle activator UI中,只需使用您的名字(類名)進(jìn)行查找服務(wù)即可:
private Message message;private JFrame frame;private ServiceTracker serviceTracker;/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext)*/public void start(BundleContext context) throws Exception {serviceTracker = new ServiceTracker(context, Message.class.getName(), null);serviceTracker.open();message = (Message)serviceTracker.getService();buildInterface();}/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)*/public void stop(BundleContext context) throws Exception {destroyInterface();serviceTracker.close();} }如果在Equinox中添加兩個(gè)捆綁包,我們將看到兩個(gè)捆綁包正在通信。 現(xiàn)在,我們需要?jiǎng)?chuàng)建實(shí)際發(fā)送消息的包。
捆綁發(fā)件人電子郵件和短信
通過(guò)電子郵件和SMS的運(yùn)輸服務(wù)將是我們系統(tǒng)中的新服務(wù)。 因此,我們?yōu)槊總€(gè)創(chuàng)建一個(gè)包。 這樣我們可以分別控制它們。 例如,我們可以通過(guò)發(fā)送短信來(lái)停止服務(wù),而只留下電子郵件,而不會(huì)影響系統(tǒng)操作。 這兩個(gè)捆綁包實(shí)際上具有相同的結(jié)構(gòu),因此我將在此處保存一些行。
發(fā)件人只有一個(gè)實(shí)現(xiàn)接口的捆綁軟件類和IMessageSender Activator類。 該接口位于核心捆綁包中,因此我們需要以與捆綁包UI中相同的方式導(dǎo)入包。
Manifest-Version: 1.0 Bundle-ManifestVersion: 2 Bundle-name: SMS Plug-in Bundle-SymbolicName: br.com.luiscm.helloworld.sms.Activator< Bundle-Version: 1.0.0 Bundle-Activator: br.com.luiscm.helloworld.sms.Activator Bundle-ActivationPolicy: lazy Bundle-RequiredExecutionEnvironment: JavaSE-1.6 Import-Package: br.com.luiscm.helloworld.core.service, org.osgi.framework;version=”1.3.0?Sender唯一的類實(shí)現(xiàn)我們的接口:
package br.com.luiscm.helloworld.sms;import br.com.luiscm.helloworld.core.service.IMessageSender;public class MessageSenderSMS implements IMessageSender {@Overridepublic void send(final String message) {System.out.println("Sending by SMS : " + message);} }通過(guò)短信發(fā)送是我們系統(tǒng)的一項(xiàng)服務(wù)。 因此,我們必須在OSGi上下文中注冊(cè)它:
public class Activator implements BundleActivator {private IMessageSender service;/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext)*/public void start(BundleContext context) throws Exception {service = new MessageSenderSMS();context.registerService(IMessageSender.class.getName(), service, null);}/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)*/public void stop(BundleContext context) throws Exception {service = null;} }郵件束實(shí)際上是相同的代碼。 唯一的區(qū)別是System.out上的消息。
請(qǐng)注意,已使用接口名稱注冊(cè)了該服務(wù)。 因此,現(xiàn)在我們有兩個(gè)具有相同名稱的服務(wù)。 每當(dāng)我們要求使用接口名稱的服務(wù)上下文時(shí),他都會(huì)執(zhí)行邏輯優(yōu)先級(jí)以僅返回一個(gè)實(shí)現(xiàn)。
現(xiàn)在我們有兩個(gè)用于發(fā)送消息的服務(wù),我們需要更改我們的包核心以使用它們。 為了實(shí)現(xiàn)此目標(biāo),請(qǐng)使用ServiceTrackerCustomizer。
ServiceTrackerCustomizer和ServiceTracker
如我們所見(jiàn),我們?cè)?jīng)做Servicetrack查找服務(wù)。 但是,對(duì)于發(fā)送者,我們需要知道何時(shí)有新的發(fā)送者服務(wù)可用或何時(shí)刪除發(fā)送者。 此信息對(duì)于在Message對(duì)象中提供服務(wù)列表很重要。
要訪問(wèn)此信息,我們使用ServiceTrackerCustomizer。 代碼很簡(jiǎn)單:
package br.com.luiscm.helloworld.core;import org.osgi.framework.BundleContext;public class MessageSenderServiceTracker implements ServiceTrackerCustomizer {private final BundleContext context;private final Message message;public MessageSenderServiceTracker(final BundleContext context, final Message message) {this.context = context;this.message = message;}@Overridepublic Object addingService(final ServiceReference serviceReference) {final IMessageSender sender = (IMessageSender)context.getService(serviceReference);message.addService(sender);System.out.println("tracker : " + sender.getClass().getName());return sender;}@Overridepublic void removedService(final ServiceReference serviceReference, Object service) {final IMessageSender sender = (IMessageSender)context.getService(serviceReference);message.removeService(sender);} }只需實(shí)現(xiàn)接口,并在添加,修改或刪除服務(wù)時(shí)根據(jù)需要編寫ServiceTrackerCustomizer代碼即可。 簡(jiǎn)單!
在我們的例子中,我們將在Message對(duì)象的服務(wù)列表中添加或刪除服務(wù)。 還帶有一條“日志”消息,以幫助我們進(jìn)行測(cè)試。
現(xiàn)在,我們需要對(duì)bundle核心激活器進(jìn)行另一項(xiàng)較小的更改。 我們必須將ServiceTrackerCustomizer注冊(cè)為諸如IMessageSender之類的服務(wù)的偵聽(tīng)器。
public class Activator implements BundleActivator {public Message messageService = new Message();private ServiceTracker messageSenderServiceTracker;/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext)*/public void start(BundleContext context) throws Exception {context.registerService(Message.class.getName(), messageService, null);final MessageSenderServiceTracker serviceTracker = new MessageSenderServiceTracker(context, messageService);messageSenderServiceTracker = new ServiceTracker(context, IMessageSender.class.getName(), serviceTracker);messageSenderServiceTracker.open();}/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)*/public void stop(BundleContext context) throws Exception {messageSenderServiceTracker.close();messageService = null;} }我們將ServiceTrackerCustomizer與ServiceTtracker一起使用。 在添加,修改或刪除服務(wù)的地方,我們的組件將被調(diào)用。
測(cè)試應(yīng)用程序
現(xiàn)在我們進(jìn)行編碼,我們測(cè)試應(yīng)用程序。
創(chuàng)建四個(gè)罐子:
* bundleCore.jar
* bundleUI.jar
* bundleSenderEmail.jar * bundleSenderSMS.jar
在Equinox中安裝四個(gè)捆綁包:
框架啟動(dòng)。
id State Bundle 0 ACTIVE org.eclipse.osgi_3.5.1.R35x_20090827.jar osgi> install file:bundleCore.jar Bundle id is 5 osgi> install file:bundleUI.jar Bundle id is 6 osgi> install file:bundleSenderEmail.jar Bundle id is 7 osgi> install file:bundleSenderSMS.jar Bundle id is 8 osgi> ss框架啟動(dòng)。
0 ACTIVE org.eclipse.osgi._3.5.1.R35x_v20090827.jar 5 INSTALLED br.com.luiscm.helloworld.core_1.0.0 6 INSTALLED br.com.luiscm.helloworld.ui_1.0.0 7 INSTALLED br.com.luiscm.helloworld.email_1.0.0 8 INSTALLED br.com.luiscm.helloworld.sms_1.0.0啟動(dòng)捆綁包并測(cè)試應(yīng)用程序。
C:\osgi>java -jar org.eclipse.osgi._3.5.1.R35x_v20090827.jar -console osgi> tracker: br.com.luiscm.heloworld.sms.SenderSMStracker: br.com.luiscm.helloworld.email.SenderEmail查看通過(guò)電子郵件和SMS發(fā)送的消息。 在控制臺(tái)Equinox中,暫停服務(wù)電子郵件:
stop再試一次發(fā)送消息。 由于該服務(wù)不再可用,因此該消息僅由SMS發(fā)送。
停止電源應(yīng)用程序模塊而不會(huì)產(chǎn)生副作用是非常明智的。 想象一下,您在SMS模塊中發(fā)現(xiàn)了一個(gè)嚴(yán)重錯(cuò)誤。 您無(wú)需花費(fèi)所有精力即可解決此問(wèn)題。 只需暫停SMS模塊。 系統(tǒng)的其余部分將繼續(xù)正常運(yùn)行。 通過(guò)這個(gè)小例子進(jìn)行測(cè)試。 暫停和啟動(dòng)服務(wù)。 這不會(huì)影響核心,更不會(huì)影響UI。
我設(shè)法解釋了什么是OSGi。 值得注意的是,這里有關(guān)于控制和類路徑配置包的更多詳細(xì)信息,此處不再關(guān)注。 這是那些有興趣的人看看其他功能的任務(wù)。
值得一看Spring-DM項(xiàng)目。 除了提供出色的IoC容器外,彈簧還使設(shè)置服務(wù)變得非常容易。
參考: OSGI –在Eclipse Brazil博客上,由我們的JCG合作伙伴 Luis Carlos Moreira da Costa 模塊化您的應(yīng)用程序 。
翻譯自: https://www.javacodegeeks.com/2012/04/osgi-modularizing-your-application.html
osgi 模塊化
總結(jié)
以上是生活随笔為你收集整理的osgi 模块化_OSGI –模块化您的应用程序的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 防盗Java EE –保护您的Java
- 下一篇: 通过Java 8中的Applicativ