精华内容
下载资源
问答
  • orm 对象关系映射框架简单实现

    千次阅读 2018-06-26 20:05:42
    1 orm:就是将对象关系数据库建立一种映射关系,精确的,一对一的,保证你对对象的某一个操作,比如修改对象属性,创建新的对象等,唯一的对应数据的一种修改数据的操作或者增加数据的操作.orm框架就是提供这种操作的,...

    1 orm:就是将对象与关系数据库建立一种映射关系,精确的,一对一的,保证你对对象的某一个操作,比如修改对象属性,创建新的对象等,唯一的对应数据的一种修改数据的操作或者增加数据的操作.orm框架就是提供这种操作的,用户只需定义一个类,就可以用对象的方法属性来操作数据库.

    class User(Model):
        # 定义类的属性到列的映射:
        id = IntegerField('id')
        name = StringField('username')
        email = StringField('email')
        password = StringField('password')
    ​
    # 创建一个实例:
    u = User(id=12345, name='Michael', email='test@orm.org', password='my-pwd')
    # 保存到数据库:
    u.save()

    2 一般的映射关系就是:

    对象关系数据库
    类名表的名字
    类的实例表中的一行数据
    类的属性表的字段

    3 主要技术就是采用元类动态创建类对象

    3.1 定义元类:元类,实现动态创建需要的类.

    class ModelMetaclass(type):
    ​
        def __new__(cls, name, bases, attrs):
            if name=='Model':
                return type.__new__(cls, name, bases, attrs)
            print('Found model: %s' % name)
            mappings = dict()
            for k, v in attrs.items():
                if isinstance(v, Field):
                    print('Found mapping: %s ==> %s' % (k, v))
                    mappings[k] = v
            for k in mappings.keys():
                attrs.pop(k)
            attrs['__mappings__'] = mappings # 保存属性和列的映射关系
            attrs['__table__'] = name # 假设表名和类名一致
            return type.__new__(cls, name, bases, attrs)

    3.2 定义基类:提供基本的类操作方法.

    class Model(dict, metaclass=ModelMetaclass):
    ​
        def __init__(self, **kw):
            super(Model, self).__init__(**kw)
    ​
        def __getattr__(self, key):
            try:
                return self[key]
            except KeyError:
                raise AttributeError(r"'Model' object has no attribute '%s'" % key)
    ​
        def __setattr__(self, key, value):
            self[key] = value
    ​
        def save(self):
            fields = []
            params = []
            args = []
            for k, v in self.__mappings__.items():
                fields.append(v.name)
                params.append('?')
                args.append(getattr(self, k, None))
            sql = 'insert into %s (%s) values (%s)' % (self.__table__, ','.join(fields), ','.join(params))
            print('SQL: %s' % sql)
            print('ARGS: %s' % str(args))

    3.3 定义Field,保存数据库表的字段名和字段类型:

    class Field(object):
    ​
        def __init__(self, name, column_type):
            self.name = name
            self.column_type = column_type
    ​
        def __str__(self):
            return '<%s:%s>' % (self.__class__.__name__, self.name)

    Field的基础上,进一步定义各种类型的Field,比如StringFieldIntegerField等等:

    class StringField(Field):
    ​
        def __init__(self, name):
            super(StringField, self).__init__(name, 'varchar(100)')
    ​
    class IntegerField(Field):
    ​
        def __init__(self, name):
            super(IntegerField, self).__init__(name, 'bigint')

    3.4 当用户定义一个class User(Model)时,Python解释器首先在当前类User的定义中查找metaclass,如果没有找到,就继续在父类Model中查找metaclass,找到了,就使用Model中定义的metaclassModelMetaclass来创建User类,也就是说,metaclass可以隐式地继承到子类,但子类自己却感觉不到。

    ModelMetaclass中,一共做了几件事情:

    1. 排除掉对Model类的修改;

    2. 在当前类(比如User)中查找定义的类的所有属性,如果找到一个Field属性,就把它保存到一个__mappings__的dict中,同时从类属性中删除该Field属性,否则,容易造成运行时错误(实例的属性会遮盖类的同名属性);

    3. 把表名保存到__table__中,这里简化为表名默认为类名。

    Model类中,就可以定义各种操作数据库的方法,比如save()delete()find()update等等。

    我们实现了save()方法,把一个实例保存到数据库中。因为有表名,属性到字段的映射和属性值的集合,就可以构造出INSERT语句。参考廖雪峰python元类.https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0014319106919344c4ef8b1e04c48778bb45796e0335839000

    展开全文
  • js开源的ORM对象关系映射框架Onela是基于node . js开源的基于对象的映射框架、支持各种关系数据库数据基础设施。重大更新:v2.0.0版本发布此版本重大更新,原有如果使用了V2.0.0之前的版本请注意,升级到最新版,最原有...
  • jetbrick 的对象关系映射框架。 文档 依赖 < groupId>com.github.subchen</ groupId> < artifactId>jetbrick-orm < version>2.0 下载 二进制文件 依赖关系 执照 Copyright 2013-2014 Guoqiang Chen, ...
  • 企业资源管理系统 用于 Java 的快速对象关系映射框架 作者: 版本:Beta 0.1
  • Reladomo是用于Java的对象关系映射(ORM)框架,具有以下企业功能: 强类型的编译时检查查询语言 双时态链 透明的多模式支持 完全支持可单元测试的代码 有关更多详细信息,请参见。 我该怎么办? 将数据建模为...
  • Inloquent:基于C ++ Qt的类似Laravel-Eloquent的对象关系映射框架
  • hibernate开放源码框架太强大了,使用jdbc的程序员可以在任何场合使用这个框架,只要jdbc被使用。
  • Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它讲POJO与数据库表建立映射关系,是一个全自动的orm框架,Hibernate可以自动生成SQL语句,自动执行,使Java程序员可以随心所欲...

    Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它讲POJO与数据库表建立映射关系,是一个全自动的orm框架,Hibernate可以自动生成SQL语句,自动执行,使Java程序员可以随心所欲的使用对象编程思维来操作数据库。Hibernate可以应用再任何使用JDBC的场合,既可以在Java的客户端程序使用,也可以在Servlet/JSP的WEB应用中使用,最具有革命意义的是:Hibernate可以在应用EJB的J2EE架构中取代CMP。完成数据持久化的重任。

    核心接口和类

    Hibernate的核心类和接口一共有6个,分别是:Session、 SessionFactory、Transaction、Query、Criteria和Configuration。这6个核心类和接口在任何开发中都有用到,通过这些接口,不仅可以持久化对象进行存取,还能够进行事务控制。

    Session : 接口负责执行被持久化对象的CRUD操作(CRUD的任务是完成与数据库的交流,包含了很多常见的SQL语句。)但需要注意的是Session对象是非线程安全的。同时,Hibernate的session不同于JSP应用中的HttpSession。这里当使用session这个术语时,其实指的是Hibernate中的Session,而以后将HttpSession对象成为用户Session。
    SessionFactory: SessionFactory接口负责初始化Hibernate。它充当数据存储源的代理,并负责创建Session对象。这里用到了工厂模式。需要注意的是SessionFactory并不是轻量级的。因为一般情况下,一个项目只需要一个SessionFactory就够了。当需要操作多个数据库时,可以为每个数据库指定一个SessionFactory。
    Transaction: Transaction 接口是一个可选的API,可以选择不使用这个接口,取而代之的是Hibernate的设计者自己写的底层事务处理代码。Transaction接口是对实际事务实现的一个抽象,这些实现包括JDBC的事务、JTA中的UserTransaction、甚至可以是CORBA事务。之所以这样设计是能让开发者能够使用一个统一事务的操作界面,使得自己的项目可以在不同的环境和容器之间方便移植。
    Query: Query接口让你方便地对数据库持久对象进行查询,它可以有两种表达方式:HQL语言或本地数据库的SQL语句。Query经常被用来绑定查询参数。限制查询记录数量,并最终执行查询操作

    Criteria: Criteria接口与Query接口非常类似,允许创建并执行面向对象的标准化查询。值得注意的是Criteria接口也是轻量级的,它不能在Session之外使用。
    Configuration: Configuration类的作用是对Hibernate进行配置,以及对它进行启动。在hibernate的启动过程中,Configuration类的实力首先定位映射文档的位置,读取这些配置,然后创建一个SessionFactory对象。虽然Configuration类在整个Hibernate项目中只扮演着一个很小的角色,但它是启动Hibernate时所遇到的第一个对象。

    主键介绍
    Assigned
    Assigned方式由用户生成主键值,并且要在save()之前指定否则会抛出异常
    特点:主键的生成值完全由用户决定,与底层数据库无关。用户需要维护主键值,在调用session.save()之前要指定主键值。
    Hilo
    Hilo使用高低位算法生成主键,高低位算法使用一个高位值和一个低位值,然后把算法得到的两个值拼接起来作为数据库中的唯一主键。Hilo方式需要额外的数据库表和字段提供高位值来源。默认情况下使用的表是
    hibernate_unique_key,默认字段叫作next_hi。next_hi必须有一条记录否则会出现错误。
    特点:需要额外的数据库表的支持,能保证同一个数据库中主键的唯一性,但不能保证多个数据库之间主键的唯一性。Hilo主键生成方式由Hibernate 维护,所以Hilo方式与底层数据库无关,但不应该手动修改hi/lo算法使用的表的值,否则会引起主键重复的异常。

    Increment
    Increment方式对主键值采取自动增长的方式生成新的主键值,但要求底层数据库的主键类型为long,int等数值型。主键按数值顺序递增,增量为1。
    /特点:由Hibernate本身维护,适用于所有的数据库,不适合多进程并发更新数据库,适合单一进程访问数据库。不能用于群集环境。/

    Identity
    Identity方式根据底层数据库,来支持自动增长,不同的数据库用不同的主键增长方式。

    特点:与底层数据库有关,要求数据库支持Identity,如MySQl中是auto_increment, SQL Server 中是Identity,支持的数据库有MySql、SQL Server、DB2、Sybase和HypersonicSQL。 Identity无需Hibernate和用户的干涉,使用较为方便,但不便于在不同的数据库之间移植程序。

    Sequence
    Sequence需要底层数据库支持Sequence方式,例如Oracle数据库等

    特点:需要底层数据库的支持序列,支持序列的数据库有DB2、PostgreSql、Oracle、SAPDb等在不同数据库之间移植程序,特别从支持序列的数据库移植到不支持序列的数据库需要修改配置文件。

    Native
    Native主键生成方式会根据不同的底层数据库自动选择Identity、Sequence、Hilo主键生成方式

    特点:根据不同的底层数据库采用不同的主键生成方式。由于Hibernate会根据底层数据库采用不同的映射方式,因此便于程序移植,项目中如果用到多个数据库时,可以使用这种方式。

    UUID
    UUID使用128位UUID算法生成主键,能够保证网络环境下的主键唯一性,也就能够保证在不同数据库及不同服务器下主键的唯一性。

    特点:能够保证数据库中的主键唯一性,生成的主键占用比较多的存贮空间

    Foreign GUID
    Foreign用于一对一关系中。GUID主键生成方式使用了一种特殊算法,保证生成主键的唯一性,支持SQL Server和MySQL

    包的作用

    net.sf.hibernate.*  [1]

    该包的类基本上都是接口类和异常类

    net.sf.hibernate.cache.*

    JCS的实现类

    net.sf.hibernate.cfg.*

    配置文件读取类

    net.sf.hibernate.collection.*

    Hibernate集合接口实现类,例如List,Set,Bag等等,Hibernate之所以要自行编写集合接口实现类是为了支持lazy loading

    net.sf.hibernate.connection.*

    几个数据库连接池的Provider

    net.sf.hibernate.dialect.*

    支持多种数据库特性,每个Dialect实现类代表一种数据库,描述了该数据库支持的数据类型和其它特点,例如是否有AutoIncrement,是否有Sequence,是否有分页sql等等

    net.sf.hibernate. eg.*

    Hibernate文档中用到的例子
    net.sf.hibernate.engine.*
    这个包的类作用比较散
    net.sf.hibernate.expression.*

    HQL支持的表达式

    net.sf.hibernate.hq.*

    HQL实现

    net.sf.hibernate. id.*

    ID生成器

    net.sf.hibernate.impl.*

    最核心的包,一些重要接口的实现类,如Session,SessionFactory,Query等

    net.sf.hibernate.jca.*

    JCA支持,把Session包装为支持JCA的接口实现类

    net.sf.hibernate.jmx.*

    JMX是用来编写App Server的管理程序的,大概是JMX部分接口的实现,使得App Server可以通过JMX接口管理Hibernate

    net.sf.hibernate.loader.*

    也是很核心的包,主要是生成sql语句的

    net.sf.hibernate.lob.*

    Blob和Clob支持

    net.sf.hibernate.mapping.*

    hbm文件的属性实现

    net.sf.hibernate.metadata.*

    PO的Meta实现

    net.sf.hibernate.odmg.*

    ODMG是一个ORM标准,这个包是ODMG标准的实现类

    net.sf.hibernate.persister.*

    核心包,实现持久对象和表之间的映射

    net.sf.hibernate.proxy.*

    Proxy和Lazy Loading支持

    net.sf.hibernate. ps.*

    该包是PreparedStatment Cache

    net.sf.hibernate.sql.*

    生成JDBC sql语句的包

    net.sf.hibernate.test.*

    测试类,你可以用junit来测试Hibernate

    net.sf.hibernate.tool.hbm2ddl.*
    用hbm配置文件生成DDL

    net.sf.hibernate.transaction.*

    Hibernate Transaction实现类

    net.sf.hibernate.type.*

    Hibernate中定义的持久对象的属性的数据类型

    net.sf.hibernate.util.*

    一些工具类,作用比较散

    net.sf.hibernate.xml.*

    XML数据绑定

    缓存管理

    Hibernate 中提供了两级Cache(高速缓冲存储器),第一级别的缓存是Session级别的缓存,它是属于事务范围的缓存。这一级别的缓存由hibernate管理的,一般情况下无需进行干预;第二级别的缓存是SessionFactory级别的缓存,它是属于进程范围或群集范围的缓存。这一级别的缓存可以进行配置和更改,并且可以动态加载和卸载。 Hibernate还为查询结果提供了一个查询缓存,它依赖于第二级缓存。
    
      第一级缓存 第二级缓存 存放数据的形式 相互关联的持久化对象的散装数据 缓存的范围
    
     事务范围,每个事务都有单独的第一级缓存进程范围或集群范围,缓存被同一个进程或集群范围内的所有事务共享 并发访问策略由于每个事务都拥有单独的第一级缓存,不会出现并发问题,无需提供并发访问策略由于多个事务会同时访问第二级缓存中相同数据,因此必须提供适当的并发访问策略,来保证特定的事务隔离级别数据过期策略没有提供数据过期策略。处于一级缓存中的对象永远不会过期,除非应用程序显式清空缓存或者清除特定的对象必须提供数据过期策略,如基于内存的缓存中的对象的最大数目,允许对象处于缓存中的最长时间,以及允许对象处于缓存中的最长空闲时间。
    
    
    缓存的软件实现 在Hibernate的Session的实现中包含了缓存的实现由第三方提供,Hibernate仅提供了缓存适配器(CacheProvider)。用于把特定的缓存插件集成到Hibernate中。启用缓存的方式只要应用程序通过Session接口来执行保存、更新、删除、加载和查询数据库数据的操作,Hibernate就会启用第一级缓存,把数据库中的数据以对象的形式拷贝到缓存中,对于批量更新和批量删除操作,如果不希望启用第一级缓存,可以绕过Hibernate API,直接通过JDBC API来执行指操作。用户可以在单个类或类的单个集合的粒度上配置第二级缓存。如果类的实例被经常读但很少被修改,就可以考虑使用第二级缓存。只有为某个类或集合配置了第二级缓存,Hibernate在运行时才会把它的实例加入到第二级缓存中。 用户管理缓存的方式第一级缓存的物理介质为内存,由于内存容量有限,必须通过恰当的检索策略和检索方式来限制加载对象的数目。Session的evit()方法可以显式清空缓存中特定对象,但这种方法不值得推荐。 第二级缓存的物理介质可以是内存和硬盘,因此第二级缓存可以存放大量的数据,数据过期策略的maxElementsInMemory属性值可以控制内存中的对象数目。管理第二级缓存主要包括两个方面:选择需要使用第二级缓存的持久类,设置合适的并发访问策略:选择缓存适配器,设置合适的数据过期策略。
    

    一级缓存
    当应用程序调用Session的save()、update()、saveOrUpdate()、get()或load(),以及调用查询接口的 list()、iterate()或filter()方法时,如果在Session缓存中还不存在相应的对象,Hibernate就会把该对象加入到第一级缓存中。当清理缓存时,Hibernate会根据缓存中对象的状态变化来同步更新数据库。 Session为应用程序提供了两个管理缓存的方法: evict(Object obj):从缓存中清除参数指定的持久化对象。 clear():清空缓存中所有持久化对象。

    二级缓存

    3.1. Hibernate的二级缓存策略的一般过程如下:

    1) 条件查询的时候,总是发出一条select * from table_name where …. (选择所有字段)这样的SQL语句查询数据库,一次获得所有的数据对象。

    2) 把获得的所有数据对象根据ID放入到第二级缓存中。

    3) 当Hibernate根据ID访问数据对象的时候,首先从Session一级缓存中查;查不到,如果配置了二级缓存,那么从二级缓存中查;查不到,再查询数据库,把结果按照ID放入到缓存。

    4) 删除、更新、增加数据的时候,同时更新缓存。

    Hibernate的二级缓存策略,是针对于ID查询的缓存策略,对于条件查询则毫无作用。为此,Hibernate提供了针对条件查询的Query Cache。

    3.2. 什么样的数据适合存放到第二级缓存中? 1 很少被修改的数据 2 不是很重要的数据,允许出现偶尔并发的数据 3 不会被并发访问的数据 4 参考数据,指的是供应用参考的常量数据,它的实例数目有限,它的实例会被许多其他类的实例引用,实例极少或者从来不会被修改

    3.3. 不适合存放到第二级缓存的数据? 1 经常被修改的数据 2 财务数据,绝对不允许出现并发 3 与其他应用共享的数据。

    3.4. 常用的缓存插件 Hibernater 的二级缓存是一个插件,下面是几种常用的缓存插件:
    l EhCache:可作为进程范围的缓存,存放数据的物理介质可以是内存或硬盘,对Hibernate的查询缓存提供了支持。

    l OSCache:可作为进程范围的缓存,存放数据的物理介质可以是内存或硬盘,提供了丰富的缓存数据过期策略,对Hibernate的查询缓存提供了支持。

    l SwarmCache:可作为群集范围内的缓存,但不支持Hibernate的查询缓存。

    l JBossCache:可作为群集范围内的缓存,支持事务型并发访问策略,对Hibernate的查询缓存提供了支持。

    这里写图片描述

    这里写图片描述

    在默认情况下,Hibernate使用EhCache进行JVM级别的缓存。用户可以通过设置Hibernate配置文件中的hibernate.cache.provider_class的属性,指定其他的缓存策略,该缓存策略必须实现org.hibernate.cache.CacheProvider接口。

    3.5. 配置二级缓存的主要步骤:

    1) 选择需要使用二级缓存的持久化类,设置它的命名缓存的并发访问策略。这是最值得认真考虑的步骤。

    2) 选择合适的缓存插件,然后编辑该插件的配置文件。

    延迟加载
    Hibernate对象关系映射提供延迟的与非延迟的对象初始化。非延迟加载在读取一个对象的时候会将与这个对象所有相关的其他对象一起读取出来。这有时会导致成百的(如果不是成千的话)select语句在读取对象的时候执行。这个问题有时出现在使用双向关系的时候,经常会导致整个数据库都在初始化的阶段被读出来了。当然,你可以不厌其烦地检查每一个对象与其他对象的关系,并把那些最昂贵的删除,但是到最后,我们可能会因此失去了本想在ORM工具中获得的便利。

    一个明显的解决方法是使用Hibernate提供的延迟加载机制。这种初始化策略只在一个对象调用它的一对多或多对多关系时才将关系对象读取出来。这个过程对开发者来说是透明的,而且只进行了很少的数据库操作请求,因此会得到比较明显的性能提升。这项技术的一个缺陷是延迟加载技术要求一个Hibernate会话要在对象使用的时候一直开着。这会成为通过使用DAO模式将持久层抽象出来时的一个主要问题。为了将持久化机制完全地抽象出来,所有的数据库逻辑,包括打开或关闭会话,都不能在应用层出现。最常见的是,一些实现了简单接口的DAO实现类将数据库逻辑完全封装起来了。一种快速但是笨拙的解决方法是放弃DAO模式,将数据库连接逻辑加到应用层中来。这可能对一些小的应用程序有效,但是在大的系统中,这是一个严重的设计缺陷,妨碍了系统的可扩展性。

    Web层延迟加载

    幸运的是,Spring框架为Hibernate延迟加载与DAO模式的整合提供了一
    种方便的解决方法。以一个Web应用为例,Spring提供了OpenSessionInViewFilterOpenSessionInViewInterceptor。我们可以随意选择一个类来实现相同的功能。两种方法唯一的不同就在于interceptor在Spring容器中运行并被配置在web应用的上下文中,而Filter在Spring之前运行并被配置在web.xml中。不管用哪个,他们都在请求将当前会话与当前(数据库)线程绑定时打开Hibernate会话。一旦已绑定到线程,这个打开了的Hibernate会话可以在DAO实现类中透明地使用。这个会话会为延迟加载数据库中值对象的视图保持打开状态。一旦这个逻辑视图完成了,Hibernate会话会在Filter的doFilter方法或者Interceptor的postHandle方法中被关闭。
    实现方法在web.xml中加入

    <filter>
    <filter-name>hibernateFilter</filter-name>
    <filter-class>
    org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
    </filter-class>
    </filter>
    <filter-mapping>
    <filter-name>hibernateFilter</filter-name>
    <url-pattern>*.do</url-pattern>
    </filter-mapping>
    

    性能优化

    初用HIBERNATE的人也许都遇到过性能问题,实现同一功能,用HIBERNATE与用JDBC性能相差十几倍很正常,如果不及早调整,很可能影响整个项目的进度。 大体上,对于HIBERNATE性能调优的主要考虑点如下:

    .数据库设计调整

    .HQL优化

    .API的正确使用(如根据不同的业务类型选用不同的集合及查询API)

    .主配置参数(日志,查询缓存,fetch_size, batch_size等)

    .映射文件优化(ID生成策略,二级缓存,延迟加载,关联优化)

    .一级缓存的管理

    .针对二级缓存,还有许多特有的策略

    .事务控制策略。

    数据库设计

    a) 降低关联的复杂性

    b) 尽量不使用联合主键

    c) ID的生成机制,不同的数据库所提供的机制并不完全一样

    d) 适当的冗余数据,不过分追求高范式

    HQL优化
    HQL如果抛开它同HIBERNATE本身一些缓存机制的关联,HQL的优化技巧同普通的SQL优化技巧一样,可以很容易在网上找到一些经验之谈

    主配置
    a) 查询缓存,同下面讲的缓存不太一样,它是针对HQL语句的缓存,即完全一样的语句再次执行时可以利用缓存数据。但是,查询缓存在一个交易系统(数据变更频繁,查询条件相同的机率并不大)中可能会起反作用:它会白白耗费大量的系统资源但却难以派上用场。

    b) fetch_size,同JDBC的相关参数作用类似,参数并不是越大越好,而应根据业务特征去设置

    c) batch_size同上。

    d) 生产系统中,切记要关掉SQL语句打印。

    缓存
    a) 数据库级缓存:这级缓存是最高效和安全的,但不同的数据库可管理的层次并不一样,比如,在Oracle中,可以在建表时指定将整个表置于缓存当中。

    b) SESSION缓存:在一个HibernateSESSION有效,这级缓存的可干预性不强,大多于HIBERNATE自动管理,但它提供清除缓存的方法,这在大批量增加/更新操作是有效的。比如,同时增加十万条记录,按常规方式进行,很可能会发现OutofMemeroy的异常,这时可能需要手动清除这一级缓存:Session.evict以及 Session.clear

    c) 应用缓存:在一个SESSIONFACTORY中有效,因此也是优化的重中之重,因此,各类策略也考虑的较多,在将数据放入这一级缓存之前,需要考虑一些前提条件:

    i. 数据不会被第三方修改(比如,是否有另一个应用也在修改这些数据?)

    ii. 数据不会太大

    iii. 数据不会频繁更新(否则使用CACHE可能适得其反)

    iv. 数据会被频繁查询

    v. 数据不是关键数据(如涉及钱,安全等方面的问题)。

    缓存有几种形式,可以在映射文件中配置:read-only(只读,适用于很少变更的静态数据/历史数据),nonstrict-read- write,read-write(比较普遍的形式,效率一般),transactional(JTA中,且支持的缓存产品较少)

    d) 分布式缓存:同c)的配置一样,只是缓存产品的选用不同,oscache, jboss cache,的大多数项目,对它们的用于集群的使用(特别是关键交易系统)都持保守态度。在集群环境中,只利用数据库级的缓存是最安全的。

    hibernate工作原理:
    1、通过Configuration().configure();读取并解析hibernate.cfg.xml配置文件。

    2、由hibernate.cfg.xml中的
    <mappingresource="com/xx/User.hbm.xml"/>读取解析映射信息。

    3、通过config.buildSessionFactory();//得到sessionFactory。

    4、sessionFactory.openSession();//得到session。

    5、session.beginTransaction();//开启事务。

    6、persistent operate;

    7、session.getTransaction().commit();//提交事务

    8、关闭session;

    9、关闭sessionFactory;

    hibernate优点:

    1、封装了jdbc,简化了很多重复性代码。

    2、简化了DAO层编码工作,使开发更对象化了。

    3、移植性好,支持各种数据库,如果换个数据库只要在配置文件中变换配置就可以了,不用改变hibernate代码。

    4、支持透明持久化,因为hibernate操作的是纯粹的(pojo)java类,没有实现任何接口,没有侵入性。所以说它是一个轻量级框架。

    hibernate延迟加载:

    get不支持延迟加载,load支持延迟加载。

    1、hibernate2对 实体对象和集合 实现了延迟加载

    2、hibernate3对 提供了属性的延迟加载功能

    hibernate延迟加载就是当使用session.load(User.class,1)或者
    session.createQuery()查询对象或者属性的时候

    这个对象或者属性并没有在内存中,只有当程序操作数据的时候,才会存在内存中,这样就实现延迟加载,节省了内存的开销,从而提高了服务器的性能。

    Hibernate的缓存机制
    一级缓存:session级的缓存也叫事务级的缓存,只缓存实体,生命周期和session一致。不能对其进行管理。
    不用显示的调用。

    二级缓存:sessionFactory缓存,也叫进程级的缓存,使用第3方插件实现的,也值缓存实体,生命周期和sessionFactory一致,可以进行管理。
    首先配置第3方插件,我们用的是EHCache,在hibernate.cfg.xml文件中加入

    <propertyname="hibernate.cache.user_second_level_cache">true</property>

    在映射中也要显示的调用,<cacheusage="read-only"/>
    二级缓存之查询缓存:对普通属性进行缓存。如果关联的表发生了修改,那么查询缓存的生命周期也结束了。
    在程序中必须手动启用查询缓存:query.setCacheable(true);

    优化Hibernate

    1、使用一对多的双向关联,尽量从多的一端维护。

    2、不要使用一对一,尽量使用多对一。

    3、配置对象缓存,不要使用集合缓存。

    4、表字段要少,表关联不要怕多,有二级缓存撑腰。

    hibernate 类与类之间关系

    关联关系

    聚集关系

    继承关系

    Hibernate继承关系映射策略分为三种:一张表对应一整棵类继承树、一个类对应一张表、每一个具体类对应一张表。

    展开全文
  • 直到某一天发现了Object-Relational Mapping(对象关系映射,简称ORM)框架Entity Framework,感谢微软又做了一件给码农减压的事情。  说了这么多,一些人可能对Entity Framework到底能够干什么还不是很清楚, 简单...

            以往,如果在程序中需要对数据库及其内容进行操作,我们都需要手动去写CURD的代码,很累,一大堆重复代码写的心情不爽。直到某一天发现了Object-Relational Mapping(对象关系映射,简称ORM)框架Entity Framework,感谢微软又做了一件给码农减压的事情。

            说了这么多,一些人可能对Entity Framework到底能够干什么还不是很清楚, 简单的说就是从此不用再写CURD的代码就可以直接操作数据库了,ORM使类提供自动化CRUD,使开发人员从数据库API和SQL中解放出来,还需要更深入的了解请移步What is Entity Framework

           本篇文章只是一步步的写一个简单的例子来说明到底怎么去用Entity Framework(下文中简称EF)。关于EF的介绍的文章很多先辈们已经写的很清楚了,EF支持三种开发模式,Code First、Database First、Model First。本文将采用Database First的方式描述一个完整的使用实例。

           1. 建立数据库。首先打开VS2015(2012以上版本都差不多),View-》Server Explorer,在Server Explorer视图中点击Data Connections右键Add Connection...将弹出如下Add Connection界面。需要配置Server name为:(localdb)\ProjectsV13,然后设置database name为你要建立的数据库的名称,最后点击ok创建数据库完成。其中server name这个值针对不同的localdb版本会不同,要查看你装的是哪个localdb版本只需要点击View-》SQL Server Object Explorer,在弹出的界面中会显示你所安装的localdb版本,找到对应的值即可。

    Add Connection界面图

         2.建立数据库表。第1步我们已经建立了localdb数据库,现在需要定义我们要使用的数据库表。展开新建的数据库,右键Tables,选择Add new table,然后会出现表格设计界面。我们可以手动设置表格的列名称和类型,完了之后需要在T-SQL里面把表名称给修改成我们定义的表名称,因为默认的表名是Table。完整的T-SQL代码如下,可以对照一下。

    CREATE TABLE [dbo].[User]
    (
        [Id] INT NOT NULL PRIMARY KEY, 
        [UserName] VARCHAR(50) NULL, 
        [Age] INT NULL
    )
    

    然后点击左上角的Update,稍等一会点击Update Database。表格创建完成了,数据库下面的Tables目录却要等一会才能刷新出新建的表,可以手动右键refresh一下就立即出来了。然后我们在新建好的表里面随便插入两条数据,后面会需要样例数据。编辑表格数据的方法是右键User(表名),点击Show table data,编辑完成数据,点击左上角刷新按键刷新一下即可。

       3.新建项目工程。为了简单,这里建立一个控制台程序。File-》new-》Project...-》Console Project。

       4.从数据库自动生成对象文件。Project-》Add new item-》Data-》ADO.Net Entity Data Model,在Name中我们输入EFContext(可随便定义)。然后点击Add。在弹出的界面中选择Code First from database,点击Next。弹出Choose Your Data Connection的界面,点击Next会弹出Choose Your Database Objects and Settings界面,勾选Tables,然后点击Finish。可以看到我们项目中会自动生成两个类文件EFContext.cs和User.cs,并修改了App.config文件。如果你细心会发现在App.config文件中增加了一个ConnectionStrings的配置项内容。

       5.现在就可以在程序中操作数据库了。打开Program.cs文件,在Main方法内插入操作代码,完成的文件代码如下所示。

    class Program
        {
            static void Main(string[] args)
            {
                using (EFContext efContext=new EFContext())
                {
                    var users = efContext.User;
                    foreach(var user in users)
                    {
                        Console.WriteLine("userName {0},Age {1}",user.UserName,user.Age);
                    }
                }
                Console.ReadKey();
            }
        }

    是不是神奇的打印出了表格中的内容。可是我们一行sql代码都没有写,这里的示例只是简单的实现了读数据库的内容,而且是先建立数据库模式(Database First),自己去发现更多更神奇的功能吧。

    总结:Entity Framework的功能很强大,建议你花一些时间了解下该框架。一篇很好的前辈写的Entity Framework 基础 很值得看看。



    展开全文
  • 待办事项清单(使用Enity框架进行对象关系映射实验的实践)
  • Eloquent 是 Laravel 的 ‘ORM’,即 ‘Object Relational Mapping’,对象关系映射。ORM 的出现是为了帮我们把对数据库的操作变得更加地方便。 Eloquent 让一个 ‘Model类’ 对应一张数据库表,并且在底层封装了很...
  • 开源对象关系映射框架Hibernate知识培训开源对象关系映射框架Hibernate知识培训
  • GreenDao是Android ORM(对象关系映射)框架,简化对数据库的操作,不用开发者自己拼装sql语句,减少出错率。开发者只需对实体类实例进行操作即可达到操作数据库的目的。Demo的GreenDao版本为3.2,通过注解方式生成...
  • 内容包括:1,培训的PPT Hibernate.ppt 2,Hibernate性能调优.pdf 3,架构风格与基于网络的软件架构设计.pdf
  • Android数据库框架——GreenDao轻量级的对象关系映射框架,永久告别sqlite 前不久,我在写了ORMLite这个框架的博文Android数据库框架——ORMLite轻量级的对象关系映射(ORM)Java包 但是对于我个人来说,我可能更加...

    Android数据库框架——GreenDao轻量级的对象关系映射框架,永久告别sqlite


    这里写图片描述

    前不久,我在写了ORMLite这个框架的博文

    但是对于我个人来说,我可能更加倾向于用GreenDao,所以今天也为大家带来了GreenDao的详细博客,希望大家喜欢,之前也详细介绍了ORM的思想,如果不明白可以先看一下前面的博文,这里就不多赘述了,我们新建一个工程

    这里写图片描述

    一.相关介绍

    传说中的优点

    • 最大性能(最快的 Android ORM)
    • 易于使用API
    • 高度优化
    • 最小内存消耗

    这里写图片描述

    那我们可以先来看看怎么集成吧!

    首先,我们查看Github上的集成步骤,先添加依赖
    我们要添加

     compile 'org.greenrobot:greendao:2.2.1'

    同时也要添加java的项目包

     compile 'org.greenrobot:greendao-generator:2.2.0'

    紧接着,我们在main目录下创建一个文件夹java-gen

    然后继续在配置文件中添加

    //仓库
    sourceSets{
            main{
                java.srcDirs = ['src/main/java','src/main/java-gen']
            }
        }

    最后,新建一个Module,选择JAVA Library,创建成功后,我们在他的build.gradle下添加

    compile 'org.greenrobot:greendao-generator:2.2.0'

    二.实体类

    到这里,我们的初步集成算是完成了,是不是比较麻烦,我们看

    这里写图片描述

    她说现在去看java-gen有生成代码,实际上我们去看并没有,因为我们需要NoteDao.java,一个数据缓存对象,那我们需要怎么去做呢?我们直接在Module的类里写了

    package com.example;
    
    import de.greenrobot.daogenerator.DaoGenerator;
    import de.greenrobot.daogenerator.Entity;
    import de.greenrobot.daogenerator.Schema;
    
    public class DaoMaker {
        public static void main(String[] args) {
            //生成数据库的实体类,还有版本号
            Schema schema = new Schema(1, "com.student.entity");
            addStudent(schema);
            //指定dao
            schema.setDefaultJavaPackageDao("com.student.dao");
            try {
                //指定路径
                new DaoGenerator().generateAll(schema, "D:\\github\\GreenDao\\app\\src\\main\\java-gen");
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
        /**
         * 创建数据库的表
         *
         * @param schema
         */
        public static void addStudent(Schema schema) {
            //创建数据库的表
            Entity entity = schema.addEntity("Student");
            //主键 是int类型
            entity.addIdProperty();
            //名称
            entity.addStringProperty("name");
            //年龄
            entity.addIntProperty("age");
            //地址
            entity.addStringProperty("address");
        }
    }
    

    写完这个时候我们要注意了,我们不能直接去运动这个工程,我们要单独编译这个java类。也就是右键

    这里写图片描述

    编译的时间有点点长,我们耐心等待一下就好了,这个时候我们可以看到控制台会打印相关的信息

    这里写图片描述

    这个时候你再去看java-gen目录,就会有东西了

    这里写图片描述

    很nice,这算是前期基本已经完工了,我们可以看他的原型图

    这里写图片描述

    三.核心类

    可以看到,GreenDao是有自己创建的类的,我们来看看是什么类

    • DaoSession:会话层,操作具体dDao类,提供了基本的持久化操作方法,比如对实体对象的insert,load,update,refresh,delete的一些操作

    • XXDao:实际生成的Dao类,通常对应具体的java类,greenDao为每一个实体类创建一个Dao,他提供了比DaoSession更为具体的付费,比如count,loadALL和inserlnTx,批量插入

    • xxEntity:持久的实体对象,通常代表数据库row标准的java属性

    • Schema:实例数据schema,通过schema的版本和缺省的java包调用构造器

    四.封装操作类

    OK,了解的差不多了,我们就来实战操作一下吧!实战的时候,我们不需要在主Activity中放太多的逻辑代码,全部封装在一个数据库的操作类中其实是最好的,所以我们先新建一个类

    package com.lgl.greendao;
    
    import android.content.Context;
    
    import com.student.dao.DaoMaster;
    import com.student.dao.DaoSession;
    
    import de.greenrobot.dao.query.QueryBuilder;
    
    /**
     * 数据库操作类
     * Created by LGL on 2016/7/2.
     */
    public class DaoManager {
    
    
        /**
         * 实现功能
         * 1.创建数据库
         * 2.创建数据库的表
         * 3.对数据库的升级
         * 4.对数据库的增删查改
         */
    
        //TAG
        public static final String TAG = DaoManager.class.getSimpleName();
        //数据库名称
        private static final String DB_NAME = "greendao.db";
        //多线程访问
        private volatile static DaoManager manager;
        //操作类
        private static DaoMaster.DevOpenHelper helper;
        //上下文
        private Context mContext;
        //核心类
        private static DaoMaster daoMaster;
        private DaoSession daoSession;
    
        //单例模式
        public static DaoManager getInstance() {
            DaoManager instance = null;
            if (manager == null) {
                synchronized (DaoManager.class) {
                    if (instance == null) {
                        instance = new DaoManager();
                        manager = instance;
                    }
                }
            }
            return instance;
        }
    
        //传递上下文
        public void  initManager(Context context){
            this.mContext = context;
        }
    
        /**
         * 判断是否存在数据库,如果没有则创建
         *
         * @return
         */
        public DaoMaster getDaoMaster() {
            if (daoMaster == null) {
                helper = new DaoMaster.DevOpenHelper(mContext, DB_NAME, null);
                daoMaster = new DaoMaster(helper.getWritableDatabase());
            }
            return daoMaster;
        }
    
        /**
         * 完成对数据库的操作,只是个接口
         *
         * @return
         */
        public DaoSession getDaoSession() {
            if (daoSession == null) {
                if (daoMaster == null) {
                    daoMaster = getDaoMaster();
                }
                daoSession = daoMaster.newSession();
            }
            return daoSession;
        }
    
        /**
         * 打开输出日志,默认关闭
         */
        public void setDebug() {
            QueryBuilder.LOG_SQL = true;
            QueryBuilder.LOG_VALUES = true;
        }
    
        /**
         * 关闭DaoSession
         */
        public void closeDaoSession() {
            if (daoSession != null) {
                daoSession.clear();
                daoSession = null;
            }
        }
    
        /**
         * 关闭Helper
         */
        public void closeHelper() {
            if (helper != null) {
                helper.close();
                helper = null;
            }
        }
    
        /**
         * 关闭所有的操作
         */
        public void closeConnection() {
            closeHelper();
            closeDaoSession();
        }
    
    
    
    
    }
    

    这个类能初始化数据库的很多操作,不过这样还不够,我们需要再写个实在点的操作类,现在只是单单实现一个插入的动作

    package com.lgl.greendao;
    
    import android.content.Context;
    
    import com.student.entity.Student;
    
    /**
     * 完成对某一张表的具体操作
     * Created by LGL on 2016/7/2.
     */
    public class CommonUtils {
    
        private DaoManager daoManager;
    
        //构造方法
        public CommonUtils(Context context) {
            daoManager = DaoManager.getInstance();
            daoManager.initManager(context);
        }
    
        /**
         * 对数据库中student表的插入操作
         * @param student
         * @return
         */
        public boolean insertStudent(Student student) {
            boolean flag = false;
            flag = daoManager.getDaoSession().insert(student) != -1 ? true : false;
            return flag;
        }
    }
    

    五.插入

    OK,那我们先去看看sql的插入是怎么做的,定义一个button

        <Button
            android:id="@+id/btn_add"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="添加数据" />

    我们只需要看他的点击事件就可以了

      case R.id.btn_add:
          dbUtils = new CommonUtils(this);
          Student student = new Student();
          student.setName("lgl");
          student.setAge(18);
          student.setAddress("北京");
          dbUtils.insertStudent(student);
        break;

    我们一运行之后,就可以在data/data/报名/databases中找到数据库了

    这里写图片描述

    只要把他到处到桌面上,就可以查看数据了

    这里写图片描述

    这里感觉不好,ID应该设置自动增长的对吧!其实只要不设置ID,他就会自增长的

    六.批量插入

    插入可以了,我们批量插入就简单了,CommonUtils中增加与一个方法

      /**
         * 批量插入
         *
         * @param students
         * @return
         */
        public boolean inserMultStudent(final List<Student> students) {
            //标识
            boolean flag = false;
            try {
                //插入操作耗时
                daoManager.getDaoSession().runInTx(new Runnable() {
                    @Override
                    public void run() {
                        for (Student student : students) {
                            daoManager.getDaoSession().insertOrReplace(student);
                        }
                    }
                });
                flag = true;
            } catch (Exception e) {
                e.printStackTrace();
            }
            return flag;
        }

    我们可以去验证一下吧,那首先先去写一个button吧

        <Button
            android:id="@+id/btn_add_mult"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="批量添加数据" />

    这样,我们就可以直接看他的点击事件了

             case R.id.btn_add_mult:
                    List<Student> list = new ArrayList<>();
                    for (int i = 0; i < 10; i++) {
                        Student studentlist = new Student();
                        studentlist.setName("lgl" + i);
                        studentlist.setAge(18 + i);
                        studentlist.setAddress("北京" + i);
                        list.add(studentlist);
                    }
                    dbUtils.inserMultStudent(list);
                    break;

    我们现在把数据库导出来看看

    这里写图片描述

    OK,这个非常的nice

    七.修改

    增删查改嘛,增加说完了,我们来说修改,我们继续写修改的方法然后去验证,这样对于读者来说也是阅读性比较好的

         /**
         * 修改
         * @param student
         * @return
         */
        public boolean updateStudent(Student student){
            boolean flag = false;
            try{
                daoManager.getDaoSession().update(student);
                flag = true;
            }catch (Exception e){
                e.printStackTrace();
            }
            return  flag;
        }

    OK,我们去验证一下,写个按钮咯?

         <Button
            android:id="@+id/btn_update"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="修改一条数据" />

    直接看点击事件

             case R.id.btn_update:
                    Student studentupdate = new Student();
                    //根据ID来修改
                    studentupdate.setId((long) 1);
                    //把年龄改成100岁
                    studentupdate.setAge(100);
                    dbUtils.updateStudent(studentupdate);
                    break;

    很好,很强势,我们导出数据库,看看有没有修改成100岁

    这里写图片描述

    好,那我们可以看到,虽然被改成了100,但是其他字段为null,因为我们修改的时候没有设置,好吧,设置一下

    这里写图片描述

    现在我们可以看到修改了

    八.删除

    删除就比较简单了,我们直接写

    /**
         * 删除
         *
         * @param student
         * @return
         */
        public boolean deleteStudent(Student student) {
            boolean flag = false;
            try {
                //删除指定ID
                daoManager.getDaoSession().delete(student);
                flag = true;
            } catch (Exception e) {
                e.printStackTrace();
            }
            return flag;
        }

    这样就写个按钮

        <Button
            android:id="@+id/btn_delete"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="删除一条数据" />

    OK,点击事件是比较简单的

             case R.id.btn_delete:
                    Student studentdelete = new Student();
                    studentdelete.setId((long) 3);
                    dbUtils.deleteStudent(studentdelete);
                    break;

    我们要删除的是他ID为3的数据,那我们运行

    这里写图片描述

    可以看到,ID为3的数据已经不见了

    九.查询

    OK,增删查改我们只剩下查询了,我们先来看一下怎么查询一条语句,写个方法

         /**
         * 查询单条
         *
         * @param key
         * @return
         */
        public Student listOneStudent(long key) {
            return daoManager.getDaoSession().load(Student.class, key);
        }

    我们再定义一个button

        <Button
            android:id="@+id/btn_load"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="查询一条数据" />

    这样我们就可以去查询了

    case R.id.btn_load:
      Log.i(TAG, dbUtils.listOneStudent(5) + "");
     break;

    十.全部查询

    全部查询比较简单

        /**
         * 全部查询
         *
         * @return
         */
        public List<Student> listAll() {
            return daoManager.getDaoSession().loadAll(Student.class);
        }

    直接去调用就好了

    case R.id.btn_load_all:
          List<Student> lists = dbUtils.listAll();
          Log.i(TAG, lists.toString());
      break;

    十一.QueryBuilder

    作为最重要的查询,GreenDao肯定是为我们做了很多事情,我们可以翻阅API找到QueryBuilder,这个复合查询该怎么用呢?我们根据条件来进行逻辑查询,原生的有,ORMLite也有,GreenDao也有,我就拿原生的和GreenDao来做对比

      /**
         * 原生查询
         */
        public void queryNative() {
            //查询条件
            String where = "where name like ? and _id > ?";
            //使用sql进行查询
            List<Student> list = daoManager.getDaoSession().queryRaw(Student.class, where,
                    new String[]{"%l%", "6"});
            Log.i(TAG, list + "");
        }
    
        /**
         * QueryBuilder查询大于
         */
        public void queryBuilder() {
            //查询构建器
            QueryBuilder<Student> queryBuilder = daoManager.getDaoSession().queryBuilder(Student.class);
            //查询年龄大于19的北京
            List<Student> list = queryBuilder.where(StudentDao.Properties.Age.ge(19)).where(StudentDao.Properties.Address.like("北京")).list();
            Log.i(TAG, list + "");
        }

    不难看出,使用起来是比较方便的,我们可以无限的加where条件查询

    十二.完结

    增删查改已经基本上都讲完了,那基本上这个框架也OK了,我们来看一下他运行的截图

    这里写图片描述

    逻辑说的还算是比较详细的,但是这个CommonUtils比较零散

    CommonUtils

    package com.lgl.greendao;
    
    import android.content.Context;
    import android.util.Log;
    
    import com.student.dao.StudentDao;
    import com.student.entity.Student;
    
    import java.util.List;
    
    import de.greenrobot.dao.query.QueryBuilder;
    
    /**
     * 完成对某一张表的具体操作
     * Created by LGL on 2016/7/2.
     */
    public class CommonUtils {
    
        //TAG
        private static final String TAG = CommonUtils.class.getSimpleName();
    
        private DaoManager daoManager;
    
        //构造方法
        public CommonUtils(Context context) {
            daoManager = DaoManager.getInstance();
            daoManager.initManager(context);
        }
    
        /**
         * 对数据库中student表的插入操作
         *
         * @param student
         * @return
         */
        public boolean insertStudent(Student student) {
            boolean flag = false;
            flag = daoManager.getDaoSession().insert(student) != -1 ? true : false;
            return flag;
        }
    
        /**
         * 批量插入
         *
         * @param students
         * @return
         */
        public boolean inserMultStudent(final List<Student> students) {
            //标识
            boolean flag = false;
            try {
                //插入操作耗时
                daoManager.getDaoSession().runInTx(new Runnable() {
                    @Override
                    public void run() {
                        for (Student student : students) {
                            daoManager.getDaoSession().insertOrReplace(student);
                        }
                    }
                });
                flag = true;
            } catch (Exception e) {
                e.printStackTrace();
            }
            return flag;
        }
    
        /**
         * 修改
         *
         * @param student
         * @return
         */
        public boolean updateStudent(Student student) {
            boolean flag = false;
            try {
                daoManager.getDaoSession().update(student);
                flag = true;
            } catch (Exception e) {
                e.printStackTrace();
            }
            return flag;
        }
    
        /**
         * 删除
         *
         * @param student
         * @return
         */
        public boolean deleteStudent(Student student) {
            boolean flag = false;
            try {
                //删除指定ID
                daoManager.getDaoSession().delete(student);
                flag = true;
            } catch (Exception e) {
                e.printStackTrace();
            }
            //daoManager.getDaoSession().deleteAll(); //删除所有记录
            return flag;
        }
    
        /**
         * 查询单条
         *
         * @param key
         * @return
         */
        public Student listOneStudent(long key) {
            return daoManager.getDaoSession().load(Student.class, key);
        }
    
        /**
         * 全部查询
         *
         * @return
         */
        public List<Student> listAll() {
            return daoManager.getDaoSession().loadAll(Student.class);
        }
    
        /**
         * 原生查询
         */
        public void queryNative() {
            //查询条件
            String where = "where name like ? and _id > ?";
            //使用sql进行查询
            List<Student> list = daoManager.getDaoSession().queryRaw(Student.class, where,
                    new String[]{"%l%", "6"});
            Log.i(TAG, list + "");
        }
    
        /**
         * QueryBuilder查询大于
         */
        public void queryBuilder() {
            //查询构建器
            QueryBuilder<Student> queryBuilder = daoManager.getDaoSession().queryBuilder(Student.class);
            //查询年龄大于19的北京
            List<Student> list = queryBuilder.where(StudentDao.Properties.Age.ge(19)).where(StudentDao.Properties.Address.like("北京")).list();
            Log.i(TAG, list + "");
        }
    }
    

    OK,这个博客算是结束了,我们下一篇再聊

    Demo下载:http://download.csdn.net/detail/qq_26787115/9566167

    欢迎加群:555974449

    展开全文
  • 针对现有的对象/关系映射组件不能实现动态改变数据类型的不足,构造了一个动态 O/R映射框架DORM。该框架包含了一个对象模型UDO和一种查询语言UQL。DORM不仅提供类型、对象、类型间的继承和组合关系的存储和处理,...
  • 前面介绍了ormlite操作单张表以及多表之间的一对多(多对一)的关系(请参考android对象关系映射框架ormlite学习之单表操作和android关系映射框架ormlite学习之OneToMany),但是我们还经常会遇到多对多的数据库表...
  • 基于对象关系映射的持久化框架实现了业务逻辑与数据访问的分离,简化了数据库访问,降低了应用开发难度。本文分析了基于Hibernate框架的两种对象关系映射策略,并比较了三种继承映射策略的优缺点,设计了一种基于...
  • 对于数据库, django 有自己的一套 ORM(对象关系映射), 或许其他的框架可以随意更换 ORM, 但 django 不建议这么做. 因为 django 内置有很多的 model, 这些 model 无疑是用 django 内置 ORM 实现的, 如果更换后, 内置...
  • mybatis框架——对象关系映射

    万次阅读 2018-07-30 19:37:48
    打个广告,帮朋友卖点东西,东西超便宜的哟【衣服鞋子等】,厂家直接出货,绝对低于市场价!!!一般都比市场价便宜3—7折【都是牌子货】,如果您感兴趣,可以扫描屏幕下方的...下面以商品订单为例,其关系图如...
  • python-sqlite-orm:SQLite的Python对象关系映射
  • CRUD-HibernateJPA JPA: Java Persistence API是标准... Hibernate: Hibernate是一种用Java语言编写的对象关系映射框架,它通过使用文件(XML)或Java注释来促进传统关系数据库和应用程序的对象模型之间的属性映射。
  • Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库。 Hibernate是一个优秀的持久化框架 什么是持久化:将程序数据在...
  • 主要介绍了研究Python的ORM框架中的SQLAlchemy库的映射关系,SQLAlchemy库是一个常见的Python中操作数据库的工具,需要的朋友可以参考下
  • Google开源的基于Android的Sqlite对象关系映射框架 ormlite 压缩包里包含 ormlite-android-4.30.jar 及帮助文档、ormlite-core-4.30.jar核心包 当然也支持在Java中的Jdbc中使用类似Hibernate Sqlite驱动包、ormlite...
  • HarmonyOS 对象关系映射(Object Relational Mapping,ORM)数据库是一款基于 SQLite 的数据库框架,屏蔽了底层 SQLite 数据库的 SQL 操作,针对实体和关系提供了增删改查等一系列的面向对象接口。应用开发者不必再...
  • Hibernate对象关系映射详解之“一对多”关系映射 之前学习Hibernate框架的时候,对这七大关系映射一直是云里雾里的,虽然可以仿照写出代码,但是不能独立编写出来。鉴于工作中这个知识点使用的几率还是非常大的,...
  • ORM(对象关系映射)

    千次阅读 2019-03-01 13:42:41
    用于实现面向对象编程语言里不同类型系统的数据之间的转换,换言之,就是用面向对象的方式去操作数据库的创建表以及增删改查等操作 二、优缺点 优点: 1.ORM使得我们的通用数据库交互变得简单易行,而且完全不用考虑...

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 206,102
精华内容 82,440
关键字:

对象关系映射框架

友情链接: opd_trans.rar