OSGI –模塊化您的應用程序

由于我是模塊化,低耦合,高凝聚力等的大力擁護者,所以……
我相信這項技術是我們使用Java平臺創建應用程序的突破。 使用OSGi,創建高度可擴展的應用程序非常簡單,例如參見Eclipse IDE。 我的目的不是要深入展示該技術的工作原理,而是要舉例說明其某些優勢。 該示例包含一個用于發送消息的系統。 用戶在TextField中鍵入消息,然后可以通過多種方式發送此消息,例如Email或SMS。 但是,在此示例中,我們有四個模塊。 圖形用戶界面,域,電子郵件的發件人以及通過SMS的發件人。

遵循OSGi的術語,每個模塊都是一個捆綁包。 捆綁包不過是一個“罐子”,其中包含MANIFEST.MF的一些附加信息。 此信息由OSGi框架使用。 就像Java中的幾乎所有內容一樣,OSGi技術是一種規范,因此有不同的實現方式可供選擇。 其中包括最著名的Equinox(Eclipse項目),Felix(Apache)和Knopflerfish。 在本文中,我們將使用Equinox。

下載Equinox。 對于本文,我們只需要jar。 運行jar來訪問Equinox的控制臺。

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

框架啟動。

id State Bundle<
0 ACTIVE org.eclipse.osgi_3.5.1.R35x_v20090827
osgi> _

正如我們現在所看到的,我們僅安裝了一個捆綁軟件。 春分包。
現在,我們將創建捆綁包并將其添加到Equinox。 創建捆綁包非常簡單。 使用以下類創建一個簡單的項目:

package br.com.luiscm.helloworld;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 {System.out.println("Hello World!");}/** (non-Javadoc)* @see org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)*/public void stop(BundleContext context) throws Exception {System.out.println("Good Bye World!");}
}

此類是我們捆綁軟件的激活器。 OSGi框架使用Activator來啟動或停止捆綁軟件。 在第一個示例中,激活器僅在啟動和停止時才打印消息。 現在,我們需要修改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?

請參閱傳遞給OSGi捆綁軟件的清單,以了解我們的一些信息。 其中包括束的名稱(SymbolicName)和Activator類。 現在,讓我們在Equinox中安裝此捆綁包。 生成項目的jar并將其安裝在Equinox中很簡單:

install file:.jar
osgi> install file:bundle.jar
Bundle id is 1
osgi>

要驗證捆綁軟件是否已正確安裝,只需運行命令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> _

該捆綁軟件已正確安裝,您現在就可以啟動它:

start
osgi> start 1
Hello World!
osgi>

要停止捆綁包:

osgi> stop 1
Goodbye World!
osgi>

現在我們知道如何創建捆綁包,讓我們開始示例。 在示例中,我們有四個捆綁包。

*域:顧名思義,它在我們的示例中存儲域類。 我們將有兩個類:Message和IMessageSender。
* SenderSMS:通過短信發送消息的IMessageSender實現。
* SenderEmail:通過電子郵件發送消息的IMessageSender實現。 * UI:GUI示例

捆綁用戶界面

我們將從UI捆綁包開始。 激活器將僅構建框架供用戶輸入消息。

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);}
}

請注意,該捆綁包取決于一個稱為Message的類。 此類是我們的領域,因此不屬于捆綁包。 這是OSGi的另一個細節。 通過服務捆綁進行通信。 我們可以將此模型視為VM中的SOA。 服務包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?

請參閱Import-Package語句。 我們正在導入軟件包捆綁包核心。 此程序包中包含我們領域提供的服務。 還要導入javax.swing包。

現在我們需要創建服務。

捆綁核心

核心捆綁包有兩個域類。 發件人的界面和“消息”字段。

接口:

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);}}
}

請參閱Message類,其中包含服務列表。 這些服務是要使用的消息的發送者。 請注意,send方法僅在郵件列表消息上交互。 到目前為止,一切都非常簡單。 現在,我們需要將Message類導出為核心服務包。 UI模塊將直接與此服務交互以發送消息。

首先,我們需要告訴它將OSGi捆綁包導出到其他捆綁包。 參見清單:

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

請參閱信息導出包。 為了使一個類對于另一個捆綁包可見,必須將其導出到包中。 在本例中,UI需要捆綁Message類,因此我們需要將包導出到該類所在的位置。 請記住,UI導入了捆綁包。

消息要將組件注冊為服務,我們需要直接與OSGi API進行交互。 啟動核心捆綁包后,我們將在OSGi上下文中注冊服務。 代碼很簡單:

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期望參數為服務名稱(建議使用類名稱),服務本身以及一些其他設置。

現在,我們需要更改UI以使用捆綁消息服務。 在bundle activator UI中,只需使用您的名字(類名)進行查找服務即可:

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中,則會看到兩個捆綁軟件正在通信。 現在,我們需要創建實際發送消息的包。

捆綁發件人電子郵件和短信

通過電子郵件和SMS的運輸服務將是我們系統中的新服務。 因此,我們為每個創建一個包。 這樣我們可以分別控制它們。 例如,我們可以通過發送短信來停止服務,而只留下電子郵件,而不會影響系統運行。 這兩個捆綁包實際上具有相同的結構,因此我將在此處保存一些行。

發件人只有一個實現接口的捆綁軟件類和IMessageSender Activator類。 該界面位于核心捆綁包中,因此我們需要以與捆綁包UI中相同的方式導入包。

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唯一的類實現我們的接口:

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);}
}

通過短信發送是我們系統的一項服務。 因此,我們必須在OSGi上下文中注冊它:

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;}
}

郵件束實際上是相同的代碼。 唯一的區別是System.out上的消息。

請注意,已使用接口名稱注冊了該服務。 因此,現在我們有兩個具有相同名稱的服務。 每當我們要求使用接口名稱的服務上下文時,他都會執行邏輯優先級以僅返回一個實現。

現在我們有兩個用于發送消息的服務,我們需要更改我們的包核心以使用它們。 為了實現此目標,請使用ServiceTrackerCustomizer。

ServiceTrackerCustomizer和ServiceTracker

如我們所見,我們曾經做Servicetrack查找服務。 但是,對于發送者,我們需要知道何時有新的發送者服務可用或何時刪除發送者。 此信息對于在Message對象中提供服務列表很重要。

要訪問此信息,我們使用ServiceTrackerCustomizer。 代碼很簡單:

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);}
}

只需實現接口,并在添加,修改或刪除服務時根據需要編寫ServiceTrackerCustomizer代碼即可。 簡單!

在我們的情況下,我們將在Message對象的服務列表中添加或刪除服務。 也有一條“日志”消息,以幫助我們進行測試。

現在,我們需要對bundle核心激活器進行另一項較小的更改。 我們必須將ServiceTrackerCustomizer注冊為諸如IMessageSender之類的服務的偵聽器。

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一起使用。 在添加,修改或刪除服務的地方,將調用我們的組件。

測試應用程序

現在我們進行編碼,我們測試應用程序。

創建四個罐子:

* bundleCore.jar
* bundleUI.jar
* bundleSenderEmail.jar * bundleSenderSMS.jar

在Equinox中安裝四個捆綁包:

框架啟動。

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

框架啟動。

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

啟動捆綁包并測試應用程序。

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

查看通過電子郵件和SMS發送的消息。 在控制臺Equinox中,暫停服務電子郵件:

stop

再試一次發送消息。 由于該服務不再可用,因此該消息僅由SMS發送。

停止電源應用程序模塊而不會產生副作用是非常明智的。 想象一下,您在SMS模塊中發現了一個嚴重錯誤。 您無需花費所有精力即可解決此問題。 只需暫停SMS模塊。 系統的其余部分將繼續正常運行。 通過這個小例子進行測試。 暫停和啟動服務。 這不會影響核心,更不會影響UI。

我設法解釋了什么是OSGi。 值得一提的是,這里有關于控制和類路徑配置包的更多詳細信息,此處不再贅述。 這是那些有興趣的人看看其他功能的任務。

值得一看Spring-DM項目。 除了提供出色的IoC容器外,彈簧還使設置服務變得非常容易。

參考: OSGI –在Eclipse Brazil博客上,由我們的JCG合作伙伴 Luis Carlos Moreira da Costa 模塊化您的應用程序 。


翻譯自: https://www.javacodegeeks.com/2012/04/osgi-modularizing-your-application.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/372936.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/372936.shtml
英文地址,請注明出處:http://en.pswp.cn/news/372936.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

jq的鏈式調用.end();

jq的鏈式調用.end(); 先上code <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title><style>li{list-style: none;width: 100px;height:20px;border:1px solid #ff0000;display: …

三、自定義視圖、視圖控制器

1.自定義視圖 自定義視圖&#xff1a;系統標準UI之外&#xff0c;自己組合而出的新的視圖。在實際開發中&#xff0c;我們經常需要自己定義視圖&#xff0c;積累自己的代碼庫。自己封裝的視圖&#xff0c;能像系統提供的UI控件一樣用于多個項目中&#xff0c;這樣可以提高我們的…

程序如何在兩個gpu卡上并行運行_深度學習分布式訓練相關介紹 - Part 1 多GPU訓練...

本篇文章主要是對深度學習中運用多GPU進行訓練的一些基本的知識點進行的一個梳理文章中的內容都是經過認真地分析&#xff0c;并且盡量做到有所考證拋磚引玉&#xff0c;希望可以給大家有更多的啟發&#xff0c;并能有所收獲介紹大多數時候&#xff0c;梯度下降算法的訓練需要較…

集成Spring和JavaServer Faces:改進的模板

隨著2.0版的發布&#xff0c;Facelet模板成為JSF規范的核心部分。 使用<ui&#xff1a;composition>和<ui&#xff1a;decorate>標記&#xff0c;可以輕松構建復雜的頁面&#xff0c;同時仍保持標記清晰。 模板在創建HTML表單時特別有用&#xff0c;但是不幸的是&a…

whmcs模板路徑

whmcs網站根目錄 比如你的域名是server.nongbin.vip&#xff0c;你需要cd /home/wwwroot/server.nongbin.vip&#xff0c;該目錄下然后&#xff0c;cd template/ 給文件夾下就是你上傳的模板文件夾轉載于:https://www.cnblogs.com/nongbin/p/6412108.html

系統英偉達gpu驅動卸載_繞過CPU,英偉達讓GPU直連存儲設備

英偉達最近發布了一個新的GPUDirect Storage&#xff0c;暫且叫做GPU直連存儲&#xff0c;讓GPU直接連到NVMe存儲設備上。這一方案用到了RDMA設備來把數據從閃存存儲轉移到GPU本地的內存里&#xff0c;無需經過CPU還有系統內存。如果這一舉措順利的話&#xff0c;英偉達就能擺脫…

37、EnumSet詳解

EnumSet類也是有順序的&#xff0c;EnumSet按照枚舉值在Enum類內定義的順序決定集合元素的順序 EnumSet在內部已位向量的形式存儲&#xff0c;這種存儲方式非常緊湊、搞笑&#xff0c;因此EnumSet占用內存很小&#xff0c;而且運行效率很好。 EnumSet集合不允許加入null元素 En…

嘲弄和存根–了解Mockito的測試雙打

介紹 我遇到的一件事是使用模擬框架的團隊假設他們在模擬。 他們并不知道Mocks只是Gerard Meszaros在xunitpatterns.com上歸類的“測試雙打”之一。 重要的是要意識到每種類型的測試雙精度在測試中都扮演著不同的角色。 用與您需要學習不同模式或重構的方式相同&#xff0c;您…

numpy 辨異(三)—— hstack/column_stack,linalg.eig/linalg.eigh

1. np.hstack np.column_stack >>> np.hstack([np.array([1, 2, 3]), np.array([4, 5, 6])]) array([1, 2, 3, 4, 5, 6])>>> np.column_stack([np.array([1, 2, 3]), np.array([4, 5, 6])]) array([[1, 4],[2, 5],[3, 6]]) 當然對等地&#xff0c;也存在&…

【代碼筆記】iOS-首頁3張圖片變化

一&#xff0c;效果圖。 二&#xff0c;工程圖。 三&#xff0c;代碼。 RootViewController.h #import <UIKit/UIKit.h>interface RootViewController : UIViewController {NSTimer *timer;UIImageView *imageView1;UIImageView *imageView2;UIImageView *imageView3;UIV…

acwing算法提高之動態規劃--數位DP

目錄 1 基礎知識2 模板3 訓練 1 基礎知識 暫無。。。 2 模板 暫無。。。 3 訓練 題目1&#xff1a;度的數量。 解題思路&#xff1a;分類討論。 C代碼如下&#xff0c; #include <iostream> #include <vector>using namespace std;const int N 35; int K,…

python 輸入數字變成密碼_如何在python中檢查數字的“密碼”

我建議使用sets和stdlib中的string包作為可接受字符的列表。在我還建議進行一點重構&#xff0c;以刪除大量帶有if / else分支的嵌套。在import stringupper set(list(string.uppercase))lower set(list(string.lowercase))numbers set(list(string.digits))while True:npw …

使用Eclipse在Amazon Ec2中部署Java Web應用程序的完整指南

嗨&#xff0c;讀者們&#xff0c; 今天&#xff0c;我將向您展示如何使用Eclipse IDE在Amazon EC2中部署簡單的Java Web應用程序。 在我們開始之前&#xff0c;我們需要一些必需的東西&#xff0c; Eclipse Java EE IDE –您可以從http://www.eclipse.org/downloads/下載&…

jquery的load方法

load方法指定一個界面會顯示在目標的標簽內部 比如MVC的一個分部視圖頁面想要顯示在某個標簽里面&#xff0c;可以寫成 $(標簽ID).load&#xff08;分部視圖名稱,data&#xff09; 其中第二個參數可選&#xff0c;主要是一些需要傳遞到該頁面的數據JSON格式組成&#xff0c;發送…

android 錄音原始文件_音頻采集:Android基于AudioRecord的實現

前言這篇文章簡單介紹下移動端Android系統下利用AudioRecord進行音頻采集方法。按照慣例開始前先提供一份源碼 AudioRecordLib 。AudioRecord采集的核心實現在于 AudioRecordCore.java 這個文件。權限申請想要使用AudioRecord這個API&#xff0c;需要在AndroidManifest.xml的配…

Spring 3和Java EE 6 –不公平和不完整的比較

這篇小文章的初稿標題為“ Spring&#xff06;Java EE –比較蘋果和橙子”。 在撰寫本文時&#xff0c;我了解到可以比較Spring Framework和Java EE&#xff0c;但這始終是不公平且不完整的工作。 Java for Enterprise和Spring Framework的發展緊密地聯系在一起。 兩者相互依存…

xml配置文件推薦方式

1.XML幫助類 /// <summary>/// Xml幫助類/// </summary>public class XmlHelper{/// <summary>/// 保存xml/// </summary>/// <typeparam name"T"></typeparam>/// <param name"path"></param>/// <p…

AFNetWorking https SSL認證

一般來講如果app用了web service , 我們需要防止數據嗅探來保證數據安全.通常的做法是用ssl來連接以防止數據抓包和嗅探 其實這么做的話還是不夠的 。 我們還需要防止中間人攻擊&#xff08;不明白的自己去百度&#xff09;。攻擊者通過偽造的ssl證書使app連接到了偽裝的假冒的…

查看環境列表_Xfce 4.14桌面環境正式發布,想要圖形界面又想節省內存?就它了...

1. Xfce 4.14桌面環境正式發布&#xff0c;它有什么新特性&#xff1f;本文主要講解Xfce 4.14桌面環境正式發布&#xff0c;它有什么新特性。Xfce已經開發了4年多&#xff0c;但是這個周末終于看到了期待已久的Xfce 4.14的發布。Xfce 4.14是這個輕量級桌面環境的最新穩定版本&a…

使用Log4jdbc記錄JDBC操作

當我們開發任何應用程序&#xff0c;完成它或結束其任何模塊時&#xff0c;我們都會開始優化過程。 大多數應用程序都包含數據庫訪問權限&#xff0c;并且如果您使用的是ORM &#xff0c;則可能會使用hibernate 。 優化休眠持久層&#xff0c;要求準備閱讀&#xff0c;理解和評…