问题引入

做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. BFS算法模板(python实现)

    BFS算法整理(python实现) 广度优先算法(Breadth-First-Search),简称BFS,是一种图形搜索演算算法. 1. 算法的应用场景 2. 算法的模板 2.1 针对树的BFS模板 ...

  2. Velocity 数值格式化(NumberTool工具类):保留两位小数和格式化千分位、取整

    Velocity 自带的工具类:NumberTool 实现数字格式化:保留两位小数和格式化千分位,以及取整. NumberTool 的 format(String format, Object obj ...

  3. eNSP——ARP及ProxyARP

    原理: ARP (Address Resolution Protocol)是用来将IP地址解析为MAC地址的协议.ARP表项可以分为动态和静态两种类型.动态ARP是利用ARP广播报文,动态执行并自动进 ...

  4. golang的定时器NewTimer、NewTicker使用

    package main import ( "fmt" "sync" "time" ) /** *ticker只要定义完成,从此刻开始计时, ...

  5. Linux下配置APUE的编译 报错之后如何处理

    APUE即Unix环境高级编程,本书中几乎所有的程序都包含一个apue.h的头文件,那如何配置这个apue.h呢? 官方地址:http://www.apuebook.com/apue3e.html 1 ...

  6. js函数(2)

    8.3函数的形参和实参 js中的函数并未指定函数形参的类型,函数调用也未对传入的实参值做任何类型的检查. 8.3.1函数的形参和实参 当调用函数时传入的实参比函数声明时指定的形参个数要少,剩下的参数都 ...

  7. redis快速开始

    1 下载地址:http://redis.io/download 2 安装步骤: 3 # 安装gcc 4 yum install gcc 5 6 # 把下载好的redis‐5.0.3.tar.gz放在/ ...

  8. Elastic Search快速上手(3):搜索

    前言 存储好数据之后,便可通过RESTful API进行搜索. 详细文档可参考: --简单搜索https://www.elastic.co/guide/cn/elasticsearch/guide/c ...

  9. 怎样设置HTTP请求头Header

    使用: xhr.setRequestHeader(); xhr.setRequestHeader('Content-Type', 'application/json'); xhr.setRequest ...

  10. C#学习基础资料记录---字典(Dictionary),时间表示方法(DateTime.Now),文件操作

    1.字典 https://www.cnblogs.com/gengaixue/p/4002244.html 2.时间的表示方法 DateTime.Now的多种用法 https://www.cnblog ...