Java技术回顾之JNDI--实例

一、JNDI在Java EE中的应用
JNDI 技术是Java EE规范中的一个重要“幕后”角色,它为Java EE容器、组件提供者和应用程序之间提供了桥梁作用:Java EE容器同时扮演JNDI提供者角色,组件提供者将某个服务的具体实现部署到容器上,应用程序通过标准的JNDI接口就可以从容器上发现并使用服务,而不用关心服务的具体实现是什么,它的具体位置在哪里。

下面以一个常见的J2EE应用场景来看四种角色(组件接口、容器、组件提供者、应用程序)是如何围绕JNDI来发挥作用的:
组件接口
数据源DataSource是一种很常见的服务。我们通常将组件接口绑定到容器的Context上供客户调用。

Java EE容器

Tomcat是一种常见的Java EE容器,其他的还有JBoss,WebLogic,它们同时也实现了JNDI提供者规范。容器通常提供一个JNDI注入场所供加入组件的具体实现,比如 Tomcat中的Server.xml配置文件。

组件提供者
众多数据库厂商提供了DataSource的实现,比如OracleDataSource,MySQLDataSource,XXXDataSource 等。我们将该实现的部署到容器中:将一系列jar加入classpath中,在Server.xml中配置DataSource实现,如:
<Resource name="jdbc/MyDB" auth="Container" type="javax.sql.DataSource" ..../>

应用程序
一个JSP/Servlet应用程序。通过JNDI接口使用 DataSource服务,如:
Context initContext = new InitialContext();
Context envContext  = (Context)initContext.lookup("java:/comp/env");
DataSource ds = (DataSource)envContext.lookup("jdbc/MyDB");

关于在Tomcat中如何配置DataSource,可以参考文档:http://tomcat.apache.org/tomcat-5.5-doc/jndi-datasource-examples-howto.html
关于在Tomcat中如何配置其他JNDI服务,可以参考文档:
http://tomcat.apache.org/tomcat-5.5-doc/jndi-resources-howto.html

二、 JNDI实例演练:在Java SE中使用JNDI
要在Java EE中环境中提供一个独立的实例不太容易。下面以一个独立的Java SE应用来演练JNDI的使用过程。在该应用中,我们使用JNDI来使用两种假想的服务:数据库服务DBService和日志服务LogService。


1、指定JNDI提供者
要使用JNDI,首先要配置JNDI提供者。在我们的 Java SE应用中,没有Java EE容器充当JNDI提供者,因此我们要指定其他的JNDI提供者。在我们的例子里,我们使用SUN的文件系统服务提供者File System Service Provider。由于SUN的文件系统服务提供者并没有包含在JDK中,我们需要从SUN网站上下载:
http://java.sun.com/products/jndi/downloads/index.html。它包含两个jar文件:fscontext.jar和providerutil.jar。我们将这两个文件加入项目的类路径中。

2、定义服务接口
首先,我们在服务接口package(xyz.service)中定义服务接口:DBService和LogService,分别表示数据库服务和日志服务。

♦ 数据库服务接口 DBService.java

package xyz.service;public interface DBService {String getLocation(); //获取数据库位置String getState(); //获取数据库状态void accessDB(); //访问数据库void setProperty(int index,String property); //设置数据库信息
}

♦日志服务接口 LogService.java

package xyz.service;public interface LogService {void log(String message); //记录日志
}
3、组件提供者实现服务接口

接下来,我们在组件提供者package(xyz.serviceprovider)中提供DBService和LogService的实现:SimpleDBService和SimpleLogService。为了让服务能够在JNDI环境中使用,根据JNDI规范,我们同时定义两个对象工厂类SimpleDBServiceFactory和SimpleLogServiceFactory,分别用来创建服务实例。

♦ 数据库服务接口实现  SimpleDBService.java
package xyz.serviceprovider;import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.Referenceable;
import javax.naming.StringRefAddr;import xyz.service.DBService;// 为了将数据库服务实例加入JNDI的Context中,我们需要实现Referenceable接口,并实现RetReference方法。
// 关于Reference和Referenceable,请参考上一篇:Java技术回顾之JNDI:JNDI API
public class SimpleDBService implements Referenceable, DBService {private String location = "mydb//local:8421/defaultdb";// 数据库服务属性之一:数据库位置private String state = "start"; // 数据库服务属性之二:数据库状态public Reference getReference() throws NamingException {// Reference是对象的引用,Context中存放的是Reference,为了从Reference中还原出对象实例,// 我们需要添加RefAddr,它们是创建对象实例的线索。在我们的// SimpleDBService中,location和state是这样两个线索。Reference ref = new Reference(getClass().getName(),SimpleDBServiceFactory.class.getName(), null);ref.add(new StringRefAddr("location", location));ref.add(new StringRefAddr("state", state));return ref;}public void accessDB() {if (state.equals("start"))System.out.println("we are accessing DB.");elseSystem.out.println("DB is not start.");}public String getLocation() {return location;}public String getState() {return state;}public void setProperty(int index, String property) {if (index == 0)location = property;elsestate = property;}
}
♦数据库服务对象工厂类 SimpleDBServiceFactory.java
package xyz.serviceprovider;import java.util.Hashtable;import javax.naming.Context;
import javax.naming.Name;
import javax.naming.Reference;
import javax.naming.spi.ObjectFactory;// 数据库服务对象工厂类被JNDI提供者调用来创建数据库服务实例,对使用JNDI的客户不可见。
public class SimpleDBServiceFactory implements ObjectFactory {// 根据Reference中存储的信息创建出SimpleDBService实例public Object getObjectInstance(Object obj, Name name, Context ctx, Hashtable<?, ?> env) throws Exception {if (obj instanceof Reference) {Reference ref = (Reference) obj;String location = (String) ref.get("location").getContent();String state = (String) ref.get("state").getContent();SimpleDBService db = new SimpleDBService();db.setProperty(0, location);db.setProperty(1, state);return db;}return null;}
}
♦ 日志服务接口实现 SimpleLogService.java
package xyz.serviceprovider;import java.text.SimpleDateFormat;
import java.util.Date;import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.Referenceable;import xyz.service.LogService;public class SimpleLogService implements Referenceable, LogService {private SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");// SimpleLogService没有任何属性,通过 SimpleLogService类名创建出来的SimpleLogService实例都是一样的,// 因此这里无需添加RefAddr了。public Reference getReference() throws NamingException {return new Reference(getClass().getName(),SimpleLogServiceFactory.class.getName(), null);}public void log(String message) {String date = sdf.format(new Date());System.out.println(date + ":" + message);}
}
♦日志服务对象工厂类 SimpleLogServiceFactory.java
package xyz.serviceprovider;import java.util.Hashtable;import javax.naming.Context;
import javax.naming.Name;
import javax.naming.Reference;
import javax.naming.spi.ObjectFactory;public class SimpleLogServiceFactory implements ObjectFactory {public Object getObjectInstance(Object obj, Name name, Context ctx,Hashtable<?, ?> env) throws Exception {if (obj instanceof Reference) {return new SimpleLogService();}return null;}
}
 4、 JNDI容器和JNDI客户端
最后,我们在JNDI应用package(xyz.jndi)中实现一个JNDI容器 JNDIContainer和一个JNDI客户端应用JNDIClient。
JNDIContainer在内部使用文件系统服务提供者 fscontext来提供命名和目录服务,配置文件JNDIContainer.properties是服务注入场所,供配置DBService和 LogService实现。

♦JNDI容器类 JNDIContainer.java
package xyz.jndi;import java.io.InputStream;
import java.util.Hashtable;
import java.util.Properties;import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;import xyz.service.DBService;
import xyz.service.LogService;public class JNDIContainer {private Context ctx = null;public void init() throws Exception {// 初始化JNDI提供者。Hashtable<Object,Object> env = new Hashtable<Object,Object>();env.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.jndi.fscontext.RefFSContextFactory");env.put(Context.PROVIDER_URL, "file:/d:/works"); // fscontext的初始目录,我们需要在d:\下创建works目录。ctx = new InitialContext(env);loadServices();}// 从配置文件 JNDIContainer.properties中读取DBService和LogService实现,绑定到Context中。private void loadServices() throws Exception {InputStream in = getClass().getResourceAsStream("JNDIContainer.properties");Properties props = new Properties();props.load(in);// inject dbserviceString s = props.getProperty("DBServiceClass");Object obj = Class.forName(s).newInstance();if (obj instanceof DBService) {DBService db = (DBService) obj;String[] ss = props.getProperty("DBServiceProperty").split(";");for (int i = 0; i < ss.length; i++)db.setProperty(i, ss[i]);ctx.rebind(props.getProperty("DBServiceName"), db);}// inject logservices = props.getProperty("LogServiceClass");obj = Class.forName(s).newInstance();if (obj instanceof LogService) {LogService log = (LogService) obj;ctx.rebind(props.getProperty("LogServiceName"), log);}}public void close() throws NamingException {ctx.close();}public Context getContext() {return ctx;}
}
♦JNDI 容器配置文件 JNDIContainer.properties
//和 JNDIContainer.java文件位于同一目录
DBServiceName=DBService
DBServiceClass=xyz.serviceprovider.SimpleDBService
DBServiceProperty=mydb//192.168.1.2:8421/testdb;startLogServiceName=LogService
LogServiceClass=xyz.serviceprovider.SimpleLogService

♦JNDI 客户端 JNDIClient.java
package xyz.jndi;import javax.naming.Context;import xyz.service.DBService;
import xyz.service.LogService;public class JNDIClient {public static void main(String[] args) {try {JNDIContainer container = new JNDIContainer();container.init();// JNDI客户端使用标准JNDI接口访问命名服务。Context ctx = container.getContext();DBService db = (DBService) ctx.lookup("DBService");System.out.println("db location is:" + db.getLocation()+ ",state is:" + db.getState());db.accessDB();LogService ls = (LogService) ctx.lookup("LogService");ls.log("this is a log message.");container.close();} catch (Exception e) {e.printStackTrace();}}
}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/497882.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【观点】智能制造:新时代智能产业革命的基石|王飞跃

来源&#xff1a; 中国科学院自动化研究所概要&#xff1a;以新的理念和新的技术发展新时期的智能制造科技&#xff0c;创新智能产业革命&#xff0c;将是中国从制造大国到制造强国&#xff0c;进而从世界大国到世界强国的必由之道和开路先锋。实体经济&#xff0c;特别是以制造…

matlab连续型随机变量,matlab连续型随机变量的分布.doc

matlab连续型随机变量的分布.doc 连续型随机变量的分布及其数字特征一、基本概念设随机变量X的分布函数为F(x)&#xff0c;若存在非负函数f(x)&#xff0c;使对任意实数x&#xff0c;有≤X{Pxd}则称X为连续型随机变量&#xff0c;并称f(x)为X的概率密度&#xff0c;它满…

CB Insights发布AI创业公司100榜单 ,7家中国公司上榜,两家二次登榜

来源&#xff1a;36氪概要&#xff1a;近日&#xff0c;硅谷知名数据公司 CB Insights 在美国旧金山发布了第二届全球最强 AI 创业公司榜单AI 100。旷视科技、出门问问、今日头条、英语流利说、优必选、商汤科技以及寒武纪上榜 。近日&#xff0c;硅谷知名数据公司 CB Insights…

深度学习发展简要笔记

来源&#xff1a;非正式组织概要&#xff1a;在Yann LeCun、Yoshua Bengio和Geoffrey Hinton三巨头nature深度学习综述《deep learning》文章中提到&#xff0c;这段期间神经网络模型被主流的计算机视觉和学术界所抛弃。一、前言深度学习的发展大致分为这么几个学期&#xff1a…

数据革命在危机中诞生 援助组织尝试灵活方法拨开战争迷雾

无国界医生组织在安曼为叙利亚提供支持。图片来源&#xff1a;Neil Brandvold来源&#xff1a;科学网概要&#xff1a;每当战争、飓风或其他灾害肆虐时&#xff0c;援助组织面临的最大问题之一就是缺乏可靠的数据。当Issam Salim&#xff08;化名&#xff09;讲述他进行的手术时…

德勤2018TMT八大预测:移动互联网迎来二次革命

来源&#xff1a;智东西概要&#xff1a;数万手机AR应用上市&#xff0c;AI芯片打入旗舰手机&#xff0c;数据中心ML芯片多样化&#xff0c;移动数字经济继续开挂。2017年&#xff0c;很多传统TMT巨头面临增长缓趋&#xff0c;于是考虑重资产化追求规模效应&#xff0c;开拓、转…

oracle 中 start tran,The value (30) of MAXTRANS parameter ignored

The value (30) of MAXTRANS parameter ignored在客户的数据库系统中(Oracle 10.2.0.3)&#xff0c;当使用expdp备份时就出现如下错误&#xff0c;非常准时。可以看到&#xff0c;这个错误就是由于备份时创建备份主表(Master Table)产生的&#xff1a;Sat Mar 7 02:40:01 2009T…

北京出台自动驾驶新规:自动驾驶车辆须配备司机应急

来源&#xff1a;人民日报概要&#xff1a;在中国境内注册的独立法人单位&#xff0c;因进行自动驾驶相关科研、定型试验&#xff0c;可申请临时上路行驶。记者今天从北京市交通委获悉&#xff0c;北京市交通委联合市公安交管局、市经济信息委等部门&#xff0c;制定发布了加快…

【未来研究】城市云脑是互联网云脑的节点,城市云脑之间如何互补与支撑

作者: 互联网进化论作者 &#xff0c;计算机博士 刘锋前言&#xff1a;本文是未来智能实验室的研究文章&#xff0c;重点探讨了基于类脑架构的智慧城市建设不应局限在城市内部&#xff0c;而是在互联网大背景下&#xff0c;不同城市之间也会发生智慧建设的协同效应。本文从城市…

JNDI全攻略

名词解释jndi是Java 命名和目录接口&#xff08;Java Naming and Directory Interface&#xff0c;JNDI&#xff09;的简称.从一开始就一直是 Java 2 平台企业版&#xff08;JEE&#xff09;的核心技术之一。在JMS&#xff0c;JMail,JDBC,EJB等技术中&#xff0c;就大量应用的这…

北京大学纳家勇治研究组在《美国国家科学院院刊》发文阐明时序记忆的神经机制

来源&#xff1a; 北京大学心理与认知科学学院官网概要&#xff1a;记忆是核心认知功能之一&#xff0c;使得人和动物能够根据经验改变自己的行为模式。近日&#xff0c;《美国国家科学院院刊》&#xff08;Proceedings of the National Academy of Sciences of the United Sta…

管道( Pipeline )模型--示例

类图 时序图 阀门接口 /*** 阀门接口* author administrator**/ public interface Valve {public String getName();public void invoke( Context context ); }基本阀门 /*** 基础阀门* author administrator**/ public class BasicValve implements Valve{private String name…

一文告诉你,NIPS 2017有多火爆 | 附PPT、视频、代码大总结

原文来源&#xff1a;Medium、GitHub作者&#xff1a;TarasSereda「雷克世界」编译&#xff1a;嗯~阿童木呀、KABUDA今年的NIPS是一场盛大的、极富教育意义和探索精神的、魅力十足且人数众多的会议。第一步&#xff0c;登记排队量子计算机Tutorials深度学习&#xff1a;实践与趋…

不入oracle数据库,Oracle数据库之操作符及函数

一、操作符&#xff1a;1、分类&#xff1a;算术、比较、逻辑、集合、连接&#xff1b;2、算术操作符&#xff1a;执行数值计算&#xff1b;--工资加1000select empno,ename,job,sal1000 from emp;3、比较操作符&#xff1a;--比较运算符(between and包头不包尾)select * from …

Facebook面部识别新突破:可识别未标记照片中用户

来源&#xff1a;凤凰科技据科技博客TechCrunch报道&#xff0c;Facebook公司希望用户了解和掌控人们上传的照片&#xff0c;即便用户没有在照片中被标记出来。周二&#xff0c;Facebook推出了一项新的面部识别功能&#xff1a;照片检查&#xff08;Photo Review&#xff09;。…

德扑 AI 之父解答 Libratus 的13个疑问:没有用到任何深度学习,DL 远非 AI 的全部

来源&#xff1a; AI科技评论概要&#xff1a;卡耐基梅隆大学计算机系在读博士生 Noam Brown 和计算机系教授 Tuomas Sandholm 来到 reddit 的机器学习分版&#xff0c;和网友们一起来了一场「你问我答」&#xff08;ask me anything&#xff09;。卡耐基梅隆大学计算机系在读博…

如何测量智能产品的AI智商水平,论AI的三种智商 |未来研究

前言&#xff1a;本文是未来智能实验室关于人工智能智商的最新研究文章&#xff0c;主要提出智能系统的智能水平会因为测试目的的不同&#xff0c;产生三种不同的智商类型&#xff0c;针对这三种AI智商&#xff0c;本文也提出对应的测试方法和数学公式。相关英文论文与2017年12…

硅谷顶级VC:“S曲线”看四大风口,创企成功机会巨大

来源&#xff1a;全球技术地图新技术走入产业应用阶段&#xff0c;在完成了前期基础的试水后&#xff0c;大企业往往以雄厚的资本实力、强大的人才团队和广阔的市场资源&#xff0c;迅速占领新技术高地。那么顺应新技术而诞生的创业企业&#xff0c;还是否有打造成功企业的机会…

新型量子计算机首个基本元件问世,扩展性更强运算速度更快

来源&#xff1a;科技日报概要&#xff1a;最新研究证明了建造这种量子计算机的可行性&#xff0c;其有潜力克服目前的量子计算方法面临的扩展问题。据物理学家组织网近日报道&#xff0c;瑞典和奥地利物理学家携手&#xff0c;研制出了单量子比特里德伯&#xff08;Rydberg&am…

Strut2中单元测试实例

项目文件结构图&#xff1a; 椭圆框中的Jar 包是单元测试时候需要引入的。 矩形框 MainTest 每个包下一个&#xff0c;为 JUnit4 的 Suite 套件&#xff0c;其作用是执行本包下的“测试类”和子包的 MainTest。 例如&#xff1a;jp.co.snjp.ht.MainTest package jp.co.snjp.h…