0基础java初学者都能做的打字通小游戏? 内含源码解读和细致讲解!!

本文涉及的产品
智能开放搜索 OpenSearch行业算法版,1GB 20LCU 1个月
实时数仓Hologres,5000CU*H 100GB 3个月
实时计算 Flink 版,5000CU*H 3个月
简介: 0基础java初学者都能做的打字通小游戏? 内含源码解读和细致讲解!!

打字通游戏实现

打字通游戏简介:主要应用了流与文件的知识,结合oop,对菜单功能,逐行比对功能,读取输入行功能,结果输出功能等进行了详细的实现,由于本游戏需要基于一定故事文本作为数据,建议先行下载资源后在控制台创建相对路径,便于游戏的实现。由于本案例只是作为知识巩固用,所以本文作者仅仅在控制台进行了源码的实现,并没有进行网页或者其他的设计。
代码:Java
工具:Idea编辑器(2019),notepad++
**本文关于故事文本的资源如下:
提取码:6666

结果展示image.png

image.png
(数据测试的时候是复制粘贴所以数据异常 但是打字结果的呈现功能是没有问题的)

Story类

//由于读取每一个故事要按行读取,要通过继承Iterator接口规范迭代器行为
public class Story implements Iterator<String>{
   
   
    //注意:该类中的方法参数都是File Story
    private List<String> list;//此处将集合命名为属性的原因见下图
    private Iterator<String> itLine;//将迭代器封装为类属性,仅供story类中按行读取的迭代操作使用(迭代器封装为属性常见于一次性操作)
    private String name;

    /**
     * 将文件名转化为故事名,初始化属性(即.log前面的部分,并且应该全部转化为大写,实例:happy_day.log -> Happy Day)
     * @param story 文件引用
     * @return
     */

    private String parseName(File story){
   
   
        final StringBuilder sb = new StringBuilder();
        for (String s : story.getName().split("\\.")[0].split("_")) {
   
   
            if(sb.length()>0) {
   
   
                sb.append(" ");
            }
            sb.append(s.substring(0, 1).toUpperCase());
            sb.append(s.substring(1));
        }
        return sb.toString();
    }

    public Story(File story){
   
   
        name = parseName(story);
        list = new ArrayList<>();
        BufferedReader reader = null;
        try {
   
   
            reader = new BufferedReader(new FileReader(story),256);//FileReader的方法参数可以是文件或者是地址
            String line;
            while(null != (line = reader.readLine())){
   
   
                list.add(line);//迭代器按行读取到的line添加到集合属性中,从而完成数据读取
            }
        } catch (Exception e) {
   
   
            e.printStackTrace();
        }finally{
   
   
            if(null != reader){
   
   
                try {
   
   
                    reader.close();
                } catch (IOException e) {
   
   
                    e.printStackTrace();
                }
            }
        }
    }

    public String getName() {
   
   
        return name;
    }

    //重置迭代器:读取完一行之后迭代器回到行首
    public void reset(){
   
   
        itLine = list.iterator();
    }

    @Override
    public boolean hasNext() {
   
   
        return itLine.hasNext();
    }

    @Override
    public String next() {
   
   
        return itLine.next();
    }
}

image.png
图1.1解释:最基础的数据存储方式就是按行的磁盘存储,但有一种常见的方式是通过集合存储之后再读取到文件之中,而优化则是直接将集合定义为该文件的属性,并且直接写到集合之中。而在代码中,story的数据类型即为File,用一个List来存储数据。

工具类

public class IO {
   
   
    // 输入工具
    private static final BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    // 获取输入行
    public static String inputLine() throws IOException{
   
   
        return br.readLine();
    }
    // 输入指定范围内整数
    public static int inputIntIn(String title,int min,int max) throws IOException{
   
   
        String str;
        System.out.printf("请输入%s:",title);
        do{
   
   
            str = br.readLine();
            int value;
            if(str.matches("\\d+") && (value = Integer.parseInt(str))>=min && value<=max){
   
   
                return value;
            }
            System.out.printf("%s输入非法,必须是%d~%d之间的整数,请重新输入!",title,min,max);
        }while(true);
    }
}

输入分析类:根据行得到单词集合迭代器,利用迭代器进行结果统计和输出。

public class InputAnalysis {
   
   
    public static final String CHAR_COUNT = "CHAR_COUNT";
    public static final String WORD_COUNT = "WORD_COUNT";
    public static final String CORRECT_CHAR_COUNT = "CORRECT_CHAR_COUNT";
    public static final String CORRECT_WORD_COUNT = "CORRECT_WORD_COUNT";


    /**
     * 传入行,提取空格和标点符号分隔出来的单词,并获得单词集合的迭代器
     * @param line 每行内容
     * @return
     */
    public static Iterator<String> lineToIterator(String line){
   
   //见下图
        List<String> words = new ArrayList<>();
        int fromIx = 0,toIx = 0;
        final char[] chars = line.toCharArray();
        while(toIx<chars.length){
   
   
            if(!Character.isLetterOrDigit(chars[toIx])){
   
   
                // 标点或者空格
                words.add(line.substring(fromIx,toIx));
                //如果是空格,则不需要读取空格
                if(Character.isSpaceChar(chars[toIx])){
   
   
                    fromIx = toIx+1;
                }else {
   
   //如果是标点,则需要读取到标点
                    fromIx = toIx;
                    words.add(line.substring(fromIx,fromIx+1));
                    fromIx++;
                }
                //fromIx需要移动到标点或者空格的下一位
            }
            toIx++;
        }
        return words.iterator();
    }

    /**
     * 利用上述方法分别获取故事行和输入行的迭代器,并进行比较
     * @param storyLine
     * @param inputLine
     * @return
     */
    public Map<String,Integer> compare(String storyLine, String inputLine){
   
   
        Map<String,Integer> map = new HashMap<>(4);
        final Iterator<String> itStory = lineToIterator(storyLine);
        final Iterator<String> itInput = lineToIterator(inputLine);
        int charCount = 0, wordCount = 0,correctCharCount = 0,correctWordCount = 0;
        // wordCount和charCount应该是在遍历storyWord的过程中需要遍历的,而correctCharCount/correctWordCount即需要在遍历的过程中对inputWord中的进行计数。
        while(itStory.hasNext()){
   
   
            final String storyWord = itStory.next();
            wordCount += 1;
            charCount += storyWord.length();
            if(itInput.hasNext()){
   
   
                final String inputWord = itInput.next();
                if(inputWord.equals(storyWord)){
   
   
                    correctWordCount+=1;
                    correctCharCount+=inputWord.length();
                }
            }
        }
        map.put(CHAR_COUNT,charCount);
        map.put(WORD_COUNT,wordCount);
        map.put(CORRECT_CHAR_COUNT,correctCharCount);
        map.put(CORRECT_WORD_COUNT,correctWordCount);
        return map;
    }


}

image.png

主类

public class YBPrinter {
   
   
    private List<Story> stories;//便于根据下标获取story

    public YBPrinter(String directory) {
   
   
        //创建文件内对象
        final File dir = new File(directory);
        //判定目录是否存在
        if(!dir.exists()){
   
   
            System.err.printf("故事目录%s不存在",directory);
            System.exit(0);
        }
        //获取目录下的内容列表
        final File[] files = dir.listFiles(f -> f.getName().matches("[a-z0-9]+(_[a-z0-9]+)*\\.log"));
        stories = new ArrayList<>(files.length);
        //遍历内容列表,构造出story,传入数组
        for (File file : files) {
   
   
            stories.add(new Story(file));
        }
    }
    //构建菜单
    private int chooseStory() throws IOException {
   
   
        System.out.println("======= 打字通小游戏 =======");
        System.out.println("\n可选文章如下:");
        int count = 0;
        final Iterator<Story> it = stories.iterator();
        //利用迭代器输出菜单
        while(it.hasNext()){
   
   
            System.out.printf("%d、%s\n",++count,it.next().getName());
        }
        //输入选择编号
        final int choice = IO.inputIntIn("请输入选项编号(输入0退出)", 0, stories.size());
        return choice;
    }

    public void start(){
   
   
        InputAnalysis analysis = new InputAnalysis();
        while(true){
   
   
            try{
   
   
                final int choice = chooseStory();
                if(choice==0){
   
   
                    return;
                }
                final Story story = stories.get(choice-1);
                story.reset();//每获取一个story,都要重置迭代器
                int charCount = 0, wordCount = 0,correctCharCount = 0,correctWordCount = 0;
                long startTime = System.currentTimeMillis();
                // 获取输入行与故事行的分析结果
                while(story.hasNext()){
   
   
                    final String storyLine = story.next();
                    if(storyLine.equals("")){
   
   
                        continue;
                    }
                    System.out.println(storyLine);
                    final String inputLine = IO.inputLine();
                    final Map<String, Integer> rst = analysis.compare(storyLine, inputLine);
                    charCount += rst.get(InputAnalysis.CHAR_COUNT);
                    wordCount += rst.get(InputAnalysis.WORD_COUNT);
                    correctCharCount += rst.get(InputAnalysis.CORRECT_CHAR_COUNT);
                    correctWordCount += rst.get(InputAnalysis.CORRECT_WORD_COUNT);
                }
                // 数据处理
                long stopTime = System.currentTimeMillis();
                // 将毫秒转化为分钟
                float timeConsumption = (stopTime-startTime)/60000.0f;
                float correctRate = correctWordCount*1.0f/wordCount;
                int charPerMinute = (int)Math.floor(correctCharCount/timeConsumption);
                System.out.printf("%s 本次打字结果如下:\n",story.getName());
                System.out.printf("总时间消耗:%.2f分钟\n",timeConsumption);
                System.out.printf("单词数:%d\n",wordCount);
                System.out.printf("字符数:%d\n",charCount);
                System.out.println(String.format("正确率:%.2f",correctRate*100)+"%");
                System.out.printf("总速度:%d 字母/分钟\n",charPerMinute);
            }catch(Exception e){
   
   
                e.printStackTrace();
            }
        }
    }
}

启动类

public class Start {
   
   
    public static void main(String[] args) {
   
   
        new YBPrinter("file/story").start();//此处为Directory中的story文件夹的相对路径,story文件夹则存放着各个故事
    }
}

分析

image.png
image.png

目录
相关文章
|
14天前
|
XML Java 编译器
Java注解的底层源码剖析与技术认识
Java注解(Annotation)是Java 5引入的一种新特性,它提供了一种在代码中添加元数据(Metadata)的方式。注解本身并不是代码的一部分,它们不会直接影响代码的执行,但可以在编译、类加载和运行时被读取和处理。注解为开发者提供了一种以非侵入性的方式为代码提供额外信息的手段,这些信息可以用于生成文档、编译时检查、运行时处理等。
50 7
|
25天前
|
数据采集 人工智能 Java
Java产科专科电子病历系统源码
产科专科电子病历系统,全结构化设计,实现产科专科电子病历与院内HIS、LIS、PACS信息系统、区域妇幼信息平台的三级互联互通,系统由门诊系统、住院系统、数据统计模块三部分组成,它管理了孕妇从怀孕开始到生产结束42天一系列医院保健服务信息。
28 4
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
71 2
|
6天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
52 13
|
19天前
|
缓存 监控 Java
Java线程池提交任务流程底层源码与源码解析
【11月更文挑战第30天】嘿,各位技术爱好者们,今天咱们来聊聊Java线程池提交任务的底层源码与源码解析。作为一个资深的Java开发者,我相信你一定对线程池并不陌生。线程池作为并发编程中的一大利器,其重要性不言而喻。今天,我将以对话的方式,带你一步步深入线程池的奥秘,从概述到功能点,再到背景和业务点,最后到底层原理和示例,让你对线程池有一个全新的认识。
50 12
|
14天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
1月前
|
人工智能 监控 数据可视化
Java智慧工地信息管理平台源码 智慧工地信息化解决方案SaaS源码 支持二次开发
智慧工地系统是依托物联网、互联网、AI、可视化建立的大数据管理平台,是一种全新的管理模式,能够实现劳务管理、安全施工、绿色施工的智能化和互联网化。围绕施工现场管理的人、机、料、法、环五大维度,以及施工过程管理的进度、质量、安全三大体系为基础应用,实现全面高效的工程管理需求,满足工地多角色、多视角的有效监管,实现工程建设管理的降本增效,为监管平台提供数据支撑。
40 3
|
16天前
|
人工智能 移动开发 安全
家政上门系统用户端、阿姨端源码,java家政管理平台源码
家政上门系统基于互联网技术,整合大数据分析、AI算法和现代通信技术,提供便捷高效的家政服务。涵盖保洁、月嫂、烹饪等多元化服务,支持多终端访问,具备智能匹配、在线支付、订单管理等功能,确保服务透明、安全,适用于家庭生活的各种需求场景,推动家政市场规范化发展。
|
1月前
|
运维 自然语言处理 供应链
Java云HIS医院管理系统源码 病案管理、医保业务、门诊、住院、电子病历编辑器
通过门诊的申请,或者直接住院登记,通过”护士工作站“分配患者,完成后,进入医生患者列表,医生对应开具”长期医嘱“和”临时医嘱“,并在电子病历中,记录病情。病人出院时,停止长期医嘱,开具出院医嘱。进入出院审核,审核医嘱与住院通过后,病人结清缴费,完成出院。
86 3
|
1月前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。