public SqlSessionFactory build(InputStream inputStream, String environment, Properties properties) {
try {
//将输入流转换成XMLConfigBuilder对象。然后调用parse方法解析。最后采用默认的SqlSessionFactory来返回
XMLConfigBuilder parser = new XMLConfigBuilder(inputStream, environment, properties);
//XMLConfigBuilder.parse()将XML转换成了configuration对象。然后直接new一个DefaultSqlSessionFactory对象
return build(parser.parse());
} catch (Exception e) {
throw ExceptionFactory.wrapException("Error building SqlSession.", e);
} finally {
ErrorContext.instance().reset();
try {
inputStream.close();
} catch (IOException e) {
// Intentionally ignore. Prefer previous error.
}
}
}
public SqlSessionFactory build(Configuration config) {
return new DefaultSqlSessionFactory(config);
}
将XML转换成configuration对象
new DefaultSqlSessionFactory,将得到的configuration对象作为属性赋给DefaultSqlSessionFactory
public DefaultSqlSessionFactory(Configuration configuration) {
this.configuration = configuration;
}
private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {
Transaction tx = null;
try {
final Environment environment = configuration.getEnvironment();
final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);
tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
final Executor executor = configuration.newExecutor(tx, execType);
return new DefaultSqlSession(configuration, executor, autoCommit);
} catch (Exception e) {
closeTransaction(tx); // may have fetched a connection so lets call close()
throw ExceptionFactory.wrapException("Error opening session. Cause: " + e, e);
} finally {
ErrorContext.instance().reset();
}
}
当调用sqlSessionFactory.openSession()
获得一个sqlSession时,调用的就是上面的方法。从transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);
里面获得事务对象Transaction。从Configuration对象里面获得Executor对象
public Executor newExecutor(Transaction transaction, ExecutorType executorType) {
executorType = executorType == null ? defaultExecutorType : executorType;
executorType = executorType == null ? ExecutorType.SIMPLE : executorType;
Executor executor;
if (ExecutorType.BATCH == executorType) {
executor = new BatchExecutor(this, transaction);
} else if (ExecutorType.REUSE == executorType) {
executor = new ReuseExecutor(this, transaction);
} else {
//默认走的这个逻辑
executor = new SimpleExecutor(this, transaction);
}
if (cacheEnabled) {
executor = new CachingExecutor(executor);
}
executor = (Executor) interceptorChain.pluginAll(executor);
return executor;
}
默认情况下使用SimpleExecutor对象作为执行器
public SimpleExecutor(Configuration configuration, Transaction transaction) {
super(configuration, transaction);
}
接下来使用getMapper方法获得一个接口的代理类
@Override
public T getMapper(Class type) {
return configuration.getMapper(type, this);
}
//Configuration类
public T getMapper(Class type, SqlSession sqlSession) {
return mapperRegistry.getMapper(type, sqlSession);
}
1.使用SqlSession调用getMapper()方法获得Mapper对象。
2.调用MapperRegistry里面的getMapper()
在解析xml文件的时候就已经处理了标签。使用XMLMapperBuilder类解析的时候有一个方法parse()进行处理。会将所有的mapper注册到MapperRegistry里面。代码逻辑如下。实现方式很简单就是将mapper标签里面的namespace属性添加到集合里面
public void parse() {
if (!configuration.isResourceLoaded(resource)) {
//
//解析mapper节点
configurationElement(parser.evalNode("/mapper"));
configuration.addLoadedResource(resource);
//这里就是绑定mapper和class的地方
bindMapperForNamespace();
}
parsePendingResultMaps();
parsePendingChacheRefs();
parsePendingStatements();
}
private void bindMapperForNamespace() {
String namespace = builderAssistant.getCurrentNamespace();
if (namespace != null) {
Class boundType = null;
try {
boundType = Resources.classForName(namespace);
} catch (ClassNotFoundException e) {
//ignore, bound type is not required
}
if (boundType != null) {
if (!configuration.hasMapper(boundType)) {
// Spring may not know the real resource name so we set a flag
// to prevent loading again this resource from the mapper interface
// look at MapperAnnotationBuilder#loadXmlResource
configuration.addLoadedResource("namespace:" + namespace);
configuration.addMapper(boundType);
}
}
}
}
3.通过前面注册了class的全路径,使用mapperProxyFactory来创建mapper接口实现类。这里的mapperProxyFactory会为每一个mapper都对应一个mapperProxyFactory。因为configuration.addMapper(boundType);
里面使用map接口为每一个mapper都创建了一个工厂
public void addMapper(Class type) {
if (type.isInterface()) {
if (hasMapper(type)) {
throw new BindingException("Type " + type + " is already known to the MapperRegistry.");
}
boolean loadCompleted = false;
try {
//每次都new一个新的MapperProxyFactory放到map中
knownMappers.put(type, new MapperProxyFactory(type));
// It's important that the type is added before the parser is run
// otherwise the binding may automatically be attempted by the
// mapper parser. If the type is already known, it won't try.
MapperAnnotationBuilder parser = new MapperAnnotationBuilder(config, type);
parser.parse();
loadCompleted = true;
} finally {
if (!loadCompleted) {
knownMappers.remove(type);
}
}
}
}
//MapperRegistry
public T getMapper(Class type, SqlSession sqlSession) {
final MapperProxyFactory mapperProxyFactory = (MapperProxyFactory) knownMappers.get(type);
if (mapperProxyFactory == null) {
throw new BindingException("Type " + type + " is not known to the MapperRegistry.");
}
try {
return mapperProxyFactory.newInstance(sqlSession);
} catch (Exception e) {
throw new BindingException("Error getting mapper instance. Cause: " + e, e);
}
}
4.最后通过JDK的动态代理创建一个代理类交给开发者使用
//MapperProxyFactory
public T newInstance(SqlSession sqlSession) {
final MapperProxy mapperProxy = new MapperProxy(sqlSession, mapperInterface, methodCache);
return newInstance(mapperProxy);
}
protected T newInstance(MapperProxy mapperProxy) {
return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[]{mapperInterface}, mapperProxy);
}
5.开发者使用接口调用方法会执行代理类MapperProxy对象里面的invoke方法。调用invoke方法有点小巧妙。代码如下:
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if (Object.class.equals(method.getDeclaringClass())) {
try {
return method.invoke(this, args);
} catch (Throwable t) {
throw ExceptionUtil.unwrapThrowable(t);
}
}
final MapperMethod mapperMethod = cachedMapperMethod(method);
return mapperMethod.execute(sqlSession, args);
}
上面代码中可以看到如果是执行Object类的方式那么直接调用method.invoke。如果不是Object的方法,那么执行的是MapperMethod方法。我们知道一个接口没有实现是不能够被实例化的,并且我们在写接口时,确实没有给任何实现,那么Mybatis是怎么帮我们做事的呢?就是上面这段代码。首先通过代理类生成代理对象。当执行接口中的方法时,都是调用这个invoke方法,当你不是调用Object下面的方法,那么统一都执行MapperMethod方法来执行。
private MapperMethod cachedMapperMethod(Method method) {
MapperMethod mapperMethod = methodCache.get(method);
if (mapperMethod == null) {
//mapperInterface:表示调用接口,也就是mapper标签上面的namespace
//method:方法名称,也就是mapper文件里面的具体语句的id
//configuration对象。因为所有的配置文件最终都放到了configuration,所以再查询使用的参数,查询之后的返回值类型等数据TypeHandler等等都在configuration里面
mapperMethod = new MapperMethod(mapperInterface, method, sqlSession.getConfiguration());
methodCache.put(method, mapperMethod);
}
return mapperMethod;
}
mapperMethod持有接口名,方法名,configuration。而接口名对应mapper标签的namespace;方法名对应里面的id。configuration持有所有配置信息例如TypeHandler,TypeAlias等等。然后调用execute()方法执行
public Object execute(SqlSession sqlSession, Object[] args) {
Object result;
if (SqlCommandType.INSERT == command.getType()) {
Object param = method.convertArgsToSqlCommandParam(args);
result = rowCountResult(sqlSession.insert(command.getName(), param));
} else if (SqlCommandType.UPDATE == command.getType()) {
Object param = method.convertArgsToSqlCommandParam(args);
result = rowCountResult(sqlSession.update(command.getName(), param));
} else if (SqlCommandType.DELETE == command.getType()) {
Object param = method.convertArgsToSqlCommandParam(args);
result = rowCountResult(sqlSession.delete(command.getName(), param));
} else if (SqlCommandType.SELECT == command.getType()) {
if (method.returnsVoid() && method.hasResultHandler()) {
executeWithResultHandler(sqlSession, args);
result = null;
} else if (method.returnsMany()) {
result = executeForMany(sqlSession, args);
} else if (method.returnsMap()) {
result = executeForMap(sqlSession, args);
} else {
Object param = method.convertArgsToSqlCommandParam(args);
result = sqlSession.selectOne(command.getName(), param);
}
} else if (SqlCommandType.FLUSH == command.getType()) {
result = sqlSession.flushStatements();
} else {
throw new BindingException("Unknown execution method for: " + command.getName());
}
if (result == null && method.getReturnType().isPrimitive() && !method.returnsVoid()) {
throw new BindingException("Mapper method '" + command.getName()
+ " attempted to return null from a method with a primitive return type (" + method.getReturnType() + ").");
}
return result;
}
从上面的代码可以看出所有的所有的mapper对象实际都是MapperMethod对象,然后MapperMethod持有方法名『全类名.方法名』还有执行的操作『select;update;insert;delete』还有sql的各种参数方法签名。然后执行sql语句
你可能感兴趣的:(mybatis源码分析-mapper对象动态代理)
echarts开发图表时遇到问题:trigger: ‘item‘正常显示tooltip,trigger: ‘axis‘则无法显示tooltip,为什么?
思想永无止境
Web前端echartsvue.jsjavascript
原因:未知!解放方案:import{markRaw}from'vue';//解决办法,让echarts从监听对象变成普通对象letchart=markRaw(echarts.init(chartRef))//原代码//letchart=echarts.init(chartRef)
Vue入门常见指令
星星不打輰
前端vue.jsjavascript前端
Vue(视图信息与脚本数据双向绑定)Vue的基础入门Vue的实现流程:1、在HTML的body标签中引入需要通过Vue进行数据绑定的标签2、在HTML中的相关的视图标签中设置标签元素的id()3、通过外链的形式将Vue的核心库进行导入:4、在另外一个script中书写对应的Vue–JS脚本--(1:创建对应的Vue对象,并且进行实例化)--(2:指定Vue对象绑定的标签元素–》el:"#app"-
python面向对象编程总结:从基础到进阶的 OOP 核心思想与设计技巧
吴师兄大模型
python人工智能面向对象编程OOP开发语言编程PYTHON
Langchain系列文章目录01-玩转LangChain:从模型调用到Prompt模板与输出解析的完整指南02-玩转LangChainMemory模块:四种记忆类型详解及应用场景全覆盖03-全面掌握LangChain:从核心链条构建到动态任务分配的实战指南04-玩转LangChain:从文档加载到高效问答系统构建的全程实战05-玩转LangChain:深度评估问答系统的三种高效方法(示例生成、手
java8 list分组
魏 无羡
listwindowsjava
在Java8中,可以使用StreamAPI和Collectors.groupingBy对List进行分组。以下是常见的分组场景和实现方法:1.基础分组(按对象属性分组)根据对象的某个字段分组(例如按category分组):importjava.util.List;importjava.util.Map;importjava.util.stream.Collectors;classProduct{p
04 NodeJs访问MySql 数据库
依旧阳光的老码农
NodeJs入门数据库mysql
在NodeJs中访问Mysql数据库一、MySql数据库中访问数据库形式:在Node.js中访问MySQL数据库有多种形式,以下是其中的一些常见方式:原生MySQL驱动:使用Node.js的原生MySQL驱动来连接和操作数据库。这是最早的方式,但在许多现代应用中,由于其复杂性,它可能不是首选。SequelizeORM:Sequelize是一个流行的Node.jsORM(对象关系映射)库,它允许你使
hasOwnProperty是什么?【JavaScript方法】
幸运小圣
JavaScriptjavascript开发语言
hasOwnPropertyhasOwnProperty是什么?用于判断一个对象是否包含特定的自身属性。它只会返回对象自身具有的属性,而不会返回从原型链上继承的属性。obj.hasOwnProperty(prop)obj是需要检查的对象。prop是需要检查是否存在的属性名(字符串)。返回值:如果对象obj具有名为prop的自身属性,则返回true。否则返回false。constobj={name:
【python】协程(coroutine)
atwdy
Pythonpythoncoroutine协程asyncio
协程(coroutine)可以理解为一个可以中途暂停保存当前执行状态信息并可以从此处恢复执行的函数,多个协程共用一个线程执行,适合执行需要“等待”的任务。所以严格意义上,多个协程同一时刻也只有一个在真正的执行,因为线程是任务调度的基本单位。注意这里的执行指的是占用CPU计算,不包括等待阻塞等场景。python中实现协程的标准库是asyncio,标准库的实现中大致可以由底层到顶层分为下面几个对象:c
PySide(PyQT),怎样在非QObject对象上应用信号槽框架
深蓝海拓
pyside6系统学习pyside6学习笔记数据库
在PySide(以及PyQt)中,信号和槽机制是基于QObject类实现的。只有继承自QObject的类才能够定义和使用信号与槽。那么如果画面中的对象不是QObject类,就无法使用信号和槽。比如下面的代码就会出错:importsysfromPySide6.QtCoreimportSignalfromPySide6.QtWidgetsimportQApplication,QGraphicsScen
2025年度支持中关村特色产业园区提升发展资金项目申报指南
项目申报-华夏泰科
人工智能大数据服务器安全运维
为进一步支持中关村特色产业园,依据《中关村国家自主创新示范区促进园区高质量发展支持资金管理办法》(京科发〔2024〕16号)“第七条支持特色产业园区提升发展”有关规定,现开展2025年度支持特色产业园区提升发展资金项目的申报工作。一、支持对象中关村特色产业园区运营单位。二、支持内容支持围绕园区产业和企业创新发展需求,编制园区发展规划、打造高水平专业化运营团队、建设专业技术服务平台、承接科技成果转化
Java中getter和setter方法的作用
以恒1
java开发语言
Java中getter和setter方法的作用在Java中,getter和setter方法是面向对象编程中实现封装的核心手段,它们的作用不仅限于访问或修改对象的属性,还为代码的灵活性、安全性和可维护性提供了重要保障。以下是其核心作用的详细说明:idea快捷键:alt加insert加enter,选择getter和setter方法即可一、核心作用1.实现封装(Encapsulation)隐藏内部实现:
Office组件无法正常使用的解决方法
weixin_30788731
问题与现象开发时调用Office组件,代码编译是通过的,但在运行时当ApplicationClass对象初始化后程序出现异常。异常信息如下:无法将类型为“Microsoft.Office.Interop.Excel.ApplicationClass”的COM对象强制转换为接口类型“Microsoft.Office.Interop.Excel._Application”。此操作失败的原因是对IID为
cpp智能指针
xianwu543
c++开发语言网络mysql数据库
普通指针的不足new和new[]的内存需要用delete和deletel]释放。程序员的主观失误,忘了或漏了释放。程序员也不确定何时释放。普通指针的释放类内的指针,在析构函数中释放。C++内置数据类型,如何释放?new出来的类,本身如何释放?C++11新增三个智能指针类型unique_ptrshared_ptrweak_ptr一、智能指针unique_ptrunique_ptr独享它指向的对象,也
单片机c51语言程序设计基础,单片机C语言程序设计基础知识全解析
咪马3213m~~
单片机c51语言程序设计基础
标识符和关键字(一)标识符标识符是用来表示源程序中自定义对象名称的符号。其中的自定义对象可以是常量、变量、数组、结构、语句标号以及函数等。在C51语言中,标识符可以由字母(a~z,A~Z)、数字(0~9)和下划线“_”组成,最多可支持32个字符。C51标识符的定义不是随意的,应遵循“简洁”和“见名知意”的原则,并需要符合一定的规则:➢标识符的第一个字符必须是字母或者下划线,不能为数字。由于有些编译
Mybatis xml 无法正常解析 Invalid bound statement
javaspringboot
问题背景使用插件mybatisX生成的mapper.xml没有办法被正常解析,项目使用的是nacos作为配置中心;问题定位首先看了一下网上说的常见几种问题,比如id不匹配,或者说pom里缺少resource配置导致target里没有xml;这些都没有出现;那接下来就只能从源码出发解决问题;通过报错的堆栈信息我发现问题的出发点是在这里publicSqlCommand(Configurationcon
Java List集合取交集的五种方式
偏爱星雾缠绕15
javalist开发语言
在Java中,List集合是用于存储一系列对象的数据结构。当我们要获取两个List集合的交集时,有多种方法可以实现。下面将介绍几种常见的方式。1.使用Java8的StreamAPIJava8引入了StreamAPI,它提供了一种声明式的方式来处理数据。importjava.util.Arrays;importjava.util.List;importjava.util.stream.Collect
QT输入窗口部件之QSlider(滑动条)
点灯大神
QTqt开发语言
QSlider如果对您有用记得点赞哦)QSlider继承QAbstractSlider。QScrollBar类提供垂直或水平滑动条小部件,滑动条是用于控制有界值的典型小部件。它允许用户沿着水平或垂直凹槽移动滑块手柄,并将手柄的位置转换为合法范围内的整数值。例子创建水平和垂直滑动条,创建两个QSlider对象,一个是水平滑动条,另一个是垂直滑动条;用一个Label来显示当前水平或垂直滑动条当前的值。
利用Java的反射和动态代理实现插件化架构
Coder_Kevin_Vans
架构jvm
插件化架构是一种将应用程序的功能模块化并以插件形式加载和运行的架构模式。利用Java的反射和动态代理可以实现插件化架构的核心功能,包括插件的加载、实例化、调用和卸载等。下面是一种基本的实现思路:1.定义插件接口:首先定义一个插件接口,用于规范插件的功能和调用方法。2.插件的实现:编写插件实现类,实现插件接口,并提供相应的功能实现。3.插件加载器:编写插件加载器,用于加载和管理插件。插件加载器可以扫
java构造方法,构造代码块,静态代码块的执行顺序
后端java
构造方法,构造代码块,静态代码块的执行顺序静态代码块:用static声明,jvm加载类时执行,仅执行一次类中直接用{}定义,每次创建对象时执行同级别的变量(static和普通变量)的执行顺序由代码执行顺序决定(从上到下也就是代码书写顺序)代码解读复制代码packagecom.rumenz;publicclassRumenz{privateIntegerid=100;privatestaticStr
万字总结!常见分布式ID解决方案(数据库、算法、开源组件)
Java爱好狂.
wpfjava后端
分布式ID分布式ID(DistributedID)是指在分布式系统中生成全局唯一的标识符,用于标识不同实体或数据对象。在分布式系统中,由于数据存储、计算和处理都分散在不同的节点上,因此需要一个可靠的方式来跟踪和标识这些数据对象。分布式ID最低要求:erlang复制代码全局唯一:ID的全局唯一性肯定是首先要满足的高性能:分布式ID的生成速度要快,对本地资源消耗要小高可用:生成分布式ID的服务要保证可
C# 从字符串中分离文件路径、文件名及扩展名
PfCoder
C#c#开发语言
C#从字符串中分离文件路径、文件名及扩展名对文件进行操作时,首先要获取文件路径信息,然后创建文件对象,通过IO流将数据读取大宋内存中并进行处理。在操作文件时,可能还需要提取文件的一些信息,比如,文件路径,文件名,扩展名等等,实例如下:主要用到打开文件选择对话框,可以选择一个或多个文件,使用需引入命名空间:Microsoft.Win32,以及对获取的文件路径进行截取Substring方法,代码如下p
web框架封装
进阶的DW
前端
一、po模式po模式全程pageobjectmodule,页面对象模型,ui层自动化的最佳实践。核心思想是将一个页面封装成独立对象,这个对象蕴含了该页面常见的核心元素操作,在测试用例层根据业务流程,组装不同的页面对象来实现自动化操作,并完成断言好处:1、如果元素发生变更,只需要修改page层2、如果业务流程发生变化,只需要更改业务业务层二、搭建po模式的框架1、搭建框架page:page层,这是一
springboot中Webclient对象怎么禁止缓存、保证每次数据都属来源后端
阿里嘎多学长
springboot缓存数据库
在SpringBoot中,使用WebClient对象进行HTTP请求时,可以通过配置请求头来禁止缓存,确保每次请求都能从后端获取最新的数据。以下是一些常用的方法来实现这一目的:设置Cache-Control请求头:在发起请求时,可以设置Cache-Control请求头为no-cache,这会告诉服务器和任何中间缓存,不应使用缓存的响应来满足请求。WebClientwebClient=WebClie
C语言、C++和C#的区别在什么地方?
破碎的天堂鸟
学习教程c语言c++开发语言
C语言、C++和C#是三种不同的编程语言,它们在设计目标、特性和应用领域上有各自的特点。C语言是一种过程式编程语言,C++是一种多范式编程语言,支持过程式编程和面向对象编程,而C#是一种高级的面向对象编程语言,主要面向.NET框架。以下是C语言、C++和C#的对比表格:对比维度C语言C++C#设计目标过程式编程语言多范式编程语言面向对象编程语言编程范式过程式过程式和面向对象面向对象内存管理手动管理
c++和c#和c语言
Random_N1
c++c#c语言
C++、C#和C语言之间有关系,但它们在设计目标、应用领域和语法特性上也有显著的区别。以下是它们之间的关系和主要区别:关系C语言:基础:C语言是一种通用的过程式编程语言,开发于20世纪70年代,用于系统编程和应用程序开发。它为其他许多现代编程语言提供了基础。C++:扩展:C++是BjarneStroustrup在C语言的基础上开发的,添加了面向对象编程、泛型编程和其他高级编程特性。C++兼容C语言
RMAN备份bug-审计日志暴涨(select action from gv$session)
CodeBe@r
#Oracleoraclebug审计日志
问题概述/oracle文件系统使用率过大,经过检查是审计日志过大,/oracle目录197G审计日志占用70G,每6个小时产生大量审计日志,日志内容全是'selectactionfromgv$session',猜测可能跟备份有关,$>df-h/oracleFilesystemSizeUsedAvailUse%Mountedon/dev/mapper/fusioncube-oracle197G164
【C++】异常
XiYang-DING
#C++c++异常
异常一、C++异常概念二、异常的用法2.1异常的抛出和捕获1.异常的抛出2.异常的匹配3.异常对象的拷贝4.`catch(...)`捕获所有异常5.派生类异常与基类捕获6.栈展开(StackUnwinding)2.2重新抛出2.3异常安全1.构造函数与异常2.析构函数与异常3.资源管理与异常4.异常安全级别2.4异常规范1.异常规格说明2.`throw()`表示不抛异常3.**无异常接口声明**4
Vue的几个常用难点
亭台烟雨中
vue前端
一、window,open()配合this.$router使用1、参数发送页//待传递的参数varapiTemp={}apiTemp.testName=row.testNameapiTemp.testClass=row.testClassapiTemp.testCode=row.testCode//与地址绑定,生成对象constrouteDate=this.$router.resolve({pat
8种Vue中数据更新了但页面没有更新的情况
cypking
vue.jsjavascript前端
文章目录1、Vue无法检测实例被创建时不存在于data中的属性2、Vue无法检测‘对象属性’的添加或移除3、Vue不能检测利用数组索引直接修改一个数组项4、Vue不能监测直接修改数组长度的变化5、在异步更新执行之前操作DOM数据不会变化6、循环嵌套层级太深,视图不更新?7、路由参数变化时,页面不更新(数据不更新)8、使用keep-alive之后数据无法实时更新问题1、Vue无法检测实例被创建时不存
Windows 图形显示驱动开发-WDDM 3.2-本机 GPU 围栏对象(四)
程序员王马
windows图形显示驱动开发驱动开发单片机stm32
用于创建、打开和销毁本机围栏的D3DKMT内核API引入了以下D3DKMT内核模式API来创建和打开本机围栏对象。D3DKMTCreateNativeFence/D3DKMT_CREATENATIVEFENCED3DKMTOpenNativeFenceFromNTHandle/D3DKMT_OPENNATIVEFENCEFROMNTHANDLEDxgkrnl调用现有的D3DKMTDestroySy
C# 设计模式之观察者模式
鲤籽鲲
C#c#设计模式观察者模式
总目录前言在现实生活中,处处可见观察者模式,例如,微信中的订阅号,只要对订阅号进行关注的客户端,如果订阅号有什么更新,就会直接推送给订阅了的用户。这就是观察者模式的一种应用。1基础介绍观察者模式定义了对象之间的一种一对多的依赖关系,使得当一个对象状态发生改变时,它的所有依赖者都能够得到相应的通知并作出相应的反应。观察者模式也被称为发布-订阅模式。观察者模式定义了一种一对多的依赖关系,让多个观察者对
如何用ruby来写hadoop的mapreduce并生成jar包
wudixiaotie
mapreduce
ruby来写hadoop的mapreduce,我用的方法是rubydoop。怎么配置环境呢:
1.安装rvm:
不说了 网上有
2.安装ruby:
由于我以前是做ruby的,所以习惯性的先安装了ruby,起码调试起来比jruby快多了。
3.安装jruby:
rvm install jruby然后等待安
java编程思想 -- 访问控制权限
百合不是茶
java访问控制权限单例模式
访问权限是java中一个比较中要的知识点,它规定者什么方法可以访问,什么不可以访问
一:包访问权限;
自定义包:
package com.wj.control;
//包
public class Demo {
//定义一个无参的方法
public void DemoPackage(){
System.out.println("调用
[生物与医学]请审慎食用小龙虾
comsci
生物
现在的餐馆里面出售的小龙虾,有一些是在野外捕捉的,这些小龙虾身体里面可能带有某些病毒和细菌,人食用以后可能会导致一些疾病,严重的甚至会死亡.....
所以,参加聚餐的时候,最好不要点小龙虾...就吃养殖的猪肉,牛肉,羊肉和鱼,等动物蛋白质
org.apache.jasper.JasperException: Unable to compile class for JSP:
商人shang
maven2.2jdk1.8
环境: jdk1.8 maven tomcat7-maven-plugin 2.0
原因: tomcat7-maven-plugin 2.0 不知吃 jdk 1.8,换成 tomcat7-maven-plugin 2.2就行,即
<plugin>
你的垃圾你处理掉了吗?GC
oloz
GC
前序:本人菜鸟,此文研究学习来自网络,各位牛牛多指教
1.垃圾收集算法的核心思想
Java语言建立了垃圾收集机制,用以跟踪正在使用的对象和发现并回收不再使用(引用)的对象。该机制可以有效防范动态内存分配中可能发生的两个危险:因内存垃圾过多而引发的内存耗尽,以及不恰当的内存释放所造成的内存非法引用。
垃圾收集算法的核心思想是:对虚拟机可用内存空间,即堆空间中的对象进行识别
shiro 和 SESSSION
杨白白
shiro
shiro 在web项目里默认使用的是web容器提供的session,也就是说shiro使用的session是web容器产生的,并不是自己产生的,在用于非web环境时可用其他来源代替。在web工程启动的时候它就和容器绑定在了一起,这是通过web.xml里面的shiroFilter实现的。通过session.getSession()方法会在浏览器cokkice产生JESSIONID,当关闭浏览器,此
移动互联网终端 淘宝客如何实现盈利
小桔子
移動客戶端淘客淘寶App
2012年淘宝联盟平台为站长和淘宝客带来的分成收入突破30亿元,同比增长100%。而来自移动端的分成达1亿元,其中美丽说、蘑菇街、果库、口袋购物等App运营商分成近5000万元。 可以看出,虽然目前阶段PC端对于淘客而言仍旧是盈利的大头,但移动端已经呈现出爆发之势。而且这个势头将随着智能终端(手机,平板)的加速普及而更加迅猛
wordpress小工具制作
aichenglong
wordpress小工具
wordpress 使用侧边栏的小工具,很方便调整页面结构
小工具的制作过程
1 在自己的主题文件中新建一个文件夹(如widget),在文件夹中创建一个php(AWP_posts-category.php)
小工具是一个类,想侧边栏一样,还得使用代码注册,他才可以再后台使用,基本的代码一层不变
<?php
class AWP_Post_Category extends WP_Wi
JS微信分享
AILIKES
js
// 所有功能必须包含在 WeixinApi.ready 中进行
WeixinApi.ready(function(Api) {
// 微信分享的数据
var wxData = {
&nb
封装探讨
百合不是茶
JAVA面向对象 封装
//封装 属性 方法 将某些东西包装在一起,通过创建对象或使用静态的方法来调用,称为封装;封装其实就是有选择性地公开或隐藏某些信息,它解决了数据的安全性问题,增加代码的可读性和可维护性
在 Aname类中申明三个属性,将其封装在一个类中:通过对象来调用
例如 1:
//属性 将其设为私有
姓名 name 可以公开
jquery radio/checkbox change事件不能触发的问题
bijian1013
JavaScriptjquery
我想让radio来控制当前我选择的是机动车还是特种车,如下所示:
<html>
<head>
<script src="http://ajax.googleapis.com/ajax/libs/jquery/1.7.1/jquery.min.js" type="text/javascript"><
AngularJS中安全性措施
bijian1013
JavaScriptAngularJS安全性XSRFJSON漏洞
在使用web应用中,安全性是应该首要考虑的一个问题。AngularJS提供了一些辅助机制,用来防护来自两个常见攻击方向的网络攻击。
一.JSON漏洞
当使用一个GET请求获取JSON数组信息的时候(尤其是当这一信息非常敏感,
[Maven学习笔记九]Maven发布web项目
bit1129
maven
基于Maven的web项目的标准项目结构
user-project
user-core
user-service
user-web
src
【Hive七】Hive用户自定义聚合函数(UDAF)
bit1129
hive
用户自定义聚合函数,用户提供的多个入参通过聚合计算(求和、求最大值、求最小值)得到一个聚合计算结果的函数。
问题:UDF也可以提供输入多个参数然后输出一个结果的运算,比如加法运算add(3,5),add这个UDF需要实现UDF的evaluate方法,那么UDF和UDAF的实质分别究竟是什么?
Double evaluate(Double a, Double b)
通过 nginx-lua 给 Nginx 增加 OAuth 支持
ronin47
前言:我们使用Nginx的Lua中间件建立了OAuth2认证和授权层。如果你也有此打算,阅读下面的文档,实现自动化并获得收益。SeatGeek 在过去几年中取得了发展,我们已经积累了不少针对各种任务的不同管理接口。我们通常为新的展示需求创建新模块,比如我们自己的博客、图表等。我们还定期开发内部工具来处理诸如部署、可视化操作及事件处理等事务。在处理这些事务中,我们使用了几个不同的接口来认证:
&n
利用tomcat-redis-session-manager做session同步时自定义类对象属性保存不上的解决方法
bsr1983
session
在利用tomcat-redis-session-manager做session同步时,遇到了在session保存一个自定义对象时,修改该对象中的某个属性,session未进行序列化,属性没有被存储到redis中。 在 tomcat-redis-session-manager的github上有如下说明: Session Change Tracking
As noted in the &qu
《代码大全》表驱动法-Table Driven Approach-1
bylijinnan
java算法
关于Table Driven Approach的一篇非常好的文章:
http://www.codeproject.com/Articles/42732/Table-driven-Approach
package com.ljn.base;
import java.util.Random;
public class TableDriven {
public
Sybase封锁原理
chicony
Sybase
昨天在操作Sybase IQ12.7时意外操作造成了数据库表锁定,不能删除被锁定表数据也不能往其中写入数据。由于着急往该表抽入数据,因此立马着手解决该表的解锁问题。 无奈此前没有接触过Sybase IQ12.7这套数据库产品,加之当时已属于下班时间无法求助于支持人员支持,因此只有借助搜索引擎强大的
java异常处理机制
CrazyMizzz
java
java异常关键字有以下几个,分别为 try catch final throw throws
他们的定义分别为
try: Opening exception-handling statement.
catch: Captures the exception.
finally: Runs its code before terminating
hive 数据插入DML语法汇总
daizj
hiveDML数据插入
Hive的数据插入DML语法汇总1、Loading files into tables语法:1) LOAD DATA [LOCAL] INPATH 'filepath' [OVERWRITE] INTO TABLE tablename [PARTITION (partcol1=val1, partcol2=val2 ...)]解释:1)、上面命令执行环境为hive客户端环境下: hive>l
工厂设计模式
dcj3sjt126com
设计模式
使用设计模式是促进最佳实践和良好设计的好办法。设计模式可以提供针对常见的编程问题的灵活的解决方案。 工厂模式
工厂模式(Factory)允许你在代码执行时实例化对象。它之所以被称为工厂模式是因为它负责“生产”对象。工厂方法的参数是你要生成的对象对应的类名称。
Example #1 调用工厂方法(带参数)
<?phpclass Example{
mysql字符串查找函数
dcj3sjt126com
mysql
FIND_IN_SET(str,strlist)
假如字符串str 在由N 子链组成的字符串列表strlist 中,则返回值的范围在1到 N 之间。一个字符串列表就是一个由一些被‘,’符号分开的自链组成的字符串。如果第一个参数是一个常数字符串,而第二个是type SET列,则 FIND_IN_SET() 函数被优化,使用比特计算。如果str不在strlist 或st
jvm内存管理
easterfly
jvm
一、JVM堆内存的划分
分为年轻代和年老代。年轻代又分为三部分:一个eden,两个survivor。
工作过程是这样的:e区空间满了后,执行minor gc,存活下来的对象放入s0, 对s0仍会进行minor gc,存活下来的的对象放入s1中,对s1同样执行minor gc,依旧存活的对象就放入年老代中;
年老代满了之后会执行major gc,这个是stop the word模式,执行
CentOS-6.3安装配置JDK-8
gengzg
centos
JAVA_HOME=/usr/java/jdk1.8.0_45
JRE_HOME=/usr/java/jdk1.8.0_45/jre
PATH=$PATH:$JAVA_HOME/bin:$JRE_HOME/bin
CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar:$JRE_HOME/lib
export JAVA_HOME
【转】关于web路径的获取方法
huangyc1210
Web路径
假定你的web application 名称为news,你在浏览器中输入请求路径: http://localhost:8080/news/main/list.jsp 则执行下面向行代码后打印出如下结果: 1、 System.out.println(request.getContextPath()); //可返回站点的根路径。也就是项
php里获取第一个中文首字母并排序
远去的渡口
数据结构PHP
很久没来更新博客了,还是觉得工作需要多总结的好。今天来更新一个自己认为比较有成就的问题吧。 最近在做储值结算,需求里结算首页需要按门店的首字母A-Z排序。我的数据结构原本是这样的:
Array
(
[0] => Array
(
[sid] => 2885842
[recetcstoredpay] =&g
java内部类
hm4123660
java内部类匿名内部类成员内部类方法内部类
在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.class文件,但是前面冠以外部类的类名和$符号。内部类可以间接解决多继承问题,可以使用内部类继承一个类,外部类继承一个类,实现多继承。
&nb
Caused by: java.lang.IncompatibleClassChangeError: class org.hibernate.cfg.Exten
zhb8015
maven pom.xml关于hibernate的配置和异常信息如下,查了好多资料,问题还是没有解决。只知道是包冲突,就是不知道是哪个包....遇到这个问题的分享下是怎么解决的。。
maven pom:
<dependency>
<groupId>org.hibernate</groupId>
<ar
Spark 性能相关参数配置详解-任务调度篇
Stark_Summer
sparkcachecpu任务调度yarn
随着Spark的逐渐成熟完善, 越来越多的可配置参数被添加到Spark中来, 本文试图通过阐述这其中部分参数的工作原理和配置思路, 和大家一起探讨一下如何根据实际场合对Spark进行配置优化。
由于篇幅较长,所以在这里分篇组织,如果要看最新完整的网页版内容,可以戳这里:http://spark-config.readthedocs.org/,主要是便
css3滤镜
wangkeheng
htmlcss
经常看到一些网站的底部有一些灰色的图标,鼠标移入的时候会变亮,开始以为是js操作src或者bg呢,搜索了一下,发现了一个更好的方法:通过css3的滤镜方法。
html代码:
<a href='' class='icon'><img src='utv.jpg' /></a>
css代码:
.icon{-webkit-filter: graysc
按字母分类:
ABCDEFGHIJKLMNOPQRSTUVWXYZ其他
首页 -
关于我们 -
站内搜索 -
Sitemap -
侵权投诉
版权所有 IT知识库 CopyRight © 2000-2050 E-COM-NET.COM , All Rights Reserved.