为了更好地应用XML,就写了这个小项目。


下面是我的项目的目录结构


项目思路

  • dao是Date Access Object 数据访问层,主要是负责操作数据
  • domain是实体层,类似于bean层,放置项目用到的实体Student
  • utils层是有关于XML操作的部分(一般实际开发中是数据库操作部分)
  • view层是视图层(实际开发中是GUI层,与用户直接打交道)
  • Student.xml在这里相当于我们的一个小小的数据库

dao层设计


按照习惯的命名规则,我命名为StudentDao.java,具体实现的功能有添加学生信息,查找学生信息,删除学生信息。这里仅仅是直接对数据操作的模块,而把底层的操作XML文档的放到了utils中。这也在一定程度上实现了分层的思想,虽然这并不明显,也并不必需!

package dao;

import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

import utils.XMLUtils;
import domain.Student;

public class StudentDao {

    /**
     * 添加学生信息模块
     * @param student
     */
    public void add(Student student) {

        try {
            Document document = XMLUtils.getDocument();
            Element student_node = document.createElement("student");
            student_node.setAttribute("examid", student.getExamid());
            student_node.setAttribute("idcard", student.getIdcard());
            Element name = document.createElement("name");
            name.setTextContent(student.getName());

            Element location = document.createElement("location");
            location.setTextContent(student.getLocation());

            Element grade = document.createElement("grade");
            // 这里是一个类型转换的隐藏之处。不太明显但是却十分的重要
            grade.setTextContent(student.getGrade() + "");

            // 将新生成的三个子节点插入到student标签内
            student_node.appendChild(name);
            student_node.appendChild(location);
            student_node.appendChild(grade);

            // 对总的xml文档中添加一个学生信息
            document.getElementsByTagName("exam").item(0)
                    .appendChild(student_node);
            //将内存中的操作对象写回到xml文件,真正实现对文件的操作
            XMLUtils.write2Xml(document);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            throw new RuntimeException(e);
        }

    }

    public void delete(String name) {

        try {
            Document document = XMLUtils.getDocument();
            NodeList name_node_list = document.getElementsByTagName("name");

            for (int i = 0; i < name_node_list.getLength(); i++) {
                if (name_node_list.item(i).getTextContent().equals(name)) {
                    Element person_node = (Element) name_node_list.item(i)
                            .getParentNode();
                    Element exam_node = (Element) person_node.getParentNode();
                    exam_node.removeChild(person_node);
                    //不要忘记将操作过的数据写回,否则原信息是不会发生变化的
                    XMLUtils.write2Xml(document);
                    System.out.println("恭喜,学生信息删除成功!");
                }
            }
        } catch (Exception e) {
            System.out.println("对不起,删除操作未成功完成!请重试!");
            throw new RuntimeException(e);
        }

    }

    /**
     * 给定学生的考号查找该同学的详细的信息(不用姓名的原因是姓名具有不唯一性)
     * @param examid
     * @return
     */
    public Student find(String examid) {
        Student student=null;
        try {
            Document document = XMLUtils.getDocument();
            NodeList examid_node_list = document.getElementsByTagName("student");

            //查找准考证号与查找值相一致的学生节点
            for(int i=0; i<examid_node_list.getLength();i++){
                Element examid_element = (Element) examid_node_list.item(i);
                if(examid_element.getAttribute("examid").equals(examid.toString().trim())){
                    //采用非递归的方式获取student的详细信息
                    student = getStudentInfo(examid_element);
                    return student;
                }else{
                    continue;
                }
            }

        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("对不起,未能正确的找到您要查找的学生的姓名!请确认后重新尝试!");
        }
        return student;
    }

    /**
     * 给定一个节点,采用非递归的方式遍历该学生节点的详细的信息
     * 缺点:不能很好地复用代码,代码维护性较差
     */
    public Student getStudentInfo(Element node){
        Student student = new Student();
        if(node!=null){
            String examid = node.getAttribute("examid");
            String idcard = node.getAttribute("idcard");
            NodeList node_list = node.getChildNodes();

            //由于collection 的不确定性,是随机取出的数据,导致bean中的数据不太对应
            Node node_name = node_list.item(1);
            String name = node_name.getTextContent();
            Node node_location = node_list.item(2);
            String location = node_location.getTextContent();
            Node node_grade = node_list.item(0);
            String grade = node_grade.getTextContent()+"0.0";

            //将获取的信息保存到bean中,并作为返回值返回!
            student.setExamid(examid);
            student.setGrade(Double.parseDouble(grade));
            student.setIdcard(idcard);
            student.setLocation(location);
            student.setName(name);
            return student;
        }
        System.out.println("this find operation is false!");
        return null;
    }

}

domain–实体层(bean)


package domain;

public class Student {

//注意将Student具有的属性设置为私有的成员,然后设置getter,setter来访问
    private String idcard;
    private String examid;
    private String name;
    private String location;
    private double grade;
    public String getIdcard() {
        return idcard;
    }
    public void setIdcard(String idcard) {
        this.idcard = idcard;
    }
    public String getExamid() {
        return examid;
    }
    public void setExamid(String examid) {
        this.examid = examid;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getLocation() {
        return location;
    }
    public void setLocation(String location) {
        this.location = location;
    }
    public double getGrade() {
        return grade;
    }
    public void setGrade(double grade) {
        this.grade = grade;
    }
}

Utils层充当工具,关联底层操作


package utils;

import java.io.File;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;

import org.w3c.dom.Document;

public class XMLUtils {

    //all the utils methods are static
    public static Document getDocument() throws Exception{

        DocumentBuilderFactory factory=DocumentBuilderFactory.newInstance();
        DocumentBuilder builder = factory.newDocumentBuilder();
        Document document = builder.parse(new File("src/Student.xml"));
        return document;
    }

    public static void write2Xml(Document document) throws Exception{

        TransformerFactory factory = TransformerFactory.newInstance();
        Transformer transformer=factory.newTransformer();

        //wrapper the two arguments
        DOMSource xmlSource = new DOMSource(document);

        StreamResult targetResult = new StreamResult(new File("src/Student.xml"));

        transformer.transform(xmlSource, targetResult);

}

}

小感悟:一般来说工具类的方法会做成静态的私有的,这样可以免去创建对象,又能很好的使用到它!


view层(与用户直接交互的模块)


这里仅仅是简单的一些有好的交互,并没有什么复杂的地方

package view;

import java.io.BufferedReader;
import java.io.InputStreamReader;

import dao.StudentDao;
import domain.Student;

public class Main {

    public static void main(String []args) throws Exception{
        System.out.println("添加学生(a),查找学生(f),删除学生(d)");
        System.out.println("请输入操作类型:");

        BufferedReader reader=new BufferedReader(new InputStreamReader(System.in));
        String type = reader.readLine();

        if(type.equalsIgnoreCase("a")){
            //add student
            //name examid idcard location grade
            try{
                System.out.println("请输入学生姓名: ");
                String name = reader.readLine();

                System.out.println("请输入学生准考证号: ");
                String examid = reader.readLine();

                System.out.println("请输入学生身份证号: ");
                String idcard = reader.readLine();

                System.out.println("请输入学生所在地: ");
                String location = reader.readLine();

                System.out.println("请输入学生成绩: ");
                String grade = reader.readLine();

                Student student = new Student();
                student.setExamid(examid);
                student.setIdcard(idcard);
                student.setLocation(location);
                student.setName(name);
                student.setGrade(Double.parseDouble(grade));

                StudentDao studentDao = new StudentDao();
                studentDao.add(student);

                System.out.println("恭喜:学生信息添加成功!");
            }catch(Exception e){
                e.printStackTrace();
                System.out.println("对不起,学生信息数据添加失败!");
            }

        }else if(type.equalsIgnoreCase("f")){
            //find student
            System.out.println("请输入学生的准考证号");
            StudentDao studentDao = new StudentDao();
            String examid = reader.readLine();
            Student student = studentDao.find(examid);
            PrintStudentInfo(student);

        }else if(type.equalsIgnoreCase("d")){
            //delete student
            System.out.println("请输入学生的姓名");
            StudentDao studentDao = new StudentDao();
            String name = reader.readLine();
            studentDao.delete(name);

        }else{
            System.out.println("Not Support!");
        }
    }

    private static void PrintStudentInfo(Student student) {
        // TODO Auto-generated method stub
        if(student!=null){
            System.out.println("学生考号: \t"+student.getExamid());
            System.out.println("学生姓名: \t"+student.getName());
            System.out.println("学生身份证号码 : \t"+student.getIdcard());
            System.out.println("学生所在地: \t"+student.getLocation());
            System.out.println("学生成绩: \t"+student.getGrade());
        }
    }

}

处理素材和结果


Student.xml文件示意如下:

<?xml version="1.0" encoding="UTF-8" standalone="no"?><exam>
    <student examid="222" idcard="111">
        <name>Spring</name>
        <location>DaLian</location>
        <grade>89</grade>
    </student>

    <student examid="444" idcard="333">
        <name>Summer</name>
        <location>ShangHai</location>
        <grade>97</grade>
    </student>

    <student examid="201492115" idcard="410728">
        <name>郭璞</name>
        <location>大连</location>
        <grade>89.0</grade>
    </student>
    <student examid="123" idcard="1234567">
        <name>未命名</name>
        <location>不知道</location>
        <grade>0.0</grade>
    </student>

</exam>

操作结果:

添加学生操作



添加学生信息结果



删除学生操作



删除学生结果



查找学生详细信息


总结:


优点:

  • 使用了分层思想(虽然有些地方并不是很明显)
  • 模块化的操作时的代码逻辑更加的清晰
  • 较好的实现了对xml文件的CRUD操作
  • 结合了具体的项目,运用到了相关的知识点

    缺点:

  • 代码处理上仍旧有很大的重复性

  • 设计模式的运用不是很明显
  • 界面有点糟糕

    有待改进之处:

  • 使用GUI,rich用户体验

  • 底层处理逻辑应更加的精简

学生信息管理小系统(以XML为存储方式)的更多相关文章

  1. 讨论两种Redis中Token的存储方式

    摘要:本文讨论一个问题:存储token时,token与对应用户id谁来作为key? 问题起源问题起源于要给公司的后台管理系统添加权限管理,选用的是开源框架shiro,而原本系统上是采用token做了登 ...

  2. 写一个小demo过程中遇到的各种问题 学生管理考勤系统(网页模拟)

    添加与新增一些小玩意的1.0.3版本:传送门 各位带哥,这不是你们要的c++.java.c#作业哈 课上要求做个小作业,学生管理考勤系统,原本想着是个练手的好机会,结果只证实了我还是个弟中弟. 设想的 ...

  3. C语言判断系统数据大/小端存储方式

    小端存储:数据的低位部分,存储于存储器的低地址空间里. 大端存储:数据的低位部分,存储于存储器的高地址空间里. 首先,一般PC数据存储方式是小端存储. 基本实现思想是:将存储器中所存的数据按字节以地址 ...

  4. iOS 应用数据存储方式(XML属性列表-plist)

    iOS 应用数据存储方式(XML属性列表-plist) 一.ios应用常用的数据存储方式 1.plist(XML属性列表归档) 2.偏好设置 3.NSKeydeArchiver归档(存储自定义对象) ...

  5. iOS开发UI篇—ios应用数据存储方式(XML属性列表-plist)

    iOS开发UI篇—ios应用数据存储方式(XML属性列表-plist) 一.ios应用常用的数据存储方式 1.plist(XML属性列表归档) 2.偏好设置 3.NSKeydeArchiver归档(存 ...

  6. ios应用数据存储方式(XML属性列表-plist) - 转

    一.ios应用常用的数据存储方式  1.plist(XML属性列表归档)  2.偏好设置  3.NSKeydeArchiver归档(存储自定义对象)  4.SQLite3(数据库,关系型数据库,不能直 ...

  7. LINUX系统下添加映射存储LUN

    LINUX系统下添加映射存储LUN(无需重启) 背景:Oracle rac环境 添加新实例,重新划分存储空间,从存储映射新的LUN. 问题:映射后,linux操作系统无法识别新的LUN,不能重启系统, ...

  8. 私人定制自己的linux小系统

     私人定制自己的linux小系统 一.前言    linux操作系统至1991.10.5号诞生以来,就源其开源性和自由性得到了很多技术大牛的青睐,每个linux爱好者都为其贡献了自己的一份力,不管是在 ...

  9. 为自己打造Linux小系统

     一.前言 Linux操作系统至1991.10.5号诞生以来,就源其开源性和自由性得到了很多技术大牛的青睐,每个Linux爱好者都为其贡献了自己的一份力,不管是在Linux内核还是开源软件等方面,都为 ...

随机推荐

  1. async/await,了解一下?

    上一篇博客我们在现实使用和面试角度讲解了Promise(原文可参考<面向面试题和实际使用谈promise>),但是Promise 的方式虽然解决了 callback hell,但是这种方式 ...

  2. Struts2 之 modelDriven & prepare 拦截器详解

    struts2 ModelDriven & Prepareable 拦截器 前面对于 Struts2 的开发环境的搭建.配置文件以及 Struts2 的值栈都已经进行过叙述了!这次博文我们讲解 ...

  3. js删除数组中的元素delete和splice的区别

    例如有一个数组是 :var textArr = ['a','b','c','d']; 这时我想删除这个数组中的b元素: 方法一:delete 删除数组 delete textArr[1]  结果为:  ...

  4. iframe嵌套页面 音频在微信公众号环境无法播放

    在微信公众号中 没有iframe的时候window.WeixinJSBridge为对象,有iframe时为undefined 要使用 window.parent.WeixinJSBridge得到 if ...

  5. 干货满满,腾讯云+社区技术沙龙 Kafka Meetup 深圳站圆满结束

    欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 云+导语:4月22日,由腾讯云和 Kafka 社区主办.开源中国协办的腾讯云+社区技术沙龙 Kafka Meetup 深圳站在腾讯大厦举行, ...

  6. 正则替换replace中$1的用法以及常用正则

    一.repalce定义 用于在字符串中用一些字符替换另一些字符,或替换一个与正则表达式匹配的子串. stringObject.replace(regexp/substr,replacement)参数一 ...

  7. SpringBoot学习之启动探究

    SpringApplication是SpringBoot的启动程序,我们通过它的run方法可以快速启动一个SpringBoot应用.可是这里面到底发生了什么?它是处于什么样的机制简化我们程序启动的?接 ...

  8. delphi 组件安装教程详解

    学习安装组件的最好方法,就是自己编写一个组件并安装一遍,然后就真正明白其中的原理了.   本例,编写了两个BPL, dclSimpleEdit.bpl 与 SimpleLabel.bpl ,其中,dc ...

  9. Windows环境下,从零开始搭建Nodejs+Express+Ejs框架(一)---安装nodejs

    第一步,安装nodejs https://nodejs.org/en/download/ 这个是nodejs的官网,由于操作系统是win7 64位的,所以,我下载的是node-v8.11.1-x64的 ...

  10. Check the string CodeForces - 960A

    A has a string consisting of some number of lowercase English letters 'a'. He gives it to his friend ...