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

本文涉及的产品
RDS DuckDB + QuickBI 企业套餐,8核32GB + QuickBI 专业版
简介: 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

目录
相关文章
|
9月前
|
存储 小程序 Java
热门小程序源码合集:微信抖音小程序源码支持PHP/Java/uni-app完整项目实践指南
小程序已成为企业获客与开发者创业的重要载体。本文详解PHP、Java、uni-app三大技术栈在电商、工具、服务类小程序中的源码应用,提供从开发到部署的全流程指南,并分享选型避坑与商业化落地策略,助力开发者高效构建稳定可扩展项目。
|
9月前
|
消息中间件 人工智能 Java
抖音微信爆款小游戏大全:免费休闲/竞技/益智/PHP+Java全筏开源开发
本文基于2025年最新行业数据,深入解析抖音/微信爆款小游戏的开发逻辑,重点讲解PHP+Java双引擎架构实战,涵盖技术选型、架构设计、性能优化与开源生态,提供完整开源工具链,助力开发者从理论到落地打造高留存、高并发的小游戏产品。
|
存储 安全 Java
Java 集合面试题从数据结构到 HashMap 源码剖析详解及长尾考点梳理
本文深入解析Java集合框架,涵盖基础概念、常见集合类型及HashMap的底层数据结构与源码实现。从Collection、Map到Iterator接口,逐一剖析其特性与应用场景。重点解读HashMap在JDK1.7与1.8中的数据结构演变,包括数组+链表+红黑树优化,以及put方法和扩容机制的实现细节。结合订单管理与用户权限管理等实际案例,展示集合框架的应用价值,助你全面掌握相关知识,轻松应对面试与开发需求。
557 3
|
JavaScript Java 关系型数据库
家政系统源码,java版本
这是一款基于SpringBoot后端框架、MySQL数据库及Uniapp移动端开发的家政预约上门服务系统。
431 6
家政系统源码,java版本
|
供应链 JavaScript 前端开发
Java基于SaaS模式多租户ERP系统源码
ERP,全称 Enterprise Resource Planning 即企业资源计划。是一种集成化的管理软件系统,它通过信息技术手段,将企业的各个业务流程和资源管理进行整合,以提高企业的运营效率和管理水平,它是一种先进的企业管理理念和信息化管理系统。 适用于小微企业的 SaaS模式多租户ERP管理系统, 采用最新的技术栈开发, 让企业简单上云。专注于小微企业的应用需求,如企业基本的进销存、询价,报价, 采购、销售、MRP生产制造、品质管理、仓库库存管理、财务应收付款, OA办公单据、CRM等。
933 23
|
前端开发 Java 关系型数据库
基于Java+Springboot+Vue开发的鲜花商城管理系统源码+运行
基于Java+Springboot+Vue开发的鲜花商城管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的鲜花商城管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。技术学习共同进步
835 7
|
Java 关系型数据库 MySQL
Java汽车租赁系统源码(含数据库脚本)
Java汽车租赁系统源码(含数据库脚本)
662 4
|
消息中间件 算法 安全
JUC并发—1.Java集合包底层源码剖析
本文主要对JDK中的集合包源码进行了剖析。
|
Java
【源码】【Java并发】【ConcurrentHashMap】适合中学体质的ConcurrentHashMap
本文深入解析了ConcurrentHashMap的实现原理,涵盖JDK 7与JDK 8的区别、静态代码块、构造方法、put/get/remove核心方法等。JDK 8通过Node数组+链表/红黑树结构优化并发性能,采用CAS和synchronized实现高效锁机制。文章还详细讲解了hash计算、表初始化、扩容协助及计数更新等关键环节,帮助读者全面掌握ConcurrentHashMap的工作机制。
371 6
【源码】【Java并发】【ConcurrentHashMap】适合中学体质的ConcurrentHashMap
|
人工智能 安全 Java
智慧工地源码,Java语言开发,微服务架构,支持分布式和集群部署,多端覆盖
智慧工地是“互联网+建筑工地”的创新模式,基于物联网、移动互联网、BIM、大数据、人工智能等技术,实现对施工现场人员、设备、材料、安全等环节的智能化管理。其解决方案涵盖数据大屏、移动APP和PC管理端,采用高性能Java微服务架构,支持分布式与集群部署,结合Redis、消息队列等技术确保系统稳定高效。通过大数据驱动决策、物联网实时监测预警及AI智能视频监控,消除数据孤岛,提升项目可控性与安全性。智慧工地提供专家级远程管理服务,助力施工质量和安全管理升级,同时依托可扩展平台、多端应用和丰富设备接口,满足多样化需求,推动建筑行业数字化转型。
446 5

热门文章

最新文章