课堂派题库格式转换程序

一、背景

这个学期开设的Java程序设计课程,需要用课堂派来签到和平常练习,老师苦于课堂派后台的课堂测试需要人工填入题目,好在课堂派运行符合格式的题目直接从word文档中导入,于是让我们来写一个小程序来实现这个格式转换。老师用的源题库是有特殊格式的,这样我们就可以通过一些特殊的标记来实现对源题库的读取,我最开始使用C++来实现的,后来补充的Java的版本。该程序主要是对string的操作,二者之中有许多相同的函数方法,但具体的写法却有所不同,同时Java的限制数组越界、不能更改String的一些特点,也让有些头疼,Java写的还是太少了!

二、格式要求

源题库样例:

目标题库格式:

 

具体要求:

1.第一行为题干,中间不要换号(课堂派后台只能识别一行题干),可以加题号,也可以不加题号,编写程序时可以通过一个选项开关让使用者决定是否需要保留题号。

2.答案要写在题目开头或者结尾,用括号括起来,大小写括号均可。在题干末尾添加本题的类型和分数,用大括号括起来。题目的类型需要通过答案的个数自己判断,题目的分数需要程序的使用者来输入。

3.题干之后是选项行,选项中不要空行,一个选项是单独的一行,选项字母需要英文大写。

4.每一道题与下一道题目之间需要空三行作为标志。

5.源题库中的有些题目可能会有解析,这里解析不需要提取处理。

6.源题库中出现的Chapter和Section的内容可以输出到控制台,目标文件中不要出现。

 

三、C++源代码

 

#include<fstream>  //ifstream
#include<iostream>
#include<string>
#include<map>
using namespace std;

struct subject
{
    int id;//题目序号
    int type;//题目类型
    string stem;//题干内容
    map<string,string>options;//选项与答案的map键值对
    int num;//选项个数
    string score;//题目分数
    string key;//答案
    string analysis;//有些题目需要解析
};

int main()
{
    string s,temp;
    string path;
    ofstream outf;
    ifstream inf;
    int cnt=0;//记录题目的个数
    int num1=0;//单选题个数
    int num2=0;//多选题个数
    int flag=0;//默认保留原题号
    char select;//是否保留题号的选项
    string score;
    struct subject sbj[100];
    string::size_type pos(0);
    cout << "请输入需要转换的文本文件路径:(例:C://in.txt)" <<endl;
    cin >> path;
    if(inf)
    {
        cout << "打开文本文件成功!" <<endl;
    }
    else
    {
        cout << "打开文本文件失败!" <<endl;
    }
    inf.open(path);//打开文本文件I://chapter1.txt
    cout << "请输入转换输出的文本文件路径:(例:C://out.txt)" <<endl;
    cin >> path;
    outf.open(path);//需要写入的文本文件

    //目标文本的读取
    cout << "转换完成后的题目是否保留原题号?(y/n)" <<endl;
    while(1)
    {
        cin >> select;
        if(select=='y')
        {
            flag=1;
            break;
        }
        else if(select=='n')
        {
            flag=0;
            break;
        }
        else
        {
            cout << "只能输入y或n,请重新输入!" <<endl;
        }
    }
    while (getline(inf, s))
    {
        if(s.substr(0,7)=="Chapter")
        {
            cout<<s<<endl;
        }
        else if(s.substr(0,7)=="Section")
        {
            cout<<s<<endl;
        }
        //提取选项信息
        else if(s.substr(0,1)>="0"&&s.substr(0,1)<="9") //提取题干信息
        {
            cnt++;
            if(flag==1)
            {
                pos =s.find(".");
                temp=s.substr(pos+1,s.length()-pos);
                sbj[cnt].stem = temp;//保存题干内容
            }
            else if(flag==0)
            {
                sbj[cnt].stem = s;
            }
            sbj[cnt].id = cnt;
            sbj[cnt].num = 0;
            while (getline(inf, s))
            {
                if(s.substr(0,4)=="Key:")//提取答案信息
                {
                    temp=s.substr(4,s.length()-4);
                    pos =temp.find(" ");
                    sbj[cnt].key=temp.substr(0,pos);
                    cout<<sbj[cnt].key<<endl;
                    for(int i=0; i<sbj[cnt].key.length(); i++)//答案替换为大写字母
                    {
                        if(sbj[cnt].key[i]>='a'&&sbj[cnt].key[i]<='z')
                        {
                            sbj[cnt].key[i]=sbj[cnt].key[i]-32;
                        }
                    }
                    pos=sbj[cnt].key.length()-1;
                    if(temp.length()>sbj[cnt].key.length())
                    {
                        sbj[cnt].analysis=temp.substr(pos,temp.length()-pos);
                    }

                }
                else if((s[0]>='a'&&s[0]<='z')||(s[0]>='A'&&s[0]<='Z'))//提取选项信息
                {
                    if(s[0]>='a'&&s[0]<='z')//选项是小写字母,转为大写字母
                    {
                        s[0]=s[0]-32;
                    }
                    sbj[cnt].options.insert(pair<string,string>(s.substr(0,1),s.substr(2,s.length()-2)));//保存选项
                    sbj[cnt].num++;
                }
                else if(s.substr(0,1)=="#")//分隔符,下一道题
                {
                    break;
                }
            }
            cout<<""<<cnt<<"道题识别成功!!"<<endl;
            cout<<sbj[cnt].stem<<endl;
            cout<<"答案:"<<sbj[cnt].key<<endl<<endl;
        }
    }
    cout<<"读取文本文件成功!"<<endl;

    //目标文本的处理
    for(int i=1; i<=cnt; i++)
    {
        pos=0;
        sbj[i].stem.insert(pos,""+sbj[i].key+"");//答案的插入
        if(sbj[i].key.length()==1)
        {
            sbj[i].stem=sbj[i].stem+"[单选题]";
            num1++;
        }
        else if(sbj[i].key.length()>1)
        {
            sbj[i].stem=sbj[i].stem+"[多选题]";
            num2++;
        }
    }
    cout<<"检测到单选题"<<num1<<"道,多选题"<<num2<<""<<endl;
    cout<<"请输入题目的分数:[1~100]"<<endl;
    cin>>score;

    for(int i=1; i<=cnt; i++)
    {
        sbj[i].score=score;
        sbj[i].stem=sbj[i].stem+"["+score+"分]";
    }
    for(int i=1; i<=cnt; i++)//文本写入
    {
        outf<<sbj[i].stem<< '\n';//写入题干
        map<string,string>::iterator iter;
        for(iter = sbj[i].options.begin(); iter != sbj[i].options.end(); iter++)
        {
            outf<<iter->first<<"."<<iter->second<< '\n';//写入选项
        }
        outf<< '\n'<< '\n'<< '\n';
    }
    inf.close();
    outf.close();
    cout<<"文本转换完成!"<<endl;
    return 0;
}

 

四、Java源代码

import java.io.*;
import java.util.*;
//import java.util.Map;
//import java.util.HashMap;
class subject
{
     int id;//题目序号
     int type;//题目类型
     int num;//选项个数
     String stem;//题干内容
     int score;//题目分数
     String key;//答案
     String analysis;//有些题目需要解析
     Map<String,String>options=new LinkedHashMap<String,String>();//map映射,LinkedHashSet正序输出
     public subject()//构造函数
     {
         id = 0;
         type = 1;
         num = 0;
         score =1;
         stem = null;
         key = null;
         analysis = null;
     }
}

public class Main {
    
    public static void main(String args[]) {
        int cnt = 0;
        int num1=0;//单选题个数
        int num2=0;//多选题个数
        int flag=0;//默认保留原题号
        char select;//是否保留题号的选项
        int score;
        String pathname_in;
        String pathname_out;
        Scanner in=new Scanner(System.in);
        subject [] sbj;
        sbj = new subject[1000];
        for(int i=0;i<sbj.length;i++){
            sbj[i]= new subject();
        }
        //String pathname = "I://chapter1.txt"; 
        System.out.println("请输入需要转换的文本文件路径:(例:C://in.txt)");
        pathname_in =in.nextLine();
        System.out.println("请输入转换输出的文本文件路径:(例:C://out.txt)");
        pathname_out =in.nextLine();
        System.out.println("转换完成后的题目是否保留原题号?(y/n)");
        while(true){
            select =in.nextLine().charAt(0);
            if(select=='y'){
                flag=1;
                break;
            }
            else if(select=='n'){
                flag=0;
                break;
            }
            else
            {
                System.out.println("只能输入y或n,请重新输入!");
            }
        }
        
        try (FileReader reader = new FileReader(pathname_in);
             BufferedReader br = new BufferedReader(reader) // 建立一个对象,它把文件内容转成计算机能读懂的语言
        ) {
            String line;
            //网友推荐更加简洁的写法
            while ((line = br.readLine()) != null) {
                if(line.length()==0){
                    continue;
                }
                else if(line.substring(0,7).equals("Chapter")){
                    System.out.println(line);
                }
                else if(line.substring(0,7).equals("Section")){
                    System.out.println(line);
                }
                else if((line.charAt(0)>='0')&&(line.charAt(0)<='9')){
                    cnt++;
                    String temp;
                    int pos;
                    if(flag==1){
                        pos =line.indexOf('.');
                        temp=line.substring(pos+1,line.length());
                        sbj[cnt].stem = temp;//保存题干内容
                    }
                    else if(flag==0){
                        sbj[cnt].stem = line;
                    }
                    sbj[cnt].id=cnt;
                    sbj[cnt].num=0;
                    while ((line = br.readLine()) != null) {
                        System.out.println(line);
                        if(line.length()==0){
                            continue;
                        }
                        else if(line.charAt(0)=='#'){
                            break;//分隔符
                        }
                        else if(line.length()<=4)
                        {
                            continue;
                        }
                        else if(line.substring(0,4).equals("Key:")){
                            temp = line.substring(4,line.length());//答案和解析的内容
                            pos = temp.indexOf(' ');
                            if(pos==-1){//没有解析说明
                                sbj[cnt].key= temp.toUpperCase();//写回
                                System.out.println(sbj[cnt].key);
                            }
                            else {
                                sbj[cnt].key=temp.substring(0,pos);//答案选项
                                sbj[cnt].key=sbj[cnt].key.toUpperCase();//写回
                                pos=sbj[cnt].key.length()-1;
                                if(temp.length()>sbj[cnt].key.length()){
                                    sbj[cnt].analysis=temp.substring(pos,temp.length()-pos);
                                }
                            }
                        }
                        else if(((line.charAt(0)>='a'&&line.charAt(0)<='z')||(line.charAt(0)>='A'&&line.charAt(0)<='Z'))&&(line.charAt(1)=='.')){
                            String opt = line.substring(0,2);//选项
                            String detail = line.substring(2,line.length());//选项描述
                            sbj[cnt].options.put(opt,detail);
                            sbj[cnt].num++;  
                        }
                        else
                        {
                            continue;
                        }
                    }
                    System.out.println("第"+cnt+"道题识别成功!!");
                    System.out.println(sbj[cnt].stem);
                    System.out.println("答案:"+sbj[cnt].key+"\n");
                }
                else
                {
                    continue;
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        System.out.println("读取文本文件成功!");
        for(int i=1;i<=cnt;i++){
            sbj[i].stem="( "+sbj[i].key+" )"+sbj[i].stem;
            if(sbj[i].key.length()==1){
                sbj[i].stem=sbj[i].stem+"[单选题]";
                num1++;
            }
            else if(sbj[i].key.length()>1){
                sbj[i].stem=sbj[i].stem+"[多选题]";
                num2++;
            }
            
        }
        System.out.println("检测到单选题"+num1+"道,多选题"+num2+"道");
        System.out.println("请输入题目的分数:[1~100]");
        score =in.nextInt();
        for(int i=1; i<=cnt; i++){
            sbj[i].score=score;
            sbj[i].stem=sbj[i].stem+"["+score+"分]";
        }
        try {
            File writeName = new File(pathname_out); // 相对路径,如果没有则要建立一个新的output.txt文件
            writeName.createNewFile(); // 创建新文件,有同名的文件的话直接覆盖
            try (FileWriter writer = new FileWriter(writeName);
                 BufferedWriter out = new BufferedWriter(writer)
            ) {
                for(int i=1; i<=cnt; i++){//文本写入
                    out.write(sbj[i].stem+"\r\n"); 
                    Iterator<Map.Entry<String, String>> entries = sbj[i].options.entrySet().iterator();  
                    while(entries.hasNext()){  
                        Map.Entry<String, String> entry = entries.next();  
                        out.write(entry.getKey()+entry.getValue()+"\r\n");
                    }  
                    out.write("\r\n"+"\r\n"+"\r\n"); 
                }
                out.flush(); // 把缓存区内容压入文件
            }
        } catch (IOException e){
            e.printStackTrace();
        }
        System.out.println("文本转换完成!");
    }
}

 

 五、遇到的一些问题

 

1.Java中关于HashMap的元素遍历的顺序问题

摘自https://www.cnblogs.com/xdp-gacl/p/3558625.html

在使用如下的方式遍历HashMap里面的元素时

for (Entry<String, String> entry : hashMap.entrySet()) {
     MessageFormat.format("{0}={1}",entry.getKey(),entry.getValue());
 }

  发现得到的元素不是按照之前加入HashMap的顺序输出的,这个问题我之前倒是没有注意过,后来上网查了一下原因,发现是:HashMap散列图、Hashtable散列表是按“有利于随机查找的散列(hash)的顺序”。并非按输入顺序。遍历时只能全部输出,而没有顺序。甚至可以rehash()重新散列,来获得更利于随机存取的内部顺序。
  总之,遍历HashMap或Hashtable时不要求顺序输出,即与顺序无关。

Map<String, String> paramMap = new HashMap<String, String>();

  可以用java.util.LinkedHashMap 就是按加入时的顺序遍历了。

Map<String, String> paramMap = new LinkedHashMap <String, String>();

  类似的还有 java.util.LinkedHashSet

 

2.C++中对字符串中所有指定的子串进行替换的函数

/*
 * string& str              源字符串
 * const string& old_value  被替换子串
 * const string& new_value  替换子串
 */
string& replace_all(string& str,const string& old_value,const string& new_value)//替换函数
{
    while (true)
    {
        string::size_type   pos(0);
        if ((pos = str.find(old_value)) != string::npos)
        {
            str.replace(pos, old_value.length(), new_value);
        }
        else
        {
            break;
        }
    }
    return   str;
}
posted @ 2019-09-12 20:50  王陸  阅读(2731)  评论(0编辑  收藏  举报