本章简介
通过之前学习的三层架构,我们已经可以将表示层、业务逻辑层和数据访问层相互分离,从而实现程序的解耦合,以及提高系统的可维护性和可扩展性等。但在之前的代码中,表示层JSP的代码里仍然嵌套了很多JAVA代码,从而造成了表示层一定程度的混乱。
能否彻底的从表示层中消除JAVA代码呢?可以使用 EL表达式和JSTL标签库。EL表达式和JSTL标签库不仅能替换JSP中的JAVA代码,而且还能使代码更加简洁。
本章演示的项目名为ELAndJSTLDemo,因为本章只是在讲三层中的表示层,因此为了使读者清晰的掌握EL及JSTL,本章项目并没有采用标准的三层架构,而是以一个个简单的JSP示例来诠释。
EL的全称是Expression Language,可以用来替代JSP页面中的JAVA代码,从而能让即使不懂JAVA的开发人员也能写出JSP代码。EL还可以实现自动转换类型,使用起来非常简单。
语法:
${EL表达式}
EL表达式通常由两部分组成:对象和属性。可以使用“点操作符”或“中括号[]操作符”来操作对象的属性。
讲解之前,我们先通过一个简单例子来回忆一下之前不用EL的使用情景。先创建两个封装数据的JavaBean,如下,
地址信息类 Address.java
package org.lanqiao.entity;public class Address{ //家庭地址 private String homeAddress ; //学校地址 private String schoolAddress ; //省略setter、getter}学生信息类 Student.java
package org.lanqiao.entity;public class Student{ private int studentNo ; private String studentName; //地址属性 private Address address ; //省略setter、getter}再创建一个Servlet用于初始化一些数据,并将此Servlet设置为项目的默认访问程序:
InitServlet.java
package org.lanqiao.servlet;//省略importpublic class InitServlet extends HttpServlet{protected void doGet(…)… { this.doPost(request, response); } protected void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException { Address address = new Address(); address.setHomeAddress("北京朝阳区"); address.setSchoolAddress("北京大兴区大族企业广域网#6F2"); Student student = new Student(); student.setStudentNo(27); student.setStudentName("颜群"); student.setAddress(address); request.setAttribute("student", student); request.getRequestDispatcher("index.jsp").forward(request, response); }}web.xml
… <welcome-file-list> <welcome-file>InitServlet</welcome-file> </welcome-file-list>…如上,程序会在InitServlet中给student对象的各个属性赋值,然后请求转发到index.jsp中。我们先用传统的Scriptlet接收对象,并将对象的属性显示到前台,如下,
index.jsp
…<body> <% Student student = (Student)request.getAttribute("student"); int studentNo = student.getStudentNo(); String studentName = student.getStudentName(); Address address = student.getAddress(); String homeAddress = address.getHomeAddress(); String schoolAddress = address.getSchoolAddress(); out.print("学号:"+studentNo +"<br/>"); out.print("姓名:"+studentName +"<br/>"); out.print("家庭地址:"+homeAddress +"<br/>"); out.print("学校地址:"+schoolAddress +"<br/>"); %></body>…部署并执行此项目http://localhost:8888/ELAndJSTLDemo/,运行结果如图,

图9-01
可以发现,程序的确能够正常的显示。但如果将index.jsp中的代码用EL表达式来实现,就会简单许多。如下是使用EL修改后的index.jsp,功能与之前的Scriptlet代码相同,
…<body> <%-- 使用EL表达式 --%> 学生对象:${requestScope.student}<br/> 学号:${requestScope.student.studentNo } <br/> 姓名:${requestScope.student.studentName } <br/> 家庭地址:${requestScope.student.address.homeAddress } <br/> 学校地址:${requestScope.student.address.schoolAddress } <br/></body>…运行http://localhost:8080/ELAndJSTLDemo/,结果如图,

图9-02
综上,使用EL可以将JSP中的JAVA代码彻底消除,并且不用再做强制的类型转换,整体的JSP代码就会简单很多。
${requestScope.student},表示在request作用域内查找student对象;${requestScope.student.studentNo },表示在request作用域内查找student对象的studentNo属性。点操作符“.”的用法和在JAVA中的用法相同,都是直接用来调用对象的属性。此外,通过${requestScope.student.address.homeAddress }可以发现,EL表达式能够级联获取对象的属性,即先在request作用域内找到student对象后,可以再次使用点操作符“.”来获取student内部的address对象……
除了点操作符以外,还可以使用中括号操作符来访问某个对象的属性,例如${requestScope.student.studentNo }可以等价写成${requestScope.student["studentNo"] }或${requestScope["student"]["studentNo"] }。除此之外,中括号[]操作符还有一些其他独有功能:
<1>如果属性名称中包含一些特殊字符,如“.”、“?”、“-”等,就必须使用中括号操作符,而不能用点操作符。例如,如果在之前的InitServlet中写了request.setAttribute("school-name", "LanQiao");那么在index.jsp中就不能用${requestScope. school-name },而必须改为${requestScope ["school-name "] }。
<2>如果要动态取值时,也必须使用中括号操作符,而不能用点操作符。例如,String data=”school”(即data是一个变量),那么就只能用${ requestScope [data]}。需要注意中括号里面的值:如果加了双引号则表示一个常量,如${requestScope ["school-name "] },表示获取"school-name "的属性值;而如果不加上双引号,则表示一个变量,如${ requestScope [data]},表示获取data所表示的”school”的属性值,即等价于${ requestScope [”school”]}。所以在使用中括号获取属性值时,一定要注意是否加引号。此外,中括号中的值,除了双引号以外,也可以使用单引号,作用是一样的。
<3>访问数组。如果要访问request作用域内的一个对象名为names的数组,就可以通过中括号来表示索引,如${requestScope .array[0]}、${ requestScope .array[1]}等。
点操作符和中括号操作符还可以用来获取Map中的属性值,如下,
InitServlet.java
package org.lanqiao.servlet;//省略importpublic class InitServlet extends HttpServlet{… protected void doPost(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException { … Map<String,String> countries = new HashMap<String,String>(); countries.put("cn", "中国"); countries.put("us", "美国"); request.setAttribute("countries", countries); request.getRequestDispatcher("index.jsp").forward(request, response); }}index.jsp
<body> … ------------------Map------------------<br/> cn:${requestScope.countries.cn }<br/> us:${requestScope.countries["us"] }<br/></body>打开浏览器执行http://localhost:8080/ELAndJSTLDemo/,运行结果如图,

图9-03
EL表达式还能够进行一些简单的运算。
| 关系运算符 | 示例 | 结果 | |
| 大于 | >(或gt) | ${2>1}或${2 gt 1} | true |
| 大于或等于 | >=(或ge) | ${2>=1}或${2 ge 1} | true |
| 等于 | ==(或eq) | ${2==1}或${2 eq 1} | false |
| 小于或等于 | <=(或le) | ${2<=1}或${2 le 1} | false |
| 小于 | <(或lt) | ${2<1}或${2 lt 1} | false |
| 不等于 | !=(或ne) | ${2!=1}或${2 ne 1} | true |
| 关系运算符 | 示例 | 结果 | |
| 逻辑或 | ||(或or) | true||false(或 true or false) | true |
| 逻辑与 | &&(或and) | true&&false(或 true and false) | false |
| 逻辑非 | !(或not) | !true (或 not true) | false |
Empty操作符用来判断一个值是否为null或不存在。我们在InitServlet中给request的作用域内增加两个变量,如下,
InitServlet.java
package org.lanqiao.servlet;//省略importpublic class InitServlet extends HttpServlet{ … protected void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException { … request.setAttribute("test","test"); request.setAttribute("nullVar",null); request.getRequestDispatcher("index.jsp").forward(request, response); }}在request作用域内增加了“test”和“nullVar”两个变量,并且“nullVar”的值是null。然后再在index.jsp中获取,如下,
index.jsp
<body> … <%-- Empty操作符 --%> 之前不存在temp变量:${empty temp }<br/> 变量nullVar的值为null:${empty nullVar }<br/> 变量test已被赋值:${empty test }<br/></body>运行结果:

图9-04
可以发现,之前不存在的变量“temp”和值为null的变量“nullVar”,用empty操作符运算出的结果为true,而之前存在值的变量“test”用empty运算的结果为false。
“隐式对象”又称“内置对象”。我们之前在JSP里曾提到过,像request、session、application等都是JSP的隐式对象,这些“隐式对象”可以不用实例化就直接使用。同样的,在EL表达式中也存在一些隐式对象。按照使用的途径不同,EL隐式对象分为了作用域访问对象、参数访问对象和JSP隐式对象,如图,

图9-05
${requestScope.student.studentNo }中的requestScope表示request作用域。即在使用EL表达式来获取一个变量的同时,可以指定该变量的作用域。EL表达式为我们提供了四个可选的作用域对象,如下表,
| 对象名 | 作用域 |
| pageScope | 把pageContext作用域中的数据映射为一个Map类的对象 |
| requestScope | 把request作用域中的数据映射为一个Map类的对象 |
| sessionScope | 把session作用域中的数据映射为一个Map类的对象 |
| applicationScope | 把application作用域中的数据映射为一个Map类的对象 |
pageScope, requestScope, sessionScope和appliationScope都可以看作是Map型变量,要获取其中的数据就可以使用点操作符或中括号操作符。如${requestScope.student}可以获取request作用域中的student属性值,${sessionScope[“user”]}可以获取session作用域中的user属性值。另外,如果不指定作用域,EL表达式就会依次从pageContextrequestsessionapplication的范围内寻找。例如,EL表达式在解析${student}时,就会先在pageContext作用域中查找是否有student属性,如果有则直接获取,如果没有则再会去request作用域中查找是否有student属性……
在JSP中,可以使用request.getParameter()和request.getParameterValues()来获取表单中的值(或地址栏、超链接中附带的值)。对应的,EL表达式可以通过param、paramValues来获取这些值。
| 对象名 | 示例 | 作用 |
| param | ${param.usename} | 等价于request.getParameter("username") |
| paramValues | ${param.hobbies} | 等价于request. getParameterValues ("hobbies") |
pageContext是JSP的一个隐式对象,同时也是EL表达式的隐式对象。因此,pageContext是EL表达式与JSP之间的一个桥梁。
在EL表达式中,可以通过pageContext来获取JSP的内置对象和ServletContext对象,如下表:
| El表达式 | 获取的对象 |
| ${pageContext.page} | 获取page对象 |
| ${pageContext.request} | 获取request对象 |
| ${pageContext.response} | 获取response对象 |
| ${pageContext.session} | 获取session对象 |
| ${pageContext.out} | 获取out对象 |
| ${pageContext.exception} | 获取exception对象 |
| ${pageContext.servletContext} | 获取servletContext对象 |
还可以获取这些对象的getXxx()方法:例如,${pageContext.request.serverPort}就表示访问request对象的getServerPort()方法。可以发现,在使用时EL去掉了方法中的get和“()”,并将首字母改为了小写。
小结:
语法: ${EL表达式}
其中${ }中的“EL表达式”可以是四个作用域访问对象、参数访问对象或JSP隐式对象pageContext三者之一。并且EL表达式可以像HTML一样直接写在JSP页面中,而不用被“<%… %>”括起来。
JSTL(JSP Standard Tag Library,JSP标准标签库),是一个不断完善的开源JSP标签库,包含了开发JSP时经常用到的一组标准标签。使用JSTL,可以像EL表达式那样不用编写JAVA代码就能开发出复杂的JSP页面。JSTL一般要结合EL表达式一起使用。
使用JSTL标签库以前,必须先在Web项目的lib目录中加入两个jar包:jstl.jar和standard.jar,然后再在需要使用JSTL的JSP页面,加入支持JSTL的taglib指令,如下,
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
其中prefix=”c”表示在当前页面中,JSTL标签库是通过标签使用的。
JSTL核心标签库主要包含三类:通用标签库、条件标签库和迭代标签库,如图,

图9-06
通用标签库包含了3个标签:赋值标签<c:set>、显示标签<c:out>、删除标签<c:remove>
①赋值标签<c:set>
<c:set>标签的作用,是给变量在某个作用域内赋值,有两个版本:“var”版和“target”版。
<1>var版
用于给page、request、session或application作用域内的变量赋值
语法:
<c:set var="elementVar" value=" elementValue" scope="scope" />
var:需要赋值的变量名
value:被赋予的变量值
scope:此变量的作用域,有4个可填项,即page,request,session和application。
示例:
<c:set var="addError" value="error" scope="request"/>
表示在request作用域内,设置一个addError变量,并将变量值赋值为“error”,等价于
request.setAttribute("addError", "error");
<2> target版
用于给JavaBean对象的属性或Map对象赋值。
a.给JavaBean对象的属性赋值
语法:
<c:set target="objectName" property="propertyName"value="propertyValue" scope="scope"/>target:需要操作的JavaBean对象,通常使用EL表达式来表示。
property:对象的属性名。
value:对象的属性值。
scope:此属性值的作用域,有4个可填项,即page,request,session和application。
示例: 先通过Servlet给JavaBean对象的属性赋值,
InitJSTLDataServlet.java
package org.lanqiao.servlet;//省略importpublic class InitJSTLDataServlet extends HttpServlet{ … protected void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {//将一个Address对象赋值后,加入request作用域,并请求转发到Jsp Address address = new Address(); address.setHomeAddress("北京朝阳区"); address.setSchoolAddress("北京大兴区大族企业广域网#6F2"); request.setAttribute("address", address); request.getRequestDispatcher("JSTLDemo.jsp").forward(request, response); }}JSTLDemo.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>…<body> 使用JSTL赋值之前:${requestScope.address.schoolAddress } <br/> <c:set target="${requestScope.address }"property="schoolAddress" value="广东东莞蓝桥基地" /> <br/> 使用JSTL赋值之后:${requestScope.address.schoolAddress }</body>运行InitJSTLDataServlet,结果如图,

图9-07
b.给Map对象赋值
语法:
<c:set target="mapName" property="mapKey"value="mapValue" scope="scope"/>target:需要操作的Map对象,通常使用EL表达式来表示。
property:表示Map对象的key。
value:表示Map对象的value。
scope:此Map对象的作用域,有4个可填项,即page,request,session和application。
示例: 先通过Servlet给Map对象的属性赋值,如下
InitJSTLDataServlet.java
package org.lanqiao.servlet;//省略importpublic class InitJSTLDataServlet extends HttpServlet{… protected void doPost(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException { … //将一个Map对象赋值后,加入request作用域,并请求转发到JSTLDemo.jsp Map<String,String> countries = new HashMap<String,String>(); countries.put("cn", "中国"); countries.put("us", "美国"); request.setAttribute("countries", countries); request.getRequestDispatcher("JSTLDemo.jsp").forward(request, response); }}再使用<c:set…/>对Map对象的属性赋值,如下,
JSTLDemo.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>…<body> 使用JSTL赋值之前:${requestScope.countries.cn }、${requestScope.countries.us } <br/> <c:set target="${requestScope.countries }" property="cn"value="中国人民共和国" /> <br/> 使用JSTL赋值之后:${requestScope.countries.cn }、${requestScope.countries.us }<br/> <br/></body>运行InitJSTLDataServlet,结果如图,

图9-08
需要注意的是,<c:set>标签不仅能对已有变量赋值;如果需要赋值的变量并不存在, <c:set>也会自动产生该对象,如下,
index.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %><body> 在request作用域内,并不存在temp变量:${requestScope.temp } <br/> 使用c:set直接给temp赋值为LanQiao <c:set var="temp" value="LanQiao" scope="request"/> <br/> 再次观察temp变量:${requestScope.temp } <br/></body>运行结果:

图9-09
②输出标签<c:out>
输出标签<c:out>类似于JSP中的<%= %>,但功能更加强大。
语法:
<c:out value="value" default="defaultValue" escapeXml="isEscape"/>
value:输出显示结果,可以使用EL表达式。
default:可选项。当value表示的对象不存在或为空时,默认的输出值。
escapeXml:可选项,值为true或false。为true时(默认情况),将value中的值以字符串的形式原封不动的显示出来;为false时,会将内容以HTML渲染后的结果显示。
示例:
先在InitJSTLDataServlet中创建address对象,然后给address中的schoolAddress属性赋值,再把address对象放入request作用域,之后请求转发到index.jsp,如下,
InitJSTLDataServlet.java
…protected void doPost(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException { Address address = new Address(); address.setSchoolAddress("北京大兴区大族企业广域网#6F2"); request.setAttribute("address", address); request.getRequestDispatcher("JSTLDemo.jsp").forward(request, response); }…index.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>…<body> request作用域中,存放了address对象及schoolAddress属性值:<c:out value="${requestScope.address.schoolAddress}" /><br/> request作用域中,不存在student对象:<c:out value="${requestScope.student}"default="student对象为空"/> <br/> 当escapeXml="true"时:<c:out value="<a href=‘https://www.baidu.com/‘>百度主页</a>" escapeXml="true"/><br/> 当escapeXml="false"时:<c:out value="<a href=‘https://www.baidu.com/‘>百度主页</a>" escapeXml="false"/></body>执行InitJSTLDataServlet,运行结果:

图9-10
从结果可以发现,当value中输出的对象不存在或为空,会输出default指定的默认值;当escapeXml为false时,会将value中的内容先渲染成HTML样式再输出显示。
③移除标签<c:remove>
<c:set>标签的作用,是给变量在某个作用域内赋值。与之相反,<c:remove>标签的作用,是移除某个作用域内的变量。
语法:
<c:remove var="variableName" scope="scope"/>
var:等待被移除的变量名
scope:变量被移除的作用域,有4个可填项:page,request,session和application。
示例:
index.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>…<body> … 并不存在的一个变量varDemo: <c:out value="${varDemo }" default="不存在"/><br/> 在request作用域内,给varDemo赋值 为LanQiao<c:set var="varDemo" value="LanQiao" scope="request"/><br/> 再次观察varDemo: <c:out value="${varDemo }" default="不存在"/><br/> 在request作用域内,将varDemo移除:<c:remove var="varDemo" scope="request"/> <br/> 再次观察varDemo: <c:out value="${varDemo }" default="不存在"/><br/></body>运行结果:

图9-11
JSTL的条件标签库,包含单重选择标签<c:if>和多重选择标签<c:choose>、<c:when>、 <c:otherwise>。
①单重选择标签<c:if>
类似于Java中的if选择语句。
语法:
<c:if test="condition" var="variableName" scope="scope"> 代码块 </c:if>test:判断条件,值为true或false,通常用EL表达式表示。当值为true时才会执行代码块中的内容。
var:可选项。保存test的判断结果(true或false)。
scope:可选项。设置此变量的作用域,有4个可填项:page,request,session和application。
示例:
JSTLDemo02.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>…<body> … <c:if test="${3>2 }" var="result" scope="request"> 3>2结果是:${result } </c:if></body>运行结果:

图9-12
②多重选择标签<c:choose>
<c:choose>的功能类似于Java中的多重if。
语法:
<c:choose> <c:when test=""> 代码块1 </c:when> <c:when test=""> 代码块2 </c:when> ... <c:otherwise> 代码块n </c:otherwise> </c:choose>其中,<c:when test="">类似于Java中的判断语句:if()和else if();<c:otherwise>类似于多重if中最后的else。具体的流程是:当<c:when>中的test为true时,执行当前<c:when>标签中的代码块;如果所有when中的test都为false,则才会执行<c:otherwise>中的代码块。
示例:
JSTLDemo02.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>…<body> … <c:set var="role" value="学生" /> <c:choose> <c:when test="${role eq ‘老师‘ }"> 老师相关代码 </c:when> <c:when test="${role eq ‘学生‘ }"> 学生相关代码 </c:when> <c:otherwise> 管理员相关代码 </c:otherwise> </c:choose></body>运行结果:

图9-13
<c:forEach>标签库在Java之中有两种for循环,一种是传统的for循环,形式如for(int i=0;i<10;i++);另一种是增强的for循环,形式如for(String name : names) ,此处的names是字符串数组。类似的,在JSTL中也提供了两种<c:forEach>标签与之相对应,一种用于遍历集合对象的成员,另一种用于让代码重复的循环执行。
①遍历集合对象的成员
语法:
<c:forEach var="variableName" items="collectionName"varStatus="variableStatusInfo" begin="beginIndex"end="endIndex" step="step"> 迭代集合对象的相关代码</c:forEach>var:当前对象的引用,即表示循环正在遍历的那个对象。例如,当循环遍历到第一个成员时,var 就代表第一个成员;当循环遍历到第二个成员时,var 就代表第二个成员……
items:当前循环的集合名。
varStatus:可选项。存放var所引用成员的相关信息,如索引号(index)等。
begin:可选项。遍历集合的开始位置,从0开始。
end:可选项。遍历集合的结束位置。
step:可选项,默认为1。遍历集合的步长,比如当step为1时,会依次遍历第0个、第1个、第2个……;当step为2时,会依次遍历第0个、第2个、第4个……。
示例: 先通过Servlet给集合中加入数据,再用<c:forEach>遍历输出。
InitJSTLForeachDataServlet.java
package org.lanqiao.servlet;//省略importpublic class InitJSTLForeachDataServlet extends HttpServlet { … protected void doPost(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException { //Address类包含家庭地址和学校地址两个属性 Address add1 =new Address("北京朝阳区","北京大兴区"); Address add2 =new Address("陕西西安","广州东莞"); List<Address> addresses = new ArrayList<Address>(); addresses.add(add1); addresses.add(add2); //强addresses集合放入request作用域内 request.setAttribute("addresses", addresses); request.getRequestDispatcher("JSTLDemo02.jsp").forward(request, response); }}JSTLDemo02.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %><body> … <c:forEach var="add" items="${addresses }" varStatus="status" > ${status.index}:家庭地址:${add.homeAddress } – 学校地址:${add.schoolAddress }<br/> </c:forEach></body>执行http://localhost:8888/ELAndJSTLDemo/InitJSTLForeachDataServlet,运行结果:

图9-14
②迭代指定的次数
语法:
<c:forEach var="variableName" varStatus="variableStatusInfo"begin="beginIndex" end="endIndex" step="step"> 循环体 </c:forEach>其中var、varStatus、begin、end、step属性的含义,与“遍历集合对象的成员”中对应的属性含义相同,并且能发现此种方式的<c:forEach>缺少了“items”属性。此种方式的<c:forEach>主要用来让循环体执行固定的次数。
示例:
JSTLDemo02.jsp
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %><body> … <c:forEach begin="0" end="2" step="1"> LanQiao<br> </c:forEach></body>以上代码中的<c:forEach>类似于Java中的for(int i=0;i<2;i++),运行结果:

图9-15
至此我们就学完了EL表达式和JSTL标签库的相关内容,读者可以使用它们来替换以前JSP页面中的Scriptlet。
一、选择题
1.下面( )不是与范围有关的EL隐式对象。(选择一项)(难度★)
A.cookieScope
B.pageScope
C.sessionScope
D.applicationScope
2.如果session中已经有属性名为user的对象,则EL表达式${not empty sessionScope.user}的值为( )。(选择一项)(难度★)
A.true
B.false
C.null
D.user
二、简答题
1.在使用EL表达式时,如果不显式指定对象的作用域范围,则系统会按照什么顺序依次查找?(难度★)
2.使用JSTL标签<c: ... >之前,需要进行哪些准备工作?(难度★)
3.请问<c:set>标签有哪几种?各如何使用?(难度★★)
4.请介绍JSTL中的for Each迭代标签有哪些属性,并简要描述各属性的含义。(难度★★★)
5.使用EL和JSTL继续优化第七章练习题中的“部门管理系统”。(难度★★★)
原文:https://www.cnblogs.com/gu-bin/p/10547363.html