【Hibernate】持久化对象的状态的变化和对应的方法

本文探讨了Hibernate框架中对象的四种状态及其转换,并通过实例对比了get与load方法的不同行为。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一)持久化对象的状态变化和对应的方法

(1)持久化对象在整个hibernate框架中运行,一共有四种不同的状态产生:
>>临时状态对象:
 A)在session一级缓存之【外】
          B)【没有】与数据库交互的能力
          C)hibernate【不会】为临时状态对象分配oid值  
>>持久化状态对象 :
 A)在session一级缓存之【内】
          B)【有】与数据库交互的能力,即CURD
          C)hibernate【会】为持久化状态对象分配oid值  

>>游离状态对象:
 A)在session一级缓存之【外】
          B)【没有】与数据库交互的能力
          C)hibernate依然存在【已分配】oid值  
>>删除状态对象
 A)在session一级缓存之【外】
          B)【没有】与数据库交互的能力
          C)hibernate依然存在【已分配】oid值  
>>GC在适当的时候回收删除状态对象,但是GC在特定的的时候,也会回收:临时状态对象,游离状态对象.

CURD方法使用的细节:
>>持久化状态对象,不能同时出现二个相同的OID值.
        >>当持久化状态对象在session一级缓中,就必须有一条记录与之一一对应.
        >>你可以使用saveOrUpdate()去替换save()和update()方法 .
          这时saveOrUpdate(临时状态对象)的话,将执行save()操作.
          这时saveOrUpdate(游离状态对象)的话,将执行update()操作.


下面演示在session一级缓存内调用get()和load()方法的区别:

贴出javabean/实体;Customers类:

package cn.lsh.web.domain;


public class Customers {
	private int id;
	private String name;
	private String gender;
	private int age;
	private String des;
	public Customers() {
		super();	
	}
	
	public Customers(int id, String name, String gender, int age, String des) {
		super();
		this.id = id;
		this.name = name;
		this.gender = gender;
		this.age = age;
		this.des = des;
	}
	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}

	public int getId() {
		return id;
	}
	public void setId(int id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getGender() {
		return gender;
	}
	public void setGender(String gender) {
		this.gender = gender;
	}
	public String getDes() {
		return des;
	}
	public void setDes(String des) {
		this.des = des;
	}
	
	
}
实体所对应的映射文件:

<!DOCTYPE hibernate-mapping PUBLIC 
    "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
    
    <hibernate-mapping package="cn.lsh.web.domain">
    	<class name="Customers" table="CUSTOMERS">
    		<id name="id" column="ID" type="int"><generator class="increment"/></id>
    		<property name="name" column="NAME" type="string"></property>
    		<property name="gender" column="GENDER" type="string"></property>
    		<property name="age" column="AGE" type="int"></property>
    		<property name="des" column="DES" type="string"></property>
    	</class>
    </hibernate-mapping>

假设我的数据库添加的数据为两条数据记录(那么所对应的第三条记录是不存在的):

<span style="font-size:14px;">mysql> select * from customers;
+----+------+--------+------+------------------+
| id | name | gender | age  | des              |
+----+------+--------+------+------------------+
|  1 | 刘备 | 男     |   25 | 三国杰出人物之一 |
|  2 | 孙权 | 男     |   35 | 三国杰出人物之一 |
+----+------+--------+------+------------------+</span>

测试:CustomerDao类(使用junit测试)

get()方法测试:

public class CustomerDao {
	private static SessionFactory sessionFactory;

	static{
		//加载映射文件
		Configuration config = new Configuration().configure();
		//创建session父类工厂
		sessionFactory = config.buildSessionFactory();
	}
	
	//添加
	@Test
	public void addCustomer(){
			Session session = sessionFactory.openSession();
			Customers c = new Customers(2,"曹操","男",30,"三国杰出人物之一");
			Transaction t = session.beginTransaction();
			try {
				session.save(c);
				t.commit();
			} catch (Exception e) {
				e.printStackTrace();
				t.rollback();
			}finally{
				session.close();
			}
		}
<span style="white-space:pre">	</span>//查找1
<span style="white-space:pre">	</span>@Test
<span style="white-space:pre">	</span>public void findCustomer1(){
<span style="white-space:pre">			</span>Session session = sessionFactory.openSession();<span style="white-space:pre">		</span>
<span style="white-space:pre">			</span>//开启事物
<span style="white-space:pre">			</span>Transaction t = session.beginTransaction();
<span style="white-space:pre">			</span>try {<span style="white-space:pre">	</span>
<span style="white-space:pre">				</span>//使用get()方法查询数据库对应的第三条数据(数据库中第三条数据不存在)
<span style="white-space:pre">				</span>Customers c1 = (Customers) session.get(Customers.class,3);
<span style="white-space:pre">				</span>System.out.println(c1.getName()+":"+c1.getDes());
<span style="white-space:pre">				</span>t.commit();
<span style="white-space:pre">			</span>} catch (Exception e) {
<span style="white-space:pre">				</span>e.printStackTrace();
<span style="white-space:pre">				</span>t.rollback();
<span style="white-space:pre">			</span>}finally{
<span style="white-space:pre">				</span>session.close();
<span style="white-space:pre">			</span>}
<span style="white-space:pre">		</span>}
<span style="white-space:pre">	</span>}
结果报异常(空指针异常):java.lang.NullPointerException
at cn.lsh.web.dao.CustomerDao.findCustomer(CustomerDao.java:61)

load()方法测试:

//查找2
	@Test
	public void findCustomer2(){
			Session session = sessionFactory.openSession();
			Transaction t = session.beginTransaction();
			try {
				Customers c1 = (Customers) session.load(Customers.class,1);
				System.out.println(c1.getName()+":"+c1.getDes());
				t.commit();
			} catch (Exception e) {
				e.printStackTrace();
				t.rollback();
			}finally{
				session.close();
			}
		}
	
	}
结果报异常(对象找不到异常):org.hibernate.ObjectNotFoundException: No row with the given identifierexists[cn.lsh.web.domain.Customers#3]

总结:

1.如果get方法找不到对应的记录时,返回java.lang.NullPointerException;
   如果get方法找得到对应的记录时,会自动封装成对象,将该对象返回.

2.如果load方法找不到对应的记录时,返回org.hibernate.ObjectNotFoundException;
   如果load方法找得到对应的记录时,会自动封装成对象,将该对象返回.

基于数据挖掘的音乐推荐系统设计与实现 需要一个代码说明,不需要论文 采用python语言,django框架,mysql数据库开发 编程环境:pycharm,mysql8.0 系统分为前台+后台模式开发 网站前台: 用户注册, 登录 搜索音乐,音乐欣赏(可以在线进行播放) 用户登陆时选择相关感兴趣的音乐风格 音乐收藏 音乐推荐算法:(重点) 本课题需要大量用户行为(如播放记录、收藏列表)、音乐特征(如音频特征、歌曲元数据)等数据 (1)根据用户之间相似性或关联性,给一个用户推荐与其相似或有关联的其他用户所感兴趣的音乐; (2)根据音乐之间的相似性或关联性,给一个用户推荐与其感兴趣的音乐相似或有关联的其他音乐。 基于用户的推荐基于物品的推荐 其中基于用户的推荐是基于用户的相似度找出相似相似用户,然后向目标用户推荐其相似用户喜欢的东西(你类似的人也喜欢**东西); 而基于物品的推荐是基于物品的相似度找出相似的物品做推荐(喜欢该音乐的人还喜欢了**音乐); 管理员 管理员信息管理 注册用户管理,审核 音乐爬虫(爬虫方式爬取网站音乐数据) 音乐信息管理(上传歌曲MP3,以便前台播放) 音乐收藏管理 用户 用户资料修改 我的音乐收藏 完整前后端源码,部署后可正常运行! 环境说明 开发语言:python后端 python版本:3.7 数据库:mysql 5.7+ 数据库工具:Navicat11+ 开发软件:pycharm
MPU6050是一款广泛应用在无人机、机器人运动设备中的六轴姿态传感器,它集成了三轴陀螺仪三轴加速度计。这款传感器能够实时监测并提供设备的角速度线性加速度数据,对于理解物体的动态运动状态至关重要。在Arduino平台上,通过特定的库文件可以方便地与MPU6050进行通信,获取并解析传感器数据。 `MPU6050.cpp``MPU6050.h`是Arduino库的关键组成部分。`MPU6050.h`是头文件,包含了定义传感器接口函数声明。它定义了类`MPU6050`,该类包含了初始化传感器、读取数据等方法。例如,`begin()`函数用于设置传感器的工作模式I2C地址,`getAcceleration()``getGyroscope()`则分别用于获取加速度角速度数据。 在Arduino项目中,首先需要包含`MPU6050.h`头文件,然后创建`MPU6050`对象,并调用`begin()`函数初始化传感器。之后,可以通过循环调用`getAcceleration()``getGyroscope()`来不断更新传感器读数。为了处理这些原始数据,通常还需要进行校准滤波,以消除噪声漂移。 I2C通信协议是MPU6050与Arduino交互的基础,它是一种低引脚数的串行通信协议,允许多个设备共享一对数据线。Arduino板上的Wire库提供了I2C通信的底层支持,使得用户无需深入了解通信细节,就能方便地与MPU6050交互。 MPU6050传感器的数据包括加速度(X、Y、Z轴)角速度(同样为X、Y、Z轴)。加速度数据可以用来计算物体的静态位置动态运动,而角速度数据则能反映物体转动的速度。结合这两个数据,可以进一步计算出物体的姿态(如角度角速度变化)。 在嵌入式开发领域,特别是使用STM32微控制器时,也可以找到类似的库来驱动MPU6050。STM32通常具有更强大的处理能力更多的GPIO口,可以实现更复杂的控制算法。然而,基本的传感器操作流程数据处理原理与Arduino平台相似。 在实际应用中,除了基本的传感器读取,还可能涉及到温度补偿、低功耗模式设置、DMP(数字运动处理器)功能的利用等高级特性。DMP可以帮助处理传感器数据,实现更高级的运动估计,减轻主控制器的计算负担。 MPU6050是一个强大的六轴传感器,广泛应用于各种需要实时运动追踪的项目中。通过 Arduino 或 STM32 的库文件,开发者可以轻松地与传感器交互,获取并处理数据,实现各种创新应用。博客其他开源资源是学习解决问题的重要途径,通过这些资源,开发者可以获得关于MPU6050的详细信息实践指南
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值