问题引入

做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. logging配置

    import logging def handle(): # 1.定义logger对象:负责产生日志,然后交给Filter过滤,然后交给不同的Handler输出 logger = logging.ge ...

  2. CF1190D Tokitsukaze and Strange Rectangle

    思路: 线段树 + 扫描线. 实现: #include <bits/stdc++.h> using namespace std; typedef long long ll; ; int n ...

  3. 深入理解JS的事件绑定、事件流模型

     一.JS事件 (一)JS事件分类 1.鼠标事件: click/dbclick/mouseover/mouseout 2.HTML事件: onload/onunload/onsubmit/onresi ...

  4. ARC083E. Bichrome Tree

    A viable configuration of the given tree can be divided into two trees, each consists of vertices of ...

  5. 【AtCoder】ARC065

    ARC065 C - 白昼夢 / Daydream 直接递推就好 #include <bits/stdc++.h> #define fi first #define se second # ...

  6. SpringBoot 对IBM MQ进行数据监听接收以及数据发送

    一.需求介绍 后端使用Spring Boot2.0框架,要实现IBM MQ的实时数据JMS监听接收处理,并形成回执通过MQ队列发送. 二.引入依赖jar包 <dependency> < ...

  7. 剑指offer14:输入一个链表,输出该链表中倒数第k个结点。

    1. 题目描述 输入一个链表,输出该链表中倒数第k个结点. 2. 思路和方法 可以用两个指针,一个指针遍历到第k个结点的时候,第二个指针再走到第一个节点,然后两个指针的距离始终保持k-1.这样,当第一 ...

  8. Java考题知识点

    挑战10个最难回答的Java面试题(附答案) - 里奥ii的文章 - 知乎 https://zhuanlan.zhihu.com/p/79186037 1.java的基本编程单元是类,基本存储单元是变 ...

  9. PHP和js判断访问设备是否是微信浏览器实例

    PHP和js判断访问设备是否是微信浏览器实例,代码非常精简,适合新手学习. js判断是否是微信浏览器: 1 function is_weixin() { 2 var ua = window.navig ...

  10. 后端排序,debug模式中map的顺序出错

    js中map遍历的顺序是按照插入的顺序来执行的.如果map的来源是字符串转换的,那么就会按照字符串中key值的顺序进行遍历.千万不要被debug中显示的顺序误导,这里应该是为了方便查看对key进行了字 ...