hibernate实体的状态java
实体Entity有三种状态,瞬时状态,持久状态,脱管状态sql
瞬时状态:transient,session 没有缓存,数据库也没有记录,oid没有值数据库
持久状态:persistent,session有缓存,数据库也有记录,oid有值缓存
脱管状态:detached,session没有缓存,数据库有记录,oid有值session
瞬时状态转持久状态dom
public void test1(){ Configuration cfg = new Configuration().configure(); //建立会话工厂 SessionFactory factory = cfg.buildSessionFactory(); //获取session对象 Session session = factory.openSession(); session.getTransaction().begin(); //建立一个对象,这个对象就是瞬时状态 User user = new User("迪丽热巴","123");//没有id,数据库没有数据session没有缓存 System.out.println(user);//User [uid=0, username=迪丽热巴, password=123] session.save(user);//通过保存后,这个对象就是持久状态,id有值,数据库有数据,session有缓存 System.out.println(user);//User [uid=1, username=迪丽热巴, password=123] session.getTransaction().commit(); session.clear(); }
持久状态转脱管状态优化
public void test2(){ Configuration cfg = new Configuration().configure(); //建立会话工厂 SessionFactory factory = cfg.buildSessionFactory(); //获取session对象 Session session = factory.openSession(); session.getTransaction().begin(); //经过get方法能够获取一个持久状态对象 User user = (User) session.get(User.class,1);//执行select语句 System.out.println(user); User user1 = (User) session.get(User.class,1);//不执行select语句,由于有session缓存 System.out.println(user1); session.getTransaction().commit(); session.clear(); }
把持久状态转脱管状态须要使用session.clear方法ui
User user = (User) session.get(User.class,1);//执行select语句 System.out.println(user); session.clear();//清除session User user1 = (User) session.get(User.class,1);//执行select语句,由于没有session缓存 System.out.println(user1);
执行两次selecthibernate
Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.birthday as birthday0_0_ from t_user user0_ where user0_.id=? User [uid=1, username=迪丽热巴, password=123] Hibernate: select user0_.id as id0_0_, user0_.username as username0_0_, user0_.password as password0_0_, user0_.birthday as birthday0_0_ from t_user user0_ where user0_.id=? User [uid=1, username=迪丽热巴, password=123]
一级缓存,快照,一级缓存刷新code
一级缓存:又称为session级别的缓存,当得到一侧会话(session),hibernate在session中建立多个集合(Map),用于存放操做数据(PO对象),为程序优化服务,若是以后须要相应的数据,hibernate优先从session缓存中获取,若是有就使用:若是没有在查询数据库,当session关闭时,一级缓存销毁
快照:与一级缓存同样的存放位置,对一级缓存数据备份,保证数据库的数据与一级缓存的数据必须一致。若是一级缓存修改了,在执行commit提交时,将自动刷新一级缓存,执行update语句,将一级缓存的数据更新到数据库
使用HQL会对数据进行一级缓存,使用SQL不会对数据进行缓存
Query query = session.createQuery("from User");//HQL对数据进行一级缓存 List<User> list = query.list(); User user = (User)session.get(User.class,1);//有缓存,不用执行select语句,就能获取数据
save和persist方法区别
save方法:瞬时态 转换 持久化 会初始化OID
执行save方法前,设置OID将忽略
若是执行查询,session缓存移除了,在执行save方法,将会执行insert
public void test1(){ Configuration cfg = new Configuration().configure(); SessionFactory factory = cfg.buildSessionFactory(); Session session = factory.openSession(); session.getTransaction().begin(); User user = new User("迪丽热巴","123"); user.setUid(13); System.out.println(user);//[uid=13, username=迪丽热巴, password=123] //执行save方法前,设置的OID将会忽略, session.save(user);//瞬时态转持久态,执行save方法会当即出发insert语句,从数据库获取主键的值 System.out.println(user);//User [uid=1, username=迪丽热巴, password=123] //执行session.clear,将会把缓存移除,再次执行save方法,将会执行insert,sql语句 session.clear(); session.save(user);//再次保存,id会自动增加 System.out.println(user);//[uid=2, username=迪丽热巴, password=123] session.getTransaction().commit();//当事务提交时会自动自行update语句 session.clear(); }
persist方法:瞬时态,转换持久态
persist保存的对象,在保存前,不能设置id,不然会报错
save和persist都是持久化对象的做用
save由于须要返回一个主键值,所以会当即执行Insert语句,
而persist在事务外部调用是则不会当即执行insert语句
在事务内部调用仍是会当即执行insert语句的
public void test1(){ Configuration cfg = new Configuration().configure(); SessionFactory factory = cfg.buildSessionFactory(); Session session = factory.openSession(); session.getTransaction().begin(); User user = new User("古娜力扎","123"); //user.setUid(30);persist保存的对象,在保存前,不能设置id,不然会报错 session.persist(user);//该方法在事务外不会执行insert sql语句 session.getTransaction().commit();//当事务提交时会自动自行update语句 session.clear(); }
一对多实体类和映射文件
多表关系
多对多:好比老师对应多个学生,学生对应多个老师,有中间表的关系
一对多:一个客户对应多个订单,表之间有主表和从表,之间的关系(主外键关系)
一个客户对应多个订单
客户与订单文件的映射
<!-- 描述一对多的关系 key中column写的是外键名称 one-to-many:一对多,里面写的class,写多的一方 --> <set name="orders"> <key column="customer_id"></key> <one-to-many class="domain.Order"></one-to-many> </set>
<!--描述customer的关系 class写一的一方 column:写外键 --> <many-to-one name="customer" class="domain.Customer" column="customer_id"/>