问题引入

做Java作业从标准输入流获取用户输入,用到了System.in.read(),然后出现了bug。

//随机生成一个小写字母,用户猜5次,读取用户输入,并判断是否猜对
import java.io.IOException;
public class LetterGuessing {
public static void main(String[] args) throws IOException {
char ch,answer;
//随机生成字母
answer=(char)(Math.random()*26+'a');
System.out.print("请输入一个小写字母:");
for(int i=1;i<=5;i++) {
//获取用户输入,可能抛出异常
ch=(char)System.in.read();
//比较大小
if(ch == answer){
System.out.println("恭喜,正确!用了"+i+"次猜对");
break;
}
else if(ch > answer)
System.out.println("您猜大了,还有"+ (5-i) +"次机会");
else
System.out.println("您猜小了,还有"+ (5-i) +"次机会");
}
}
}

输入字符'a',按下Enter,却没有等我下次输入,循环就运行了三次。

问题来源

System.in.read()按字节读,一次读入一个字节。后边有详细讲解。

经调试,可知三次循环中ch分别为a,\r,\n

为什么a+Enter,会变成a\r\n呢

Windows下存在两种文件读写方式,一个是二进制方式,另一种是文本方式

文本方式中写时“换行”会变成“回车-换行”,即\r\n;读时“回车-换行”会变成“换行”。

二进制方式中读写是严格按照一个字节一个字节的方式进行的。

在这里虽然没有用到文件,但道理应该是一样的

read()函数是按照一个字节一个字节读取的,即二进制方式。

可能可以推导出,我们向输入流中输入数据默认是按照文本方式。

解决方法

方法一

在代码第10行后,加两行System.in.read();

目的是读取掉输入流中的/r和/n。

这种方法的局限性就是输入字母前后不能加空格,因为它不会使空格从输入流中删除。

方法二

不用read()读取,用以下代码代替

import java.util.Scanner;
Scanner input=new Scanner(System.in);
ch=input.next().charAt(0);

这种方法就比较好,读取字符串(忽略空格和换行,空格和换行不会留在输入流里),然后取字符串的第一个字符。

知识点

System.in

官方文档:https://docs.oracle.com/javase/10/docs/api/java/lang/System.html#in

System是个类,in是System的一个成员,官方介绍如下:

public static final InputStream in

The “standard” input stream. This stream is already open and ready to supply input data. Typically this stream corresponds to keyboard input or another input source specified by the host environment or user.

in是一个InputStream类型的对象,所以只需要了解InputStream即可。

InputStream

官方文档: https://docs.oracle.com/javase/10/docs/api/java/io/InputStream.html

public abstract class InputStream	//抽象类
extends Object //继承Object类
implements Closeable //实现Closeable接口

官方介绍如下:

This abstract class is the superclass of all classes representing an input stream of bytes.

翻译为:这个抽象类是所有字节流类的父类。

字节流的含义:读取方式为一个字节一个字节地读取,而字符流是二个字节二个字节的读。

Applications that need to define a subclass of InputStream must always provide a method that returns the next byte of input.

翻译为:需要定义一个InputStream子类的应用必须提供一个返回输入下一字节的方法(函数)。

read()

官方文档:https://docs.oracle.com/javase/10/docs/api/java/io/InputStream.html#read()

public abstract int read()
throws IOException

Reads the next byte of data from the input stream. The value byte is returned as an int in the range 0 to255.

这个是重点,它返回下一字节的ASCII码

If no byte is available because the end of the stream has been reached, the value -1 is returned. This method blocks until input data is available, the end of the stream is detected, or an exception is thrown.

A subclass must provide an implementation of this method.

Returns:

the next byte of data, or -1 if the end of the stream is reached.

Throws:

IOException - if an I/O error occurs.

作者:@臭咸鱼

本文为作者原创,转载请注明出处:https://chouxianyu.github.io/2018/09/22/Java字节流read函数/#more

欢迎转发和评论

Java字节流read函数的更多相关文章

  1. Java字节流实现文件夹的拷贝

    import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io ...

  2. java String.split()函数的用法分析

    java String.split()函数的用法分析 栏目:Java基础 作者:admin 日期:2015-04-06 评论:0 点击: 3,195 次 在java.lang包中有String.spl ...

  3. 使用Java字节流拷贝文件

    本文给出使用Java字节流实现文件拷贝的例子 package LearnJava; import java.io.*; public class FileTest { public static vo ...

  4. java 字节流和字符流的区别 转载

    转载自:http://blog.csdn.net/cynhafa/article/details/6882061 java 字节流和字符流的区别 字节流与和字符流的使用非常相似,两者除了操作代码上的不 ...

  5. java 字节流和字符流的区别

    转载自:http://blog.csdn.net/cynhafa/article/details/6882061 java 字节流和字符流的区别 字节流与和字符流的使用非常相似,两者除了操作代码上的不 ...

  6. C# 二进制字节流查找函数IndexOf

    C# 二进制字节流查找函数IndexOf /// <summary> /// 报告指定的 System.Byte[] 在此实例中的第一个匹配项的索引. /// </summary&g ...

  7. Android使用JNI(从java调用本地函数)

    当编写一个混合有本地C代码和Java的应用程序时,需要使用Java本地接口(JNI)作为连接桥梁.JNI作为一个软件层和API,允许使用本地代码调用Java对象的方法,同时也允许在Java方法中调用本 ...

  8. Java 字节流实现文件读写操作(InputStream-OutputStream)

    Java 字节流实现文件读写操作(InputStream-OutputStream) 备注:字节流比字符流底层,但是效率底下. 字符流地址:http://pengyan5945.iteye.com/b ...

  9. 关于C++与Java中虚函数问题的读书笔记

    之前一直用C++编程,对虚函数还是一些较为肤浅的理解.可近期由于某些原因搞了下Java,发现有些知识点不熟,于是站在先驱巨人的肩上谈谈C++与Java中虚函数问题. Java中的虚函数 以下是段别人的 ...

随机推荐

  1. AssassinGo: 基于Go的高并发可拓展式Web渗透框架

    转载自FreeBuf.COM AssassinGo是一款使用Golang开发,集成了信息收集.基础攻击探测.Google-Hacking域名搜索和PoC批量检测等功能的Web渗透框架,并且有着基于Vu ...

  2. ssh自动添加hostkey到know_hosts

    The authenticity of host ‘git.sws.com (10.42.1.88)’ can’t be established. ECDSA key fingerprint is 5 ...

  3. Git速成学习第三课:创建与合并分支

    本来第三课想记录一下远程仓库的创建与克隆0.0但是想了想还是不写了. 这里写一下分支管理中的创建与合并. Git速成学习笔记整理于廖雪峰老师的官网网站:https://www.liaoxuefeng. ...

  4. Had I not seen the Sun(如果我不曾见过太阳)

    Had I not seen the Sun by Emily Dickinson Had I not seen the Sun I could have borne the shade But Li ...

  5. CSharp实体生成器

    专门为C#开发人员定制的一个实体生成器,以往的生成器功能达到了,但是很多细节未考虑到,所以我借鉴“先人”的一些已有的东西,重新定制了一个.当然,需要源码的同学,直接使用IL Spy这个小工具就可以看到 ...

  6. 简单深入Joomla!3.1.5模块_组件开发(一)

    简单深入Joomla!3.1.5模块_组件开发 主要内容: 1, 模块(访问数据库,链接到组件,数据基本流向) 2, 组件CRUD(MVC模式,访问数据库,表单提交,AJAX提交,数据基本流向) 3, ...

  7. mysql常用引擎

    经常用MySQL数据库,但是,你在用的时候注意过没有,数据库的存储引擎,可能有注意但是并不清楚什么意思,可能根本没注意过这个问题,使用了默认的数据库引擎,当然我之前属于后者,后来成了前者,然后就有了这 ...

  8. Nginx 小入门记录 之 Nginx 配置文件解读(二)

    上一小节主要是记录一些环境准备和Nginx的安装,接下来对Nginx基本配置进行记录. 查看配置文件安装记录 可以通过以下Linux命令进行查看: rpm -ql nginx rpm 是liunx的包 ...

  9. oauth2中org.springframework.security.core.userdetails.User无法转换为封装的AuthorizationInfoBean

    用springboot + oauth2 + redis搭建了一个项目,创建一个自定义的AuthorizationInfoBean继承org.springframework.security.core ...

  10. luogu P1734 最大约数和 (01 背包)

    链接:https://www.luogu.org/problemnew/show/P1734 题面: 题目描述 选取和不超过S的若干个不同的正整数,使得所有数的约数(不含它本身)之和最大. 输入输出格 ...