上篇文章分析Mapper的查询操作最终都会调用SqlSession的selectList方法,接下来几篇文章分析一下DefaultSqlSession的selectList的执行过程。
public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) { try { MappedStatement ms = configuration.getMappedStatement(statement); return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER); } catch (Exception e) { throw ExceptionFactory.wrapException("Error querying database. Cause: " + e, e); } finally { ErrorContext.instance().reset(); } }
首先调用DefaultSqlSession内的executor对象,如果开启了二级缓存executor就是类CachingExecutor的实例,然后调用executor的query方法其中第二个参数为我们在上篇文章提到过被转换的参数对象可能是null,ParamMap或一个对象,如果这个对象是Collection或数组则会再将这个参数对象转换为一个Map对象。
private Object wrapCollection(final Object object) { if (object instanceof Collection) { StrictMap map = new StrictMap(); map.put("collection", object); if (object instanceof List) { map.put("list", object); } return map; } else if (object != null && object.getClass().isArray()) { StrictMap map = new StrictMap(); map.put("array", object); return map; } return object; } 回过头来再看CachingExecutor的query方法: public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException { BoundSql boundSql = ms.getBoundSql(parameterObject); CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql); return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); } 由于这个query方法比较复杂,这篇我先分析获取BoundSql对象也就是MappedStatement的getBoundSql方法。BoundSql内包含了执行的sql语句和命名参数。 public class BoundSql { private final String sql; private final List parameterMappings; private final Object parameterObject; private final Map, Object> additionalParameters; private final MetaObject metaParameters; 我画了一副时序图来说明获取BoundSql的过程。 下面按时序图的步骤查看源代码。 1.mapperStatement的getBoundSql方法会调用其成员变量sqlSource的getBoundSql方法 public BoundSql getBoundSql(Object parameterObject) { DynamicContext context = new DynamicContext(configuration, parameterObject); rootSqlNode.apply(context); SqlSourceBuilder sqlSourceParser = new SqlSourceBuilder(configuration); Class parameterType = parameterObject == null ? Object.class : parameterObject.getClass(); SqlSource sqlSource = sqlSourceParser.parse(context.getSql(), parameterType, context.getBindings()); BoundSql boundSql = sqlSource.getBoundSql(parameterObject); for (Map.Entry, Object> entry : context.getBindings().entrySet()) { boundSql.setAdditionalParameter(entry.getKey(), entry.getValue()); } return boundSql; } DynamicSqlSource会创建一个DynamicContext对象,这个对象作为SqlNode的apply方法的入参,SqlNode每调用一次apply方法,就是将此节点含有的sql片段拼接到context中去,此外还包含了参数对象,通过MetaObject(本文最后介绍)可以使用A.B[0].C的形式 2.DynamicSqlSource所持有的SqlNode为MixedSqlNode,它内部的contents对象就是顶级节点下的兄弟sql片段,依次调用兄弟片段的apply方法拼接sql,当然其中每个兄弟sql片段下还有可能为多个sql片段。假如一个sql片段是一个不含任何可变参数的那么它就是StaticTextSqlNode,静态节点就是单纯的拼接一下sql,当然还有其他种类的SqlNode,譬如WhereSqlNode,看过它的代码我们就可以看出对于where后第一个and或or是不会拼入sql中的,我们这里就不一一列举了。 public boolean apply(DynamicContext context) { for (SqlNode sqlNode : contents) {//MixedSqlNode sqlNode.apply(context); } return true; } public boolean apply(DynamicContext context) {//StaticTextSqlNode context.appendSql(text); return true; } 3.此时context中已经含有完整的sql语句了只不过还没有替换掉mybatis的变量占位符信息(类似#{prop,jdbcType=...})。下一步通过SqlSourceBuilder的parse方法将sql语句转换为标准的带有Jdbc站位符?的语句,并将变量占位符信息抽像为ParameterMapping。返回一个包含sql和List的StaticSqlSource。 public SqlSource parse(String originalSql, Class parameterType, Map, Object> additionalParameters) { ParameterMappingTokenHandler handler = new ParameterMappingTokenHandler(configuration, parameterType, additionalParameters); GenericTokenParser parser = new GenericTokenParser("#{", "}", handler); String sql = parser.parse(originalSql); return new StaticSqlSource(configuration, sql, handler.getParameterMappings()); } GenericTokenParser将#{}中的内容过滤出来,使用ParameterMappingTokenHandler将其转换为ParameterMapping对象,然后用?将#{}替换。 4.通过以上步骤将一个DynamicSqlSource转换成了一个含有List的StaticSqlSource,调用getBoundSql方法返回BoundSql对象。 public BoundSql getBoundSql(Object parameterObject) { return new BoundSql(configuration, sql, parameterMappings, parameterObject); } MetaObject介绍 上面第一步中说到MetaObject可以将参数对象对外提供A.B[0].C的访问形式,那么他是怎么做到的呢。 Configuration的newMetaObject方法调用MetaObject.forObject()方法将自身持有的DefaultObjectFactory,DefaultObjectWrapperFactory和DefaultReflectorFactory一并传入。 public MetaObject newMetaObject(Object object) { return MetaObject.forObject(object, objectFactory, objectWrapperFactory, reflectorFactory); } public static MetaObject forObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory, ReflectorFactory reflectorFactory) { if (object == null) { return SystemMetaObject.NULL_META_OBJECT; } else { return new MetaObject(object, objectFactory, objectWrapperFactory, reflectorFactory); } } private MetaObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory, ReflectorFactory reflectorFactory) { this.originalObject = object; this.objectFactory = objectFactory; this.objectWrapperFactory = objectWrapperFactory; this.reflectorFactory = reflectorFactory; if (object instanceof ObjectWrapper) { this.objectWrapper = (ObjectWrapper) object; } else if (objectWrapperFactory.hasWrapperFor(object)) { this.objectWrapper = objectWrapperFactory.getWrapperFor(this, object); } else if (object instanceof Map) { this.objectWrapper = new MapWrapper(this, (Map) object); } else if (object instanceof Collection) { this.objectWrapper = new CollectionWrapper(this, (Collection) object); } else { this.objectWrapper = new BeanWrapper(this, object); } } 然后MetaObject通过构造方法将这些参数保留下来,构造函数对不同的object类型创建不同的ObjectWrapper对象,MetaObject通过ObjectWrapper对object提供了方便的多层级的getter和setter方法。 MetaObject对外提供属性的访问是getValue方法: public Object getValue(String name) { PropertyTokenizer prop = new PropertyTokenizer(name); if (prop.hasNext()) { MetaObject metaValue = metaObjectForProperty(prop.getIndexedName()); if (metaValue == SystemMetaObject.NULL_META_OBJECT) { return null; } else { return metaValue.getValue(prop.getChildren()); } } else { return objectWrapper.get(prop); } } public MetaObject metaObjectForProperty(String name) { Object value = getValue(name); return MetaObject.forObject(value, objectFactory, objectWrapperFactory, reflectorFactory); } 1.根据PropertyTokenizer判断是否是一个多层级的属性访问,如果是则先取顶层对象将其重新包裹为一个MetaObject对象,然后对这个对象再次调用getValue方法,这个时候getValue的参数为去除第一层级的字符串了,直到入参name为单层级的字符串也是就不带"."的字符串然后调用objectWrapper的get方法取出name所对应的属性值。这里我们以BeanWrapper为例看看这个方法的实现。 public Object get(PropertyTokenizer prop) { if (prop.getIndex() != null) { Object collection = resolveCollection(prop, object); return getCollectionValue(prop, collection); } else { return getBeanProperty(prop, object); } }首先判断当前访问的属性是否含有索引值也即是collection[key]这种形式,如果有则先得到这个集合然后在对集合(或Map或数组)按索引取值,如果不是类集合类型则使用metaClass使用getGetInvoker调用对象的getter方法取值。MetaClass内部使用的是反射技术,有兴趣的可以看看这个类的实现。 你可能感兴趣的:(Mybatis源码分析) mybatis源码分析-资源加载-下篇 cjxz 处理mapper节点构造函数中已经有很多很多默认类型匹配。这就是为什么在写sql的时候返回类型会自动映射到相应的java类型上面,这里已经处理好了。继续看最复杂的mapper在上面处理configuration节点的最后一句mapperElement(root.evalNode("mappers"));。这个是配置文件里面最复杂的,所以再处理上面Mybatis多写了两个类专门处理mapper数据X 2.10、mybatis源码分析之sql执行过程以select为例 小manong 在研究select执行过程之前先来介绍一个重要的类一、SqlNode和SqlSource在myabtis初始化过程中可以知道,映射配置文件中的sql节点会被解析为MappedStatement对象,其中sql语句解析成SqlSource对象,sql语句中定义的Sql节点、文本节点等,则由SqlNode接口的响应实现。1、SqlSource接口SqlSource接口结构publicinterface Mybatis系列-tkmybatis源码分析-01-mybatis、mybatis-spring和tkmybatis的关系 tinygodd MybatisspringboottkMybatismybatismybatis-spring mybatis、mybatis-spring和tkmybatis的关系定义MyBatis是一款优秀的持久层框架,它支持自定义SQL、存储过程以及高级映射。MyBatis免除了几乎所有的JDBC代码以及设置参数和获取结果集的工作。MyBatis可以通过简单的XML或注解来配置和映射原始类型、接口和JavaPOJO(PlainOldJavaObjects,普通老式Java对象)为数据库中的记录。MyB MyBatis 源码分析(五):异常模块 小徐很努力 源码解读mybatisjava开发语言 1、前言上一篇我们解了Mybatis解析器模块,本篇我们来了解反射模块。本文,我们来分享MyBatis的异常模块。对应exceptions包,如下图所示:在MyBatis源码分析(二):项目结构中,简单介绍了这个模块:定义了MyBatis专有的PersistenceException和TooManyResultsException异常。实际上,MyBatis不仅仅在exceptions包下有异常, MyBatis源码分析(六):数据源模块 小徐很努力 源码解读mybatis 1.概述本文,我们来分享MyBatis的数据源模块,对应datasource包。如下图所示:在MyBatis源码分析(二):项目结构中,简单介绍了这个模块如下:数据源是实际开发中常用的组件之一。现在开源的数据源都提供了比较丰富的功能,例如,连接池功能、检测连接状态等,选择性能优秀的数据源组件对于提升ORM框架乃至整个应用的性能都是非常重要的。MyBatis自身提供了相应的数据源实现,当然MyBat MyBatis源码分析(二):项目结构 小徐很努力 源码解读mybatisjava开发语言 目录1、前言2、代码统计3、整体架构3.1、基础支持层3.1.1、反射模块3.1.2、类型模块3.1.3、日志模块3.1.4、IO模块3.1.5、解析器模块3.1.6、数据源模块3.1.7、缓存模块3.1.8、Binding模块3.1.9、注解模块3.1.10、异常模块3.2、核心处理层3.2.1、配置解析模块3.2.2、SQL解析模块3.2.3、插件模块3.3、接口层3.3.1、session模 MyBatis 源码分析(四):反射模块 小徐很努力 源码解读java开发语言mybatis 前言上一篇我们了解了Mybatis解析器模块,MyBatis源码分析(三):解析器模块本篇我们来了解反射模块。相比parsing包来说,reflection包的代码量大概是2-3倍。当然,不要慌,都是比较简单的代码。当然,这是一篇非常非常非常长的博客,如果想要比较好的理解这个模块,一定要基于MyBatis提供的这个模块的单元测试,多多调试。对应reflection包。如下图所示:在MyBatis源 MyBatis源码分析(三):解析器模块 小徐很努力 源码解读mybatisjava开发语言 目录1、前言2、源码分析2.1、XPathParser2.1.1、属性讲解2.1.2、构造方法讲解2.2、eval方法2.2.1、eval元素2.2.2、eval节点2.3、XMLMapperEntityResolver2.4、PropertyParser2.5、GenericTokenParser2.6、TokenHandler2.6.1VariableTokenHandler2.6.2hand MyBatis源码分析(一):搭建调试环境 小徐很努力 源码解读mybatis 目录拉取源码安装环境调试测试1、mybatis-config.xml2、AutoConstructorMapper.xml3、AutoConstructorMapper4、CreateDB.sql5、POJO5.1AnnotatedSubject5.2、PrimitiveSubject5.3、BadSubject5.4、ExtensiveSubject6、AutoConstructorTest6. Mybatis源码分析:@Mapkey的使用 辰鬼丫 SpringSpringMVCMyBatisjavajavaspring @Mapkey的使用在多值查询的时候,通常要把方法返回类型设置为List类型,Mybatis为我们提供了另一种解决方式,通过K-V的形式将查询结果保存在Map中,这种实现方式只需要在方法上标注为@Mapkey即可。如下代码:查询一个学生表,包含id,name,age三个字段,使用@mapkey将id号作为K,查询结果作为V.使用@Mapkey时最好将K设置为唯一的,否则后续的结果将会覆盖已查询到的 Mybatis 使用记录 antRain Spring+springbootmybatis Mybatis使用记录Mybatis下载源码编译democonfig.propertiesmybatis-config.xmlUserMapper.xml测试代码mybatis-spring下载编译demomybatis-spring-boot-starter测试demomybatisplus编译Mybatis下载源码编译参考mybatis源码分析00:获取源码及demo准备org.mybatis MyBatis源码分析之核心流程介绍(下) 波波烤鸭 深入浅出Mybatis专栏mybatis源码分析 接上一篇我们继续来介绍1.SqlSession程序每一次操作数据库,都需要创建一个会话,我们用openSession()方法来创建。接下来我们看看SqlSession创建过程中做了哪些操作SqlSessionsqlSession=factory.openSession();通过前面创建的DefaultSqlSessionFactory的openSession方法来创建@Overridepubl MyBatis源码分析之核心流程介绍(上) 波波烤鸭 深入浅出Mybatis专栏mybatis源码分析 本文我们来看看MyBatis的核心流程核心流程分析 首先来看看MyBatis的主要工作流程图 分析源码我们还是从编程式的Demo入手/***MyBatisgetMapper方法的使用*/@Testpublicvoidtest2()throwsException{//1.获取配置文件InputStreami Mybatis源码分析——结果集ResultSet自动映射成实体类对象 小波同学 前言上一篇文章我们已经将SQL发送到了数据库,并返回了ResultSet,接下来就是将结果集ResultSet自动映射成实体类对象。这样使用者就无需再手动操作结果集,并将数据填充到实体类对象中。这可大大降低开发的工作量,提高工作效率。映射结果入口我们来看看上次看源码的位置publicclassPreparedStatementHandlerextendsBaseStatementHandler{@ Mybatis 源码分析(四)之 Mybatis 的执行流程梳理 挂机的啊洋zzZ mybatis.pngMybatis源码分析(四)之Mybatis整体的执行流程前面了解到Mybatis的执行流程,首先读取我们的mybatis-config.xml配置文件,然后构建Configuration类,这个类会像上下文信息一样会传来传去,以便我们获取其中的信息。构建Configuration过程中,会读取我们的配置信息,其中包含读取我们的mapper的配置,并将mapper的信息以ke [MyBatis源码分析 - 数据源模块] 小胡_鸭 一、简介 数据源是实际开发中常用的组件之一。现在开源的数据源都提供了比较丰富的功能,例如,连接池功能、检测连接状态等,选择性能优秀的数据源组件对于提升ORM框架乃至整个应用的性能都是非常重要的。 MyBatis自身提供了相应的数据源实现,当然MyBatis也提供了与第三方数据源集成的接口,这些功能都位于数据源模块之中,该模块位于org.apache.ibatis.datasource包中,相关 MyBatis源码分析 拧螺丝专业户 mybatis MyBatis源码分析MyBatis是常用的持久层框架,帮助我们减少了很多的访问数据库的代码。这次我们就来看看MyBatis是怎么做到这些的?看看它里面用到了哪些值得我们借鉴的技术。一、示例程序为了方便后续在本地进行debug调试,首先准备一个示例程序。1、项目工程结构2、父工程pom.xml文件内容:4.0.0org.examplekeepLearnOnMavenpom1.0-SNAPSHOTs 8 Mybatis源码分析 刺豚灬 #Springspring 1.3.2源码执行流程1.通过@MapperScan导入了MapperScannerRegistrar类2.MapperScannerRegistrar类实现了ImportBeanDefinitionRegistrar接口,所以Spring在启动时会调用MapperScannerRegistrar类中的registerBeanDefinitions方法3.在registerBeanDefiniti Mybatis源码分析(01)-JDBC操作回顾,Mybatis整体架构概览 couthz 一JDBC的规范操作及问题回顾假设数据库test中有一张表account直接使用JDBC,dao层的实现类可能会是如下的写法:publicclassAccountDaoImplimplementsAccountDao{//问题1:数据库配置信息,存在硬编码问题,修改信息就要重新编译privateStringdriver="com.mysql.jdbc.driver";privateStringur Mybatis源码分析(二)Mybatis-config.xml的初始化 长安不及十里 #Mybatis源码分析mybatisjava源码分析学习文件读写 目录一环境搭建二配置文件初始化2.1ClassLoader2.1.1Java类加载器2.2获取配置文件三扩展3.1VFS3.1.1DefaultVFS3.1.2JBoss6VFS3.2ResolverUtil工具类(重庆)涂鸦一条街系列文章:文章状态时间描述(一)Mybatis基本使用已复习2022-12-14对Mybtais的基本使用,能够开发(二)Mybatis-config.xml的初始化已 mybatis源码分析一(加载配置文件) 为梦想前进 最近一直在看mybatis源码,稍有心得,接下来就然我们一起看下springboot整合mybatis的源码的步骤是怎样的废话不多说,咱们就一起看看源码吧首先,咱们看下配置文件,下面是我配置的配置文件,没什么多说的,都是基本配置,映射文件的位置,实体类的位置,数据库的基本信息等之前一直有个疑问,就是咱们在配置文件中写这些配置的时候都会自动提示,一直不知道是怎么回事,看了源码,才知道原因,原来,在s Mybatis源码分析-一级缓存【BaseExecutor】 你看起来很好吃_fb4a 本文主题:Executor执行体系回顾为什么要有一级缓存?一级缓存、二级缓存有什么区别?一级缓存属于通用逻辑,那么结构上它是如何设计的?一级缓存是用什么实现的?一级缓存命中条件有哪些?一级缓存有哪些清空场景?Spring和Mybatis整合一级缓存失效?一级缓存的注意事项Executor执行体系Executor执行体系.png这是从SqlSession到Executor实现的执行体系图,从图中可以 一周学完MyBatis源码,万字总结 互联网全栈架构 数据库mybatismysqlspringjava 关注公众号“java后端技术全栈”回复“000”获取优质面试资料大家好,我是老田。之前,我给大家分享给很多MyBatis源码分析的一系列文章。今天,就自己的感受来做一个整体的总结。众所周知,MyBatis是对JDBC进行封装而成的产品,所以,聊MyBatis源码之前我们得先了解JDBC。推荐:JDBC这个问题,问的小伙伴一脸懵逼JDCBJDBC案例:public class JdbcDemo { Redis学习、缓存、持久化、哨兵模式 程序dunk 面试专题redisnosql数据库 总结不易,如果对你有帮助,请点赞关注支持一下微信搜索程序dunk,关注公众号,获取博客源码我写代码是为了更好的表达自我,这是艺术创作,而不单单是为了把事情搞定。—Antirez序号内容1Java基础面试题2JVM面试题3Java并发编程面试4计算机网络知识点汇总5MySQL面试题6Mybatis源码分析+面试7Spring面试题8SpringMVC面试题9SpringBoot面试题10Spring MyBatis源码分析 森火123 java架构javaintellij-ideamaven 分析完了MyBatis的架构和执行流程,终于到了源码分析的章节,估计很多小伙伴的大刀都已经饥渴难耐了,好了接下来咱么就要开始“DoubleKill”了。这篇文章咱么主要根据MyBatis的执行流程,通过Debug的方式,来一步步非常详细的带着大家看下MyBatis的从加载配置文件、解析配置文件、创建四大核心对象(Executor、ParameterHandler、ResultSetHandler、 mybatis源码分析(一):自己动手写一个简单的mybaits框架 猫清扬 框架解决了什么问题目前主流的JavaWeb项目都采用SSM(springspringmvcmybatis)框架,其中mybatis框架源码是最简单的,想入手源码学习的同学很推荐从mybatis开始。本系列文章是我对mybatis源码学习的一些梳理总结,可以帮助你更高效得理解mybatis。在学习一个框架源码之前你首先要问自己几个问题。1.这个框架解决了什么问题?2.为了解决这个问题,你是如何设计的 Mybatis源码分析(十四)Mybatis的设计模式梳理 长安不及十里 #Mybatis源码分析mybatis设计模式java源码分析缓存 目录一工厂模式1.1案例说明1.2源码设计模式分析二单例模式2.1案例说明2.2源码设计模式分析三代理模式3.1动态代理案例3.2源码设计模式分析四建造者模式4.1案例4.2源码设计模式分析五装饰器模式5.1代码案例5.2源码设计模式分析六组合模式6.1案例6.2源码设计模式分析系列文章:文章状态时间描述(一)Mybatis基本使用已复习2022-12-14对Mybtais的基本使用,能够开发(二 Mybatis源码分析 不知名的艾坤 SSMSpringBoot源码分析mybatisjava开发语言 1.Mybatis整体三层设计SSM中,Spring、SpringMVC已经在前面文章源码分析总结过了,Mybatis源码相对Spring和SpringMVC而言是的简单的,只有一个项目,项目下分了很多包。从宏观上了解Mybatis的整体框架分为三层,分别是基础支持层、核心处理层、和接口层。如下图MyBatis的主要工作流程图2.1接口层首先接口层是我们打交道最多的。核心对象是SqlSession Mybatis源码分析一-Mybatis基础架构以及设计模式,java工厂模式教程视频 m0_64383449 程序员面试java后端 1、mybatis源码下载地址MyBatis源码下载地址:https://github.com/MyBatis/MyBatis-32、源码包导入过程:下载Mybatis源码检查Maven版本,必须是3.25以上Mybatis工程是Maven项目,JDK必须是1.8版本pom文件汇总添加true全部改为false在工程目录下执行mvncleaninstall-Dmaven.test.skip=tru Mybatis源码分析一-Mybatis基础架构以及设计模式,javase项目实战 m0_65483457 程序员面试java后端 MyBatis源码下载地址:https://github.com/MyBatis/MyBatis-32、源码包导入过程:下载Mybatis源码检查Maven版本,必须是3.25以上Mybatis工程是Maven项目,JDK必须是1.8版本pom文件汇总添加true全部改为false在工程目录下执行mvncleaninstall-Dmaven.test.skip=true将当前工程安装到本地仓库二、 开发者关心的那些事 圣子足道 ios游戏编程apple支付 我要在app里添加IAP,必须要注册自己的产品标识符(product identifiers)。产品标识符是什么? 产品标识符(Product Identifiers)是一串字符串,它用来识别你在应用内贩卖的每件商品。App Store用产品标识符来检索产品信息,标识符只能包含大小写字母(A-Z)、数字(0-9)、下划线(-)、以及圆点(.)。你可以任意排列这些元素,但我们建议你创建标识符时使用 负载均衡器技术Nginx和F5的优缺点对比 bijian1013 nginxF5 对于数据流量过大的网络中,往往单一设备无法承担,需要多台设备进行数据分流,而负载均衡器就是用来将数据分流到多台设备的一个转发器。 目前有许多不同的负载均衡技术用以满足不同的应用需求,如软/硬件负载均衡、本地/全局负载均衡、更高 LeetCode[Math] - #9 Palindrome Number Cwind javaAlgorithm题解LeetCodeMath 原题链接:#9 Palindrome Number 要求: 判断一个整数是否是回文数,不要使用额外的存储空间 难度:简单 分析: 题目限制不允许使用额外的存储空间应指不允许使用O(n)的内存空间,O(1)的内存用于存储中间结果是可以接受的。于是考虑将该整型数反转,然后与原数字进行比较。 注:没有看到有关负数是否可以是回文数的明确结论,例如 画图板的基本实现 15700786134 画图板 要实现画图板的基本功能,除了在qq登陆界面中用到的组件和方法外,还需要添加鼠标监听器,和接口实现。 首先,需要显示一个JFrame界面: public class DrameFrame extends JFrame { //显示 linux的ps命令 被触发 linux Linux中的ps命令是Process Status的缩写。ps命令用来列出系统中当前运行的那些进程。ps命令列出的是当前那些进程的快照,就是执行ps命令的那个时刻的那些进程,如果想要动态的显示进程信息,就可以使用top命令。 要对进程进行监测和控制,首先必须要了解当前进程的情况,也就是需要查看当前进程,而 ps 命令就是最基本同时也是非常强大的进程查看命令。使用该命令可以确定有哪些进程正在运行 Android 音乐播放器 下一曲 连续跳几首歌 肆无忌惮_ android 最近在写安卓音乐播放器的时候遇到个问题。在MediaPlayer播放结束时会回调 player.setOnCompletionListener(new OnCompletionListener() { @Override public void onCompletion(MediaPlayer mp) { mp.reset(); Log.i("H java导出txt文件的例子 知了ing javaservlet 代码很简单就一个servlet,如下: package com.eastcom.servlet; import java.io.BufferedOutputStream; import java.io.IOException; import java.net.URLEncoder; import java.sql.Connection; import java.sql.Resu Scala stack试玩, 提高第三方依赖下载速度 矮蛋蛋 scalasbt 原文地址: http://segmentfault.com/a/1190000002894524 sbt下载速度实在是惨不忍睹, 需要做些配置优化 下载typesafe离线包, 保存为ivy本地库 wget http://downloads.typesafe.com/typesafe-activator/1.3.4/typesafe-activator-1.3.4.zip 解压r phantomjs安装(linux,附带环境变量设置) ,以及casperjs安装。 alleni123 linuxspider 1. 首先从官网 http://phantomjs.org/下载phantomjs压缩包,解压缩到/root/phantomjs文件夹。 2. 安装依赖 sudo yum install fontconfig freetype libfreetype.so.6 libfontconfig.so.1 libstdc++.so.6 3. 配置环境变量 vi /etc/profil JAVA IO FileInputStream和FileOutputStream,字节流的打包输出 百合不是茶 java核心思想JAVA IO操作字节流 在程序设计语言中,数据的保存是基本,如果某程序语言不能保存数据那么该语言是不可能存在的,JAVA是当今最流行的面向对象设计语言之一,在保存数据中也有自己独特的一面,字节流和字符流 1,字节流是由字节构成的,字符流是由字符构成的 字节流和字符流都是继承的InputStream和OutPutStream ,java中两种最基本的就是字节流和字符流 类 FileInputStream Spring基础实例(依赖注入和控制反转) bijian1013 spring 前提条件:在http://www.springsource.org/download网站上下载Spring框架,并将spring.jar、log4j-1.2.15.jar、commons-logging.jar加载至工程1.武器接口 package com.bijian.spring.base3; public interface Weapon { void kil HR看重的十大技能 bijian1013 提升能力HR成长 一个人掌握何种技能取决于他的兴趣、能力和聪明程度,也取决于他所能支配的资源以及制定的事业目标,拥有过硬技能的人有更多的工作机会。但是,由于经济发展前景不确定,掌握对你的事业有所帮助的技能显得尤为重要。以下是最受雇主欢迎的十种技能。 一、解决问题的能力 每天,我们都要在生活和工作中解决一些综合性的问题。那些能够发现问题、解决问题并迅速作出有效决 【Thrift一】Thrift编译安装 bit1129 thrift 什么是Thrift The Apache Thrift software framework, for scalable cross-language services development, combines a software stack with a code generation engine to build services that work efficiently and s 【Avro三】Hadoop MapReduce读写Avro文件 bit1129 mapreduce Avro是Doug Cutting(此人绝对是神一般的存在)牵头开发的。 开发之初就是围绕着完善Hadoop生态系统的数据处理而开展的(使用Avro作为Hadoop MapReduce需要处理数据序列化和反序列化的场景),因此Hadoop MapReduce集成Avro也就是自然而然的事情。 这个例子是一个简单的Hadoop MapReduce读取Avro格式的源文件进行计数统计,然后将计算结果 nginx定制500,502,503,504页面 ronin47 nginx 错误显示 server { listen 80; error_page 500/500.html; error_page 502/502.html; error_page 503/503.html; error_page 504/504.html; location /test {return502;}} 配置很简单,和配 java-1.二叉查找树转为双向链表 bylijinnan 二叉查找树 import java.util.ArrayList; import java.util.List; public class BSTreeToLinkedList { /* 把二元查找树转变成排序的双向链表 题目: 输入一棵二元查找树,将该二元查找树转换成一个排序的双向链表。 要求不能创建任何新的结点,只调整指针的指向。 10 / \ 6 14 / \ Netty源码学习-HTTP-tunnel bylijinnan javanetty Netty关于HTTP tunnel的说明: http://docs.jboss.org/netty/3.2/api/org/jboss/netty/channel/socket/http/package-summary.html#package_description 这个说明有点太简略了 一个完整的例子在这里: https://github.com/bylijinnan JSONUtil.serialize(map)和JSON.toJSONString(map)的区别 coder_xpf jqueryjsonmapval() JSONUtil.serialize(map)和JSON.toJSONString(map)的区别 数据库查询出来的map有一个字段为空 通过System.out.println()输出 JSONUtil.serialize(map): {"one":"1","two":"nul Hibernate缓存总结 cuishikuan 开源sshjavawebhibernate缓存三大框架 一、为什么要用Hibernate缓存? Hibernate是一个持久层框架,经常访问物理数据库。 为了降低应用程序对物理数据源访问的频次,从而提高应用程序的运行性能。 缓存内的数据是对物理数据源中的数据的复制,应用程序在运行时从缓存读写数据,在特定的时刻或事件会同步缓存和物理数据源的数据。 二、Hibernate缓存原理是怎样的? Hibernate缓存包括两大类:Hib CentOs6 dalan_123 centos 首先su - 切换到root下面1、首先要先安装GCC GCC-C++ Openssl等以来模块:yum -y install make gcc gcc-c++ kernel-devel m4 ncurses-devel openssl-devel2、再安装ncurses模块yum -y install ncurses-develyum install ncurses-devel3、下载Erang 10款用 jquery 实现滚动条至页面底端自动加载数据效果 dcj3sjt126com JavaScript 无限滚动自动翻页可以说是web2.0时代的一项堪称伟大的技术,它让我们在浏览页面的时候只需要把滚动条拉到网页底部就能自动显示下一页的结果,改变了一直以来只能通过点击下一页来翻页这种常规做法。 无限滚动自动翻页技术的鼻祖是微博的先驱:推特(twitter),后来必应图片搜索、谷歌图片搜索、google reader、箱包批发网等纷纷抄袭了这一项技术,于是靠滚动浏览器滚动条 ImageButton去边框&Button或者ImageButton的背景透明 dcj3sjt126com imagebutton 在ImageButton中载入图片后,很多人会觉得有图片周围的白边会影响到美观,其实解决这个问题有两种方法 一种方法是将ImageButton的背景改为所需要的图片。如:android:background="@drawable/XXX" 第二种方法就是将ImageButton背景改为透明,这个方法更常用 在XML里; <ImageBut JSP之c:foreach eksliang jspforearch 原文出自:http://www.cnblogs.com/draem0507/archive/2012/09/24/2699745.html <c:forEach>标签用于通用数据循环,它有以下属性 属 性 描 述 是否必须 缺省值 items 进行循环的项目 否 无 begin 开始条件 否 0 end 结束条件 否 集合中的最后一个项目 step 步长 否 1 Android实现主动连接蓝牙耳机 gqdy365 android 在Android程序中可以实现自动扫描蓝牙、配对蓝牙、建立数据通道。蓝牙分不同类型,这篇文字只讨论如何与蓝牙耳机连接。 大致可以分三步: 一、扫描蓝牙设备: 1、注册并监听广播: BluetoothAdapter.ACTION_DISCOVERY_STARTED BluetoothDevice.ACTION_FOUND BluetoothAdapter.ACTION_DIS android学习轨迹之四:org.json.JSONException: No value for hyz301 json org.json.JSONException: No value for items 在JSON解析中会遇到一种错误,很常见的错误 06-21 12:19:08.714 2098-2127/com.jikexueyuan.secret I/System.out﹕ Result:{"status":1,"page":1,& 干货分享:从零开始学编程 系列汇总 justjavac 编程 程序员总爱重新发明轮子,于是做了要给轮子汇总。 从零开始写个编译器吧系列 (知乎专栏) 从零开始写一个简单的操作系统 (伯乐在线) 从零开始写JavaScript框架 (图灵社区) 从零开始写jQuery框架 (蓝色理想 ) 从零开始nodejs系列文章 (粉丝日志) 从零开始编写网络游戏 jquery-autocomplete 使用手册 macroli jqueryAjax脚本 jquery-autocomplete学习 一、用前必备 官方网站:http://bassistance.de/jquery-plugins/jquery-plugin-autocomplete/ 当前版本:1.1 需要JQuery版本:1.2.6 二、使用 <script src="./jquery-1.3.2.js" type="text/ja PLSQL-Developer或者Navicat等工具连接远程oracle数据库的详细配置以及数据库编码的修改 超声波 oracleplsql 在服务器上将Oracle安装好之后接下来要做的就是通过本地机器来远程连接服务器端的oracle数据库,常用的客户端连接工具就是PLSQL-Developer或者Navicat这些工具了。刚开始也是各种报错,什么TNS:no listener;TNS:lost connection;TNS:target hosts...花了一天的时间终于让PLSQL-Developer和Navicat等这些客户 数据仓库数据模型之:极限存储--历史拉链表 superlxw1234 极限存储数据仓库数据模型拉链历史表 在数据仓库的数据模型设计过程中,经常会遇到这样的需求: 1. 数据量比较大; 2. 表中的部分字段会被update,如用户的地址,产品的描述信息,订单的状态等等; 3. 需要查看某一个时间点或者时间段的历史快照信息,比如,查看某一个订单在历史某一个时间点的状态, 比如,查看某一个用户在过去某一段时间内,更新过几次等等; 4. 变化的比例和频率不是很大,比如,总共有10 10点睛Spring MVC4.1-全局异常处理 wiselyman spring mvc 10.1 全局异常处理 使用@ControllerAdvice注解来实现全局异常处理; 使用@ControllerAdvice的属性缩小处理范围 10.2 演示 演示控制器 package com.wisely.web; import org.springframework.stereotype.Controller; import org.spring 按字母分类: ABCDEFGHIJKLMNOPQRSTUVWXYZ其他
回过头来再看CachingExecutor的query方法:
public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException { BoundSql boundSql = ms.getBoundSql(parameterObject); CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql); return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); }
由于这个query方法比较复杂,这篇我先分析获取BoundSql对象也就是MappedStatement的getBoundSql方法。BoundSql内包含了执行的sql语句和命名参数。
public class BoundSql { private final String sql; private final List parameterMappings; private final Object parameterObject; private final Map, Object> additionalParameters; private final MetaObject metaParameters;
我画了一副时序图来说明获取BoundSql的过程。
下面按时序图的步骤查看源代码。
1.mapperStatement的getBoundSql方法会调用其成员变量sqlSource的getBoundSql方法
public BoundSql getBoundSql(Object parameterObject) { DynamicContext context = new DynamicContext(configuration, parameterObject); rootSqlNode.apply(context); SqlSourceBuilder sqlSourceParser = new SqlSourceBuilder(configuration); Class parameterType = parameterObject == null ? Object.class : parameterObject.getClass(); SqlSource sqlSource = sqlSourceParser.parse(context.getSql(), parameterType, context.getBindings()); BoundSql boundSql = sqlSource.getBoundSql(parameterObject); for (Map.Entry, Object> entry : context.getBindings().entrySet()) { boundSql.setAdditionalParameter(entry.getKey(), entry.getValue()); } return boundSql; }
DynamicSqlSource会创建一个DynamicContext对象,这个对象作为SqlNode的apply方法的入参,SqlNode每调用一次apply方法,就是将此节点含有的sql片段拼接到context中去,此外还包含了参数对象,通过MetaObject(本文最后介绍)可以使用A.B[0].C的形式
2.DynamicSqlSource所持有的SqlNode为MixedSqlNode,它内部的contents对象就是顶级节点下的兄弟sql片段,依次调用兄弟片段的apply方法拼接sql,当然其中每个兄弟sql片段下还有可能为多个sql片段。假如一个sql片段是一个不含任何可变参数的那么它就是StaticTextSqlNode,静态节点就是单纯的拼接一下sql,当然还有其他种类的SqlNode,譬如WhereSqlNode,看过它的代码我们就可以看出对于where后第一个and或or是不会拼入sql中的,我们这里就不一一列举了。
public boolean apply(DynamicContext context) { for (SqlNode sqlNode : contents) {//MixedSqlNode sqlNode.apply(context); } return true; }
public boolean apply(DynamicContext context) {//StaticTextSqlNode context.appendSql(text); return true; }
3.此时context中已经含有完整的sql语句了只不过还没有替换掉mybatis的变量占位符信息(类似#{prop,jdbcType=...})。下一步通过SqlSourceBuilder的parse方法将sql语句转换为标准的带有Jdbc站位符?的语句,并将变量占位符信息抽像为ParameterMapping。返回一个包含sql和List的StaticSqlSource。
public SqlSource parse(String originalSql, Class parameterType, Map, Object> additionalParameters) { ParameterMappingTokenHandler handler = new ParameterMappingTokenHandler(configuration, parameterType, additionalParameters); GenericTokenParser parser = new GenericTokenParser("#{", "}", handler); String sql = parser.parse(originalSql); return new StaticSqlSource(configuration, sql, handler.getParameterMappings()); }
GenericTokenParser将#{}中的内容过滤出来,使用ParameterMappingTokenHandler将其转换为ParameterMapping对象,然后用?将#{}替换。
4.通过以上步骤将一个DynamicSqlSource转换成了一个含有List的StaticSqlSource,调用getBoundSql方法返回BoundSql对象。
public BoundSql getBoundSql(Object parameterObject) { return new BoundSql(configuration, sql, parameterMappings, parameterObject); }
上面第一步中说到MetaObject可以将参数对象对外提供A.B[0].C的访问形式,那么他是怎么做到的呢。
Configuration的newMetaObject方法调用MetaObject.forObject()方法将自身持有的DefaultObjectFactory,DefaultObjectWrapperFactory和DefaultReflectorFactory一并传入。
public MetaObject newMetaObject(Object object) { return MetaObject.forObject(object, objectFactory, objectWrapperFactory, reflectorFactory); }
public static MetaObject forObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory, ReflectorFactory reflectorFactory) { if (object == null) { return SystemMetaObject.NULL_META_OBJECT; } else { return new MetaObject(object, objectFactory, objectWrapperFactory, reflectorFactory); } }
private MetaObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory, ReflectorFactory reflectorFactory) { this.originalObject = object; this.objectFactory = objectFactory; this.objectWrapperFactory = objectWrapperFactory; this.reflectorFactory = reflectorFactory; if (object instanceof ObjectWrapper) { this.objectWrapper = (ObjectWrapper) object; } else if (objectWrapperFactory.hasWrapperFor(object)) { this.objectWrapper = objectWrapperFactory.getWrapperFor(this, object); } else if (object instanceof Map) { this.objectWrapper = new MapWrapper(this, (Map) object); } else if (object instanceof Collection) { this.objectWrapper = new CollectionWrapper(this, (Collection) object); } else { this.objectWrapper = new BeanWrapper(this, object); } }
然后MetaObject通过构造方法将这些参数保留下来,构造函数对不同的object类型创建不同的ObjectWrapper对象,MetaObject通过ObjectWrapper对object提供了方便的多层级的getter和setter方法。
MetaObject对外提供属性的访问是getValue方法:
public Object getValue(String name) { PropertyTokenizer prop = new PropertyTokenizer(name); if (prop.hasNext()) { MetaObject metaValue = metaObjectForProperty(prop.getIndexedName()); if (metaValue == SystemMetaObject.NULL_META_OBJECT) { return null; } else { return metaValue.getValue(prop.getChildren()); } } else { return objectWrapper.get(prop); } }
public MetaObject metaObjectForProperty(String name) { Object value = getValue(name); return MetaObject.forObject(value, objectFactory, objectWrapperFactory, reflectorFactory); }
1.根据PropertyTokenizer判断是否是一个多层级的属性访问,如果是则先取顶层对象将其重新包裹为一个MetaObject对象,然后对这个对象再次调用getValue方法,这个时候getValue的参数为去除第一层级的字符串了,直到入参name为单层级的字符串也是就不带"."的字符串然后调用objectWrapper的get方法取出name所对应的属性值。这里我们以BeanWrapper为例看看这个方法的实现。
public Object get(PropertyTokenizer prop) { if (prop.getIndex() != null) { Object collection = resolveCollection(prop, object); return getCollectionValue(prop, collection); } else { return getBeanProperty(prop, object); } }