2006 年夏天发布的 EJB 3.0 规范提供了一个大大简化但功能更为强大的 EJB 框架,该框架演示了批注与传统 EJB 2.x 部署描述符相比的显著优势。J2SE 5.0 中引入的批注是修饰符,可以在类、字段、方法、参数、本地变量、构造符、枚举和程序包中使用。大量 EJB 3.0 新特性中都强调了批注的使用,这些特性包括:基于普通旧式 Java 对象的 EJB 类、EJB 管理器类的相关性注入、引入可以拦截其他业务方法调用的拦截器或方法,以及显著增强的 Java 持久性 API (JPA) 等。
为了说明 JPA 的概念,我们来看一个实际示例。最近,我的办公室需要实施税务登记系统。与大多数系统一样,该系统具有自己的复杂性和挑战性。由于其特殊的挑战涉及了数据访问和对象关系映射 (ORM),因此我们决定在实施该系统的同时试用新的 JPA。
在该项目期间,我们面临以下几个挑战:
- 应用程序中使用的实体之间存在多种关系。
- 应用程序支持对关系数据进行复杂搜索。
- 应用程序必须确保数据完整性。
- 应用程序在持久保存数据之前需要对其进行验证。
- 需要批量操作。
数据模型
首先来看看我们的关系数据模型的简化版本,该版本足以解释 JPA 的细微之处。从业务角度而言,主申请人提交税务登记申请。申请人可以有零个或多个合伙人。申请人和合伙人必须指定两个地址,即注册地址和经营地址。主申请人还必须声明和描述其过去受到的所有处罚。
定义实体。我们通过将实体映射到单独的表定义了以下实体:
实体 |
映射到的表 |
Registration |
REGISTRATION |
Party |
PARTY |
Address |
ADDRESS |
Penalty |
PENALTY |
CaseOfficer |
CASE_OFFICER |
表 1. 实体-表映射
识别要映射到数据库表和列的实体很容易。下面是一个简化的 Registration 实体示例。(我将在后面介绍该实体的其他映射和配置。)
@Entity
@Table(name="REGISTRATION")
public class Registration implements Serializable{
@Id
private int id;
@Column(name="REFERENCE_NUBER")
private String referenceNumber;
..........
}
对我们而言,使用 JPA 实体的主要好处是我们感觉就像对常规的 Java 类进行编码一样:无需再使用复杂的生命周期方法。我们可以使用批注将持久性特性分配给实体。我们发现无需使用其他数据传输对象 (DTO) 层,并且可以重用实体以便在层之间移动。数据的可移动性突然变得更好了。
支持多态性。通过查看我们的数据模型,我们注意到我们使用了 PARTY 表同时存储申请人和合伙人记录。这些记录不但具有一些相同的属性,而且还具有各自特有的属性。
我们希望在继承层次中对此模型进行建模。利用 EJB 2.x,我们只能使用一个 Party 实体 bean,然后通过在代码内实施逻辑来根据 party 类型创建申请人或合伙人对象。另一方面,JPA 使我们可以在实体级别指定继承层次。
我们决定通过一个抽象的实体 Party 和两个具体的实体 Partner 和 Applicant 对继承层次进行建模:
@Entity
@Table(name="PARTY_DATA")
@Inheritance(strategy= InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name="PARTY_TYPE")
public abstract class Party implements Serializable{
@Id
protected int id;
@Column(name="REG_ID")
protected int regID;
protected String name;
.........
}
两个具体的类 Partner 和 Applicant 现在将继承抽象的 Party 类的特征。
@Entity
@DiscriminatorValue("0")
public class Applicant extends Party{
@Column(name="TAX_REF_NO")
private String taxRefNumber;
@Column(name="INCORP_DATE")
private String incorporationDate;
........
}
如果 party_type 列的值为 0,则持久性提供程序将返回一个 Applicant 实体的实例;如果该列的值为 1,持久性提供程序将返回一个 Partner 实体的实例。
构建关系。我们的应用程序数据模型中的 PARTY 表包含 REGISTRATION 表的一个外键列 (reg_id)。在该结构中,Party 实体成为实体的拥有方或关系的源,因为我们在其中指定连接列。Registration 成为关系的目标。
每个 ManyToOne 关系都很可能是双向的;即两个实体之间还存在 OneToMany 关系。下表显示我们的关系定义:
关系 |
拥有方 |
多重性/映射 |
Registration->CaseOfficer |
CaseOfficer |
OneToOne |
Registration->Party |
Party |
ManyToOne |
Party->Address |
Address |
ManyToOne |
Party->Penalty |
Penalty |
ManyToOne |
反向关系 |
Registration->CaseOfficer |
|
OneToOne |
Registration->Party |
|
OneToMany |
Party->Address |
|
OneToMany |
Party->Penalty |
|
OneToMany |
表 2.关系
public class Registration implements Serializable{
....
@OneToMany(mappedBy = "registration")
private Collection<Party> parties;
....
}
public abstract class Party implements Serializable{
....
@ManyToOne
@JoinColumn(name="REG_ID")
private Registration registration;
....
注意:mappedBy 元素指明连接列是在关系的另一端指定的。
接下来,我们需要考虑由 JPA 规范定义、持久性提供程序实施的关系的行为。我们希望如何获取相关数据,EAGER 还是 LAZY?我们查看了由 JPA 定义的关系的默认 FETCH 类型,然后向表 2 中添加了额外的一列以包括我们的发现:
关系 |
拥有方 |
多重性/映射 |
默认的 FETCH 类型 |
Registration->CaseOfficer |
CaseOfficer |
OneToOne |
EAGER |
Party->Registration |
Party |
ManyToOne |
EAGER |
Address->Party |
Address |
ManyToOne |
EAGER |
Penalty->Party |
Penalty |
ManyToOne |
EAGER |
|
|
|
|
反向关系 |
Registration->Party |
|
OneToMany |
LAZY |
Party->Address |
|
OneToMany |
LAZY |
Party->Penalty |
|
OneToMany |
LAZY |
表 3. 设置默认的 FETCH 类型
通过查看业务要求,似乎当我们获得 Registration 详细信息后,我们总是需要显示与该登记相关联的 Party 的详细信息。如果将 FETCH 类型设置为 LAZY,我们需要反复调用数据库以获取数据。这意味着,如果将 Registration->Party 关系的 FETCH 类型改为 EAGER,我们会获得更好的性能。在该设置下,持久性提供程序将相关数据作为单个 SQL 的一部分返回。
同样,当我们在屏幕上显示 Party 详细信息时,我们需要显示其相关联的 Address。因此,将 Party-Address 关系改为使用 EAGER 获取类型是很有帮助的。
另一方面,我们可以将 Party->Penalty 关系的 FETCH 类型设为 LAZY,因为我们不需要显示处罚的详细信息,除非用户这样要求。如果我们使用了 EAGER 获取类型,当 m 个当事人每人有 n 个处罚时,我们最终就要加载 m*n 个 Penalty 实体,这会产生不必要的大对象图形,从而降低性能。
public class Registration implements Serializable{
@OneToMany(mappedBy = "registration", fetch = FetchType.EAGER)
private Collection<Party> parties;
.....
}
public abstract class Party implements Serializable{
@OneToMany (mappedBy = "party", fetch = FetchType.EAGER)
private Collection<Address> addresses;
@OneToMany (mappedBy = "party", fetch=FetchType.LAZY)
private Collection<Penalty> penalties;
.....
}
访问惰性关系。考虑使用惰性加载方法,请考虑持久性上下文的范围。您可以在 EXTENDED 持久性上下文或 TRANSACTION 范围内的持久性上下文之间进行选择。EXTENDED 持久性上下文在事务之间保持活动状态,作用非常类似会话状态的会话 bean。
由于我们的应用程序不是会话式的,持久性上下文不需要在事务之间可持续;因此,我们决定使用 TRANSACTION 范围内的持久性上下文。但是,这带来了惰性加载的问题。获取了实体并结束了事务之后,就可以分离实体了。在我们的应用程序中,尝试加载任何以惰性方式加载的关系数据将产生未定义的行为。
大多数情况下,当办事员检索登记数据时,我们不需要显示处罚记录。但是对于管理员,我们需要额外显示处罚记录。考虑到大多数情况下,我们不需要显示处罚记录,将关系的 FETCH 类型更改为 EAGER 就没什么意义了。相反,我们可以通过检测经营者使用系统的时间来触发关系数据的惰性加载。这会使关系数据在实体已分离时也可用,并可以在以后进行访问。下面的示例解释了这个概念:
Registration registration = em.find(Registration.class, regID);
Collection<Party> parties = registration.getParties();
for (Iterator<Party> iterator = parties.iterator(); iterator.hasNext();) {
Party party = iterator.next();
party.getPenalties().size();
}
return registration;
在上面的示例中,我们只调用 Party 实体的处罚集合的 size() 方法。这样做确实有效并且触发了惰性加载,即使在 Registration 实体分离时,所有集合也会填充并可用。(或者,您可以使用 JP-QL 的一个名为 FETCH JOIN 的特殊特性,我们会在本文的后面对此进行讨论。)
关系和持久性
接下来,我们需要考虑关系在持久保存数据的上下文中的行为方式。本质上讲,如果对关系数据进行了任何更改,我们希望在对象级别进行同样的更改并通过持久性提供程序持久保存这些更改。在 JPA 中,我们可以使用 CASCADE 类型控制持久性行为。
JPA 中定义了四种 CASCADE 类型:
- PERSIST:持久保存拥有方实体时,也会持久保存该实体的所有相关数据。
- MERGE:将分离的实体重新合并到活动的持久性上下文时,也会合并该实体的所有相关数据。
- REMOVE:删除一个实体时,也会删除该实体的所有相关数据。
- ALL:以上都适用。
创建实体。 我们决定在所有情况下,当我们新建一个父实体时,我们希望其所有相关的子实体也自动持久保存。这简化了编码:我们只需正确设置关系数据,而无需在每个实体上单独调用 persist() 操作。这意味着简化了编码,因为我们只需正确设置关系数据,而无需在每个实体上单独调用 persist() 操作。
因此,级联类型 PERSIST 是对我们最具吸引力的选项。我们将所有关系定义重新调整为使用该选项。
更新实体 在事务内获取数据,然后在事务外对实体进行更改并持久保存更改,这是很常见的。例如,在我们的应用程序中,用户可以检索现有的登记,更改主申请人的地址。当我们获取一个现有的 Registration 实体并因此获取了该实体在特定事务内的所有相关数据时,事务在此处结束,数据被发送到表示层。此时,该 Registration 以及所有其他相关的实体实例与持久性上下文相分离。
在 JPA 中,为了持久保存分离实体上的更改,我们使用 EntityManager 的 merge() 操作。此外,为将更改传播到关系数据,所有关系定义必须包括 CASCADE 类型 MERGE 以及关系映射的配置中定义的任何其他 CASCADE 类型。
在该背景下,我们确保了为所有关系定义指定了正确的 CASCADE 类型。
删除实体。 接下来,我们需要确定删除某些实体时会发生什么。例如,如果我们删除一个 Registration,我们可以安全地删除与该 Registration 相关联的所有 Party。但是反过来却不是这样。此处的技巧是通过在关系上级联 remove() 操作以避免意外删除实体。正如您将在下一部分中看到的那样,由于引用完整性约束,这样的操作可能不会成功。
我们得出以下结论:在遵循 OnetoMany 的清晰的父子关系中(如 Party 和 Address 或 Party 和 Penalty),仅在关系的父 (ONE) 方指定 CASCADE 类型 REMOVE 是安全的。然后,我们对关系定义进行了相应的重新调整。
public abstract class Party implements Serializable{
@OneToMany (mappedBy = "party", fetch = FetchType.EAGER, cascade =
{CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE})
private Collection<Address> addresses;
@OneToMany (mappedBy = "party", fetch=FetchType.LAZY, cascade =
{CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE})
private Collection<Penalty> penalties;
.....
}
管理关系
根据 JPA,管理关系是程序员的唯一职责。持久性提供程序不承担有关关系数据状态的任何事情,因此它们不尝试管理关系。
假定了该事实,我们重新检查了我们用来管理关系和查明潜在问题区域的策略。我们发现:
- 如果我们尝试设置父级和级子之间的关系,但父级不再存在于数据库中(可能被其他用户删除),这将导致数据完整性问题。
- 如果我们尝试删除一条父记录而没有首先删除其子记录,将违反引用完整性。
因此,我们规定了以下编码原则:
- 如果我们获得一个实体以及该实体在某个事务内的相关实体,在该事务外部更改关系,然后尝试在新的事务内持久保存更改,那么最好重新获取父实体。
- 如果我们尝试删除一条父记录而不删除子记录,那么我们必须将所有子记录的外键字段设置为 NULL,然后再删除该父记录。
考虑 CaseWorker 和 Registration 之间的 OneToOne 关系。删除特定的登记时,我们并不删除办事员;因此,我们需要先将 reg_id 外键设置为空,然后才能删除任何登记。
@Stateless
public class RegManager {
.....
public void deleteReg(int regId){
Registration reg = em.find(Registration.class, regId);
CaseOfficer officer =reg.getCaseOfficer();
officer.setRegistration(null);
em.remove(reg);
}
}
数据完整性
一个用户查看某条登记记录时,另一个用户可能正在对同一应用程序进行更改。如果第一个用户随后对该申请进行了其他更改,他可能面临在不知情的情况下用旧数据覆盖该应用程序的风险。
为了解决此问题,我们决定使用“乐观锁定”。在 JPA 中,实体可以定义一个版本列,我们可以用该列实施乐观锁定。
public class Registration implements Serializable{
@Version
private int version;
.....
}
持久性提供程序会将版本列的内存中值与数据库中的该值进行匹配。如果两个值不同,持久性提供程序将报告异常。
验证
当我们说主申请人至少必须有一个地址且地址至少必须包含首行和邮政编码时,我们是对 Party 和 Address 实体应用业务规则。然而,如果我们说每个地址行必须始终少于 100 个字符时,该验证是 Address 实体固有的。
在我们的应用程序中,由于大多数工作流和面向流程的逻辑都在会话 Bean 层进行编码,因此我们决定实施到该层的跨对象/业务规则类型验证。然而,我们在实体内放置了固有验证。使用 JPA,我们可以将任何方法与实体的生命周期事件相关联。
以下实例验证了 Address 行包含的字符不能超过 100 个,并在持久保存 Address 实体之前调用该方法(通过 @PrePersist 批注)。出现故障时,该方法将向调用者抛出业务异常(扩展自 RuntimeException 类),然后可以使用该异常向用户传递一条消息。
public class Address implements Serializable{
.....
@PrePersist
public void validate()
if(addressLine1!=null && addressLine1.length()>1000){
throw new ValidationException("Address Line 1 is longer than 1000 chars.");
}
}
搜索
我们的税务登记应用程序提供了一个搜索工具,用来查找有关特定登记的详细信息、其当事人以及其他详细信息。提供一个有效的搜索工具涉及很多挑战,如编写有效的查询以及为了浏览大型结果列表而实施分页。JPA 指定了一个 Java 持久性查询语言 (JP-QL),与实体一同使用以实施数据访问。这是对 EJB 2.x EJB QL 的主要改进。我们成功地使用 JP-QL 提供了有效的数据访问机制。
查询
在 JPA 中,我们可以选择动态创建查询或定义静态查询。这些静态或命名查询支持参数;参数值在运行时指定。由于我们的查询范围定义得相当好,因此我们决定将命名查询与参数结合使用。命名查询也更为有效,因为持久性提供程序可以缓存转换的 SQL 查询,以供将来使用。
我们的应用程序为此提供了一个简单的使用案例:用户输入一个申请引用号以检索登记详细信息。我们在 Registration 实体上提供了一个命名查询,如下所示:
@Entity
@Table(name="REGISTRATION")
@NamedQuery(name="findByRegNumber", query = "SELECT r FROM REGISTRATION r WHERE r.appRefNumber=?1")
public class Registration implements Serializable{
.....
}
例如,我们应用程序内的一个搜索要求需要特别注意:用于检索所有当事人及其罚款总额的报告查询。由于该应用程序允许存在无处罚的当事人,因此简单的 JOIN 操作不会列出无处罚的当事人。 为解决此问题,我们使用了 JP-QL 的 OUTER JOIN 工具。我们还可以使用 GROUP BY 子句累积处罚。我们在 Party 实体中添加了另一个命名查询,如下所示:
@Entity
@Table(name="PARTY_DATA")
@Inheritance(strategy= InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name="PARTY_TYPE")
@NamedQueries({@NamedQuery(name="generateReport",
query=" SELECT NEW com.ssg.article.ReportDTO(p.name, SUM(pen.amount))
FROM Party p LEFT JOIN p.penalties pen GROUP BY p.name""),
@NamedQuery(name="bulkInactive",
query="UPDATE PARTY P SET p.status=0 where p.registrationID=?1")})
public abstract class Party {
.....
}
注意,在上面的命名查询“generateReport”示例中,我们实例化了该查询本身内的一个新 ReportDTO 对象。这仍然是 JPA 的一个十分强大的功能。
我们可以批量操作吗?
在我们的应用程序中,官员可以检索登记并使其处于非活动状态。在这种情况下,我们还应该将所有与该 Registration 相关联的 Party 都设置为非活动状态。这通常意味着将 PARTY 表中的 Status 列设置为 0。为了提高性能,我们将使用批量更新,而不是针对每个 Party 执行单独的 SQL。
幸运的是,JPA 提供了进行此操作的方法:
@NamedQuery(name="bulkInactive", query="UPDATE PARTY p SET p.status=0 where p.registrationID=?1")
public abstract class Party implements Serializable{
.....
}
注意:批量操作直接向数据库发出 SQL,这意味着并不更新持就性上下文以反映更改。使用超出单个事务范围的扩展的持久性上下文时,缓存的实体可能包含陈的数据。
及早获取。
另一个挑战性的要求是选择性数据显示。例如,如果管理员搜索登记,我们需要显示登记方记录的所有处罚。然而,该信息并不提供给普通办事员。对于某些登记,我们需要显示登记方记录的所有处罚。然而,该信息并不提供给普通办事员。
Party 和 Penalty 之间的关系是 OneToMany。前面提到过,此关系的默认 FETCH 类型为 LAZY。但为了满足这个搜索选择性显示要求,将 Penalty 详细信息作为单个 SQL 获取以避免多个 SQL 调用是很有意义的。
JP-QL 中的 FETCH Join 特性帮我们解决了这个问题。如果我们希望暂时覆盖 LAZY 获取类型,可以使用 Fetch Join。然而,如果频繁使用该特性,考虑将 FETCH 类型重新调整为 EAGER 是很明智的。
@NamedQueries({@NamedQuery(name="generateReport",
query=" SELECT NEW com.ssg.article.ReportDTO(p.name, SUM(pen.amount))
FROM Party p LEFT JOIN p.penalties pen GROUP BY p.name""),
@NamedQuery(name="bulkInactive",
query="UPDATE PARTY P SET p.status=0 where p.registrationID=?1"),
@NamedQuery(name="getItEarly", query="SELECT p FROM Party p JOIN FETCH p.penalties")})
public abstract class Party {
.....
}
结论
总的说来,JPA 简化了持久性编码。我们发现它功能齐备且十分有效。它丰富的查询界面和极大改进的查询语言简化了复杂关系情况的处理。它的继承支持帮助我们在持久性级别保持逻辑域模型,我们可以跨层重新用相同的实体。JPA 的所有优点使其成为大家今后明确的选择
转自:http://hi.baidu.com/walkerstar/blog/item/2ee2a02415011237c8955998.html
相关推荐
《JAVA中的微服务体系结构》是一本深入探讨Java微服务架构的书籍源码,它提供了丰富的实践案例和详尽的代码实现,旨在帮助开发者理解和构建自己的微服务应用。在这个压缩包中,你将找到作者精心编写的...
11. **Chapter 14**:可能是一个综合性的案例研究或最佳实践,结合前面章节的知识点,展示一个完整项目从开始到结束的开发过程。 通过对这些章节源代码的学习和实践,读者不仅能够掌握NetBeans IDE的使用,还能深入...
Delphi 12.3控件之TraeSetup-stable-1.0.12120.exe
基于GPRS,GPS的电动汽车远程监控系统的设计与实现.pdf
内容概要:本文详细介绍了如何利用MATLAB/Simulink 2018a进行单机无穷大系统的暂态稳定性仿真。主要内容包括搭建同步发电机模型、设置无穷大系统等效电源、配置故障模块及其控制信号、优化求解器设置以及绘制和分析转速波形和摇摆曲线。文中还提供了多个实用脚本,如故障类型切换、摇摆曲线计算和极限切除角的求解方法。此外,作者分享了一些实践经验,如避免常见错误和提高仿真效率的小技巧。 适合人群:从事电力系统研究和仿真的工程师和技术人员,尤其是对MATLAB/Simulink有一定基础的用户。 使用场景及目标:适用于需要进行电力系统暂态稳定性分析的研究项目或工程应用。主要目标是帮助用户掌握单机无穷大系统的建模和仿真方法,理解故障对系统稳定性的影响,并能够通过仿真结果评估系统的性能。 其他说明:文中提到的一些具体操作和脚本代码对于初学者来说可能会有一定的难度,建议结合官方文档或其他教程一起学习。同时,部分技巧和经验来自于作者的实际操作,具有一定的实用性。
KUKA机器人相关资料
基于DLR模型的PM10–能见度–湿度相关性 研究.pdf
内容概要:本文详细介绍了如何使用MATLAB/Simulink进行光伏并网系统的最大功率点跟踪(MPPT)仿真,重点讨论了电导增量法的应用。首先阐述了电导增量法的基本原理,接着展示了如何在Simulink中构建光伏电池模型和MPPT控制系统,包括Boost升压电路的设计和PI控制参数的设定。随后,通过仿真分析了不同光照强度和温度条件对光伏系统性能的影响,验证了电导增量法的有效性,并提出了针对特定工况的优化措施。 适合人群:从事光伏系统研究和技术开发的专业人士,尤其是那些希望通过仿真工具深入理解MPPT控制机制的人群。 使用场景及目标:适用于需要评估和优化光伏并网系统性能的研发项目,旨在提高系统在各种环境条件下的最大功率点跟踪效率。 其他说明:文中提供了详细的代码片段和仿真结果图表,帮助读者更好地理解和复现实验过程。此外,还提到了一些常见的仿真陷阱及解决方案,如变步长求解器的问题和PI参数整定技巧。
KUKA机器人相关文档
内容概要:本文详细探讨了双馈风力发电机(DFIG)在Simulink环境下的建模方法及其在不同风速条件下的电流与电压波形特征。首先介绍了DFIG的基本原理,即定子直接接入电网,转子通过双向变流器连接电网的特点。接着阐述了Simulink模型的具体搭建步骤,包括风力机模型、传动系统模型、DFIG本体模型和变流器模型的建立。文中强调了变流器控制算法的重要性,特别是在应对风速变化时,通过实时调整转子侧的电压和电流,确保电流和电压波形的良好特性。此外,文章还讨论了模型中的关键技术和挑战,如转子电流环控制策略、低电压穿越性能、直流母线电压脉动等问题,并提供了具体的解决方案和技术细节。最终,通过对故障工况的仿真测试,验证了所建模型的有效性和优越性。 适用人群:从事风力发电研究的技术人员、高校相关专业师生、对电力电子控制系统感兴趣的工程技术人员。 使用场景及目标:适用于希望深入了解DFIG工作原理、掌握Simulink建模技能的研究人员;旨在帮助读者理解DFIG在不同风速条件下的动态响应机制,为优化风力发电系统的控制策略提供理论依据和技术支持。 其他说明:文章不仅提供了详细的理论解释,还附有大量Matlab/Simulink代码片段,便于读者进行实践操作。同时,针对一些常见问题给出了实用的调试技巧,有助于提高仿真的准确性和可靠性。
linux之用户管理教程.md
内容概要:本文详细介绍了利用三菱PLC(特别是FX系列)和组态王软件构建3x3书架式堆垛式立体库的方法。首先阐述了IO分配的原则,明确了输入输出信号的功能,如仓位检测、堆垛机运动控制等。接着深入解析了梯形图编程的具体实现,包括基本的左右移动控制、复杂的自动寻址逻辑,以及确保安全性的限位保护措施。还展示了接线图和原理图的作用,强调了正确的电气连接方式。最后讲解了组态王的画面设计技巧,通过图形化界面实现对立体库的操作和监控。 适用人群:从事自动化仓储系统设计、安装、调试的技术人员,尤其是熟悉三菱PLC和组态王的工程师。 使用场景及目标:适用于需要提高仓库空间利用率的小型仓储环境,旨在帮助技术人员掌握从硬件选型、电路设计到软件编程的全流程技能,最终实现高效稳定的自动化仓储管理。 其他说明:文中提供了多个实用的编程技巧和注意事项,如避免常见错误、优化性能参数等,有助于减少实际应用中的故障率并提升系统的可靠性。
基于STM32的循迹避障小车 主控:STM32 显示:OLED 电源模块 舵机云台 超声波测距 红外循迹模块(3个,左中右) 蓝牙模块 按键(6个,模式和手动控制小车状态) TB6612驱动的双电机 功能: 该小车共有3种模式: 自动模式:根据红外循迹和超声波测距模块决定小车的状态 手动模式:根据按键的状态来决定小车的状态 蓝牙模式:根据蓝牙指令来决定小车的状态 自动模式: 自动模式下,检测距离低于5cm小车后退 未检测到任何黑线,小车停止 检测到左边或左边+中间黑线,小车左转 检测到右边或右边+中间黑线,小车右转 检测到中边或左边+中间+右边黑线,小车前进 手动模式:根据按键的状态来决定小车的状态 蓝牙模式: //需切换为蓝牙模式才能指令控制 *StatusX X取值为0-4 0:小车停止 1:小车前进 2:小车后退 3:小车左转 4:小车右转
矢量边界,行政区域边界,精确到乡镇街道,可直接导入arcgis使用
内容概要:本文探讨了基于IEEE33节点的主动配电网优化方法,旨在通过合理的调度模型降低配电网的总运行成本。文中详细介绍了模型的构建,包括风光发电、储能装置、柴油发电机和燃气轮机等多种分布式电源的集成。为了实现这一目标,作者提出了具体的约束条件,如储能充放电功率限制和潮流约束,并采用了粒子群算法进行求解。通过一系列实验验证,最终得到了优化的分布式电源运行计划,显著降低了总成本并提高了系统的稳定性。 适合人群:从事电力系统优化、智能电网研究的专业人士和技术爱好者。 使用场景及目标:适用于需要优化配电网运行成本的研究机构和企业。主要目标是在满足各种约束条件下,通过合理的调度策略使配电网更加经济高效地运行。 其他说明:文章不仅提供了详细的理论推导和算法实现,还分享了许多实用的经验技巧,如储能充放电策略、粒子群算法参数选择等。此外,通过具体案例展示了不同电源之间的协同作用及其经济效益。
KUKA机器人相关文档
内容概要:本文详细介绍了将光热电站(CSP)和有机朗肯循环(ORC)集成到综合能源系统中的优化建模方法。主要内容涵盖系统的目标函数设计、关键设备的约束条件(如CSP储热罐、ORC热电耦合)、以及具体实现的技术细节。文中通过MATLAB和YALMIP工具进行建模,采用CPLEX求解器解决混合整数规划问题,确保系统在经济性和环境效益方面的最优表现。此外,文章还讨论了碳排放惩罚机制、风光弃能处理等实际应用场景中的挑战及其解决方案。 适合人群:从事综合能源系统研究的专业人士,尤其是对光热发电、余热利用感兴趣的科研工作者和技术开发者。 使用场景及目标:适用于需要评估和优化包含多种能源形式(如光伏、风电、燃气锅炉等)在内的复杂能源系统的项目。目标是在满足供电供热需求的同时,最小化运行成本并减少碳排放。 其他说明:文中提供了大量具体的MATLAB代码片段作为实例,帮助读者更好地理解和复现所提出的优化模型。对于初学者而言,建议从简单的确定性模型入手,逐渐过渡到更复杂的随机规划和鲁棒优化。
网站设计与管理作业一.ppt
内容概要:本文详细介绍了如何使用MATLAB搭建双闭环Buck电路的仿真模型。首先定义了主电路的关键参数,如输入电压、电感、电容等,并解释了这些参数的选择依据。接着分别对电压外环和电流内环进行了PI控制器的设计,强调了电流环响应速度需要显著高于电压环以确保系统的稳定性。文中还讨论了仿真过程中的一些关键技术细节,如PWM死区时间的设置、低通滤波器的应用以及参数调整的方法。通过对比单闭环和双闭环系统的性能,展示了双闭环方案在应对负载突变时的优势。最后分享了一些调试经验和常见问题的解决方案。 适合人群:从事电力电子、电源设计领域的工程师和技术人员,尤其是有一定MATLAB基础的读者。 使用场景及目标:适用于需要进行电源管理芯片设计验证、电源系统性能评估的研究人员和工程师。主要目标是提高电源系统的稳定性和响应速度,特别是在负载变化剧烈的情况下。 其他说明:文章不仅提供了详细的理论分析,还包括了大量的代码片段和具体的调试步骤,帮助读者更好地理解和应用所学知识。同时提醒读者注意仿真与实际情况之间的差异,鼓励在实践中不断探索和改进。
内容概要:本文详细探讨了MATLAB环境下冷热电气多能互补微能源网的鲁棒优化调度模型。首先介绍了多能耦合元件(如风电、光伏、P2G、燃气轮机等)的运行特性模型,展示了如何通过MATLAB代码模拟这些元件的实际运行情况。接着阐述了电、热、冷、气四者的稳态能流模型及其相互关系,特别是热电联产过程中能流的转换和流动。然后重点讨论了考虑经济成本和碳排放最优的优化调度模型,利用MATLAB优化工具箱求解多目标优化问题,确保各能源设备在合理范围内运行并保持能流平衡。最后分享了一些实际应用中的经验和技巧,如处理风光出力预测误差、非线性约束、多能流耦合等。 适合人群:从事能源系统研究、优化调度、MATLAB编程的专业人士和技术爱好者。 使用场景及目标:适用于希望深入了解综合能源系统优化调度的研究人员和工程师。目标是掌握如何在MATLAB中构建和求解复杂的多能互补优化调度模型,提高能源利用效率,降低碳排放。 其他说明:文中提供了大量MATLAB代码片段,帮助读者更好地理解和实践所介绍的内容。此外,还提及了一些有趣的发现和挑战,如多能流耦合的复杂性、鲁棒优化的应用等。