Spring <idref>与<ref>的区别

本文详细解释了Spring框架中<idref>和<ref>的区别,包括它们对应的属性类型和使用场景。通过示例代码演示了如何在配置文件中正确使用这两个元素。

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

Spring中,<idref>和<ref>还是比较容易混淆的。。 他们的本质区别是<idref>对应的是id的值,对应该值的property一般来说是String类型, 而<idref>对应的是bean的值, 对应该值的property是bean的类型。 这样说还是比较模糊的,看下面的例子。


package Impl;

public class Teacher {

public Teacher(){

}

public Teacher(String name,Student student){

}

}


package Impl;

public class Student {
private Teacher teacher; //Teacher类型
// private IPhone phone;
private String teacherId; //String类型

public Student(){

}

public Student(Teacher teacher){
this.teacher = teacher;
}

public Teacher getTeacher() {
return teacher;
}

public void setTeacher(Teacher teacher) {
this.teacher = teacher;
}

public String getTeacherId() {
return teacherId;
}

public void setTeacherId(String teacherId) {
this.teacherId = teacherId;
}

}



<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p = "http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd"
>

<bean id="student" class="Impl.Student" >
<property name="teacher">
<ref bean="teacher"/>
</property>
<property name="teacherId">
<idref bean="teacher"/>
</property>
</bean>


<bean id="teacher" class="Impl.Teacher">
<constructor-arg index="0" type="java.lang.String">
<value>Morgan</value>
</constructor-arg>

<constructor-arg index="1" ref="student">
</constructor-arg>
</bean>

</beans>



package test;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import Impl.Student;

public class TestConstructorBeans {

public static void main(String args[]){
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
// context.getBean("teacher");

ApplicationContext context2 = new ClassPathXmlApplicationContext(new String[]{"constructorBeans.xml"},context);
//context.getBean("phone2");
Student s = (Student)context2.getBean("student");
System.out.println(s.getTeacherId());
}

}



需要指出的是,<idref bean="teacher"/> 可以用<value>teacher</value>替代,但是Spring是不会检出value是否存在的。。而是用idref, sping在加载的时候会检出idref的值是否存在。 因为拿到另一个bean的id, 基本上都是想在运行时通过getBean得到bean的实例。
<think>好的,我现在需要帮用户设计一个包含特定要求的DTD文档,针对学生信息的XML文件。用户的要求包括内部一般实体、外部参数实体以及多种属性类型声明。首先,我得回忆一下DTD的基本结构相关概念。 首先,内部一般实体是在DTD中定义的,用于替换文本,比如常见的实体用来表示常用字符串或特殊符号。用户需要这个,所以得在DTD里用<!ENTITY>声明,然后在XML中使用&实体名;引用。例如,定义学校名称作为内部实体,方便多处使用。 然后是外部参数实体,这需要创建一个外部DTD文件,然后在主DTD中通过<!ENTITY % 实体名 SYSTEM "URI">来引用。参数实体通常用于模块化DTD,比如把公用部分放在外部文件中,方便复用。用户可能希望将课程信息作为外部参数实体引入,这样可以保持主DTD的简洁。 接下来是属性声明,用户提到了CDATA、枚举、ID、IDREF、ENTITY类型。需要为元素定义这些属性。比如,学生的ID属性设为ID类型,确保唯一性;性别用枚举限制为男或女;成绩用CDATA;可能还需要ENTITY类型引用图片等外部资源。 然后要考虑XML文档的结构。学生信息通常包含学生列表,每个学生有学号、姓名、性别、年龄、课程、成绩等。课程可能需要嵌套结构,比如课程名、学分、教师。此外,可能需要关联成绩课程,或者学生之间的引用,比如导师用IDREF指向另一个学生的ID。 需要确保ID类型的唯一性,IDREF的有效性引用。枚举类型属性要正确限制可选值。外部参数实体的引用要路径正确,并且参数实体在外部DTD中定义,比如课程相关的元素属性。 还需要注意内部一般实体参数实体的声明方式不同,参数实体以%开头,并且在DTD中使用%实体名;来引用。而一般实体直接在XML内容中使用。 可能会遇到的问题包括实体引用是否正确,属性类型是否匹配,外部文件路径是否正确。需要测试DTD的有效性,确保XML文档验证通过。 现在需要把这些部分整合成一个完整的DTD示例,同时给出对应的XML示例,并解释每个部分的作用,满足用户的需求。还要注意引用之前提供的资料,比如实体声明必须在DOCTYPE中,外部实体分SYSTEMPUBLIC,这里使用SYSTEM引用本地文件。 最后,检查是否涵盖了所有要求:内部一般实体、外部参数实体、各种属性类型。确保结构清晰,语法正确,并且有详细的注释说明每个部分的作用。</think>### DTD设计步骤示例(含多种实体属性声明) --- #### 一、XML文档示例(目标结构) ```xml <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE 学生列表 SYSTEM "student.dtd"> <学生列表> <学生 学号="S001" 性别="男" 成绩等级="A"> <姓名>&学校名称;实验中学</姓名> <年龄>18</年龄> <课程列表> %课程模块; </课程列表> <导师 导师编号="T001"/> <照片 来源="student_photo"/> </学生> </学生列表> ``` --- #### 二、DTD文件(student.dtd)完整代码 ```dtd <!-- 1. 内部一般实体定义 --> <!ENTITY 学校名称 "北京市"> <!-- 2. 外部参数实体引用 --> <!ENTITY % 课程模块 SYSTEM "course_module.dtd"> %课程模块; <!-- 3. 根元素及属性声明 --> <!ELEMENT 学生列表 (学生+)> <!ELEMENT 学生 (姓名, 年龄, 课程列表, 导师, 照片)> <!-- 属性类型声明 --> <!ATTLIST 学生 学号 ID #REQUIRED 性别 (男|女) #REQUIRED 成绩等级 CDATA #IMPLIED > <!-- 关联外部ENTITY类型 --> <!ENTITY student_photo SYSTEM "photo.jpg"> <!ATTLIST 照片 来源 ENTITY #REQUIRED > <!-- IDREF类型引用 --> <!ATTLIST 导师 导师编号 IDREF #REQUIRED > ``` --- #### 三、外部参数实体文件(course_module.dtd) ```dtd <!ELEMENT 课程列表 (课程+)> <!ELEMENT 课程 (名称, 学分, 教师)> <!ATTLIST 课程 类型 (必修|选修) "必修" > <!ELEMENT 名称 (#PCDATA)> <!ELEMENT 学分 (#PCDATA)> <!ELEMENT 教师 (#PCDATA)> ``` --- ### 关键语法解析 1. **内部一般实体** - 定义:`<!ENTITY 学校名称 "北京市">` - 使用:`<姓名>&学校名称;实验中学</姓名>` - 作用:实现文本复用(如统一学校前缀)[^2] 2. **外部参数实体** - 定义:`<!ENTITY % 课程模块 SYSTEM "course_module.dtd">` - 引用:`%课程模块;` - 作用:模块化DTD设计(如分离课程相关声明)[^3] 3. **属性类型声明** - **ID类型**:`学号 ID` → 唯一标识学生 - **枚举类型**:`性别 (男|女)` → 限制可选值 - **CDATA类型**:`成绩等级 CDATA` → 存储任意文本 - **IDREF类型**:`导师编号 IDREF` → 引用其他元素的ID - **ENTITY类型**:`来源 ENTITY` → 关联外部资源(如图片)[^1] --- ### 验证XML有效性 1. 确保所有ID值唯一(如`学号="S001"`不可重复) 2. IDREF必须匹配已存在的ID(如`导师编号="T001"`需对应实际ID) 3. 枚举属性值必须从预设列表选择(如`性别`只能是男/女) --- ### 扩展应用场景 - **数据关联**:通过ID/IDREF实现学生导师的关系绑定 - **资源管理**:ENTITY类型可用于集中管理图片或文件路径 - **版本控制**:外部参数实体便于统一更新多文档共享的模块[^3] ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值