cin详解(get()、getline()、clear()、sync())

简介: 简述在C中,输入输出用scanf和printf,在输入数据的同时还需说明数据的类型,如果输入数据较多,那就很麻烦,而C++中也有相似的东西cin和cout,它们来自C++的一个名叫” iostream”的类库。iostream是由istream(输入流)和ostream(输出流)派生。所以在iostream中就有了输入和输出的相关对象:cin:标准输入(stan

简述

在C中,输入输出用scanf和printf,在输入数据的同时还需说明数据的类型,如果输入数据较多,那就很麻烦,而C++中也有相似的东西cin和cout,它们来自C++的一个名叫” iostream”的类库。

iostream是由istream(输入流)和ostream(输出流)派生。所以在iostream中就有了输入和输出的相关对象:

  • cin:标准输入(standard input)的istream类对象,cin使我们可以从设备读取数据。
  • cout:标准输出(standard output)的ostream类对象。对应于标准输出流,默认情况下是显示器。这是一个被缓冲的输出,可以被重定向。
  • cerr:标准错误流,用于显示错误消息。默认情况下被关联到标准输出流,但它不被缓冲,也就说错误消息可以直接发送到显示器,而无需等到缓冲区或者新的换行符时,才被显示。一般情况下不被重定向。

cout与cerr的区别:cout的输出可以重定向到一个文件中,而cerr必须输出在显示器上。

暂时先介绍这些,以下主要介绍cin中get()、getline()、clear()、sync()的用法。

get()

首先看看get(),它是一个读取单个字符的方法。字符变量 = cin.get(),相当于cin.get(字符变量)。

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    char str;
    str = cin.get();      //读取单个字符,在屏幕输入,相当于cin.get(str);
    cout << str << endl;  //输出刚刚载入的单个字符
    system("pause");      //进行暂停,否则会一闪而过

    return 0;
}

运行程序后:

输入:a
输出:a

但当输入的为多个英文字符时,那又会如何呢?

输入:abcd
输出:a

结论:get()只能读取第一个字符。

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    char str1;
    char str2;
    str1 = cin.get();  //读取单个字符,在屏幕输入
    str2 = cin.get();
    cout << str1 << str2 << endl;  //输出刚刚载入的单个字符
    system("pause");  //进行暂停,否则会一闪而过

    return 0;
}

运行程序后:

输入:abcd
输出:ab

既然get()是读取第一个字符,那str2为什么不也是a呢?

原理如下:

在cin这个对象里,有一个储存字符的流,可以想象成缓冲区,事实上是cin里封装的一个东西。当我们在程序上输入字符后,对象cin获得了我们输入的字符。例如获得abcd,然后再通过.get()把流里面的第一个字符去掉,赋给str1,这时,cin里储存的流的数据为bcd,而str1则获得了a。当我们再次运行str2 = cin.get()时,同理把cin里流的数据的b拿出来给了str2,此后,cin里面的流的数据为cd,而str2则为b,所以最后输出时,便能输出ab了。

还有个补充,究竟什么时候才输入数据呢?我们可以再通过上面的代码进行尝试,我们输入单个字母’a’,然后按回车,发现并没有输出数据,而是再等待一次输入数据,我们再输入字母’b’,按回车后便输出ab了。相信到这里,大家都应该明白了,因为当我们第一次输入a后,通过str1 = cin.get()使cin里的流没有数据,清空了。所以到第二次要再赋给str2值时,它找不到数据,要重新再输入数据。由此来看可以知道,当cin里的流数据清空时,便需要重新输入才能赋值。

而get()还有个用法:

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    char str1;
    char str2;
    str1 = cin.get();  //读取单个字符,在屏幕输入
    cin.get();
    str2 = cin.get();
    cout << str1 << str2 << endl;  //输出刚刚载入的单个字符
    system("pause");

    return 0;
}

运行程序后:

输入:abcd
输出:ac

程序中有3个get(),由此可知,当空回调get()时,get()便自动在cin中的流数据中删除一个字母,起了一个删除作用。

getline()

对get()有了一定了解之后,对getline()的学习就可以更快了,原理是一致的,但是getline()则是获取一整行文本。

原型:getline(char *line, int size, char=’/n’)

参数一:字符指针
参数二:字符长度
参数三:结束标识符。

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    char str[200];
    cin.getline(str, sizeof(str));  //第三个不输入,默认回车为结束标符
    cout << str << endl;  //输出
    system("pause");

    return 0;
}

这样,我们输入多个英文或数字,然后按回车,就会输出刚刚输出的东西了。

接下来,我们讨论第三个参数的作用。

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    char str[200];
    cin.getline(str, sizeof(str), 'X'); //以单个英文字母'X'作为终止标识符
    cout << str << endl; //输出
    system("pause");

    return 0;
}

当我们输入多个数字或者字母时,例如:

输入:abcdeX(回车) 输出:abcde
输入:aXbcde(回车) 输出:a
输入:Xabcde(回车) 输出:

这样X便成了终止符,如上可知:当遇到第一个结束符标志时,就结束,输出其前面的所有字符。其原理和get()一样。或许我们可以像get()那样尝试一下:

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    char str1[200];
    char str2[200];
    cin.getline(str1, sizeof(str1), 'X'); //以单个英文字母'X'作为终止标识符
    cin.getline(str2, sizeof(str2), 'Y'); //以单个英文字母'Y'作为终止标识符
    cout << "第一行是:" << str1 << endl; //输出
    cout << "第二行是:" << str2 << endl;
    system("pause");

    return 0;
}

输入:abcdXXXefghYYYigkl(回车)
输出: 第一行:abcd 第二行:XXefgh

如上可知,当遇到第一个结束符’X’结束输出abcd,之后遇到第一个结束符’Y’结束输出XXefgh。

clear()

接下来谈谈clear()的作用,第一次看到这东西,很多人以为就是清空cin里面的数据流,而实际上却与此相差甚远,首先看看以下代码:

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    int a;
    cin >> a;
    ios::iostate state = cin.rdstate();
    cout << state << endl;
    if (state == ios::goodbit)
    {
        cout << "输入数据的类型正确,无错误!" << endl;
    }
    else if (state == ios_base::failbit)
    {
        cout << "输入数据类型错误,非致命错误,可清除输入缓冲区挽回!" << endl;
    }
    system("pause");

    return 0;
}

我们定义要输入的变量是整型,但如果输入了英文字母或者汉字,那就会发生错误,cin里有个方法能检测这个错误,就是rdstate()。

当rdstate()返回0(即ios::goodbit)时表示无错误,可以继续输入或者操作,若返回2则发生非致命错误即ios::failbit,则不能继续输入或操作。而clear()则可以控制我们此时cin里对这个问题的一个标识。

语法:cin.clear(标识符)

标识符号为:

  • goodbit:无错误。
  • Eofbit:已到达文件尾。
  • failbit:非致命的输入/输出错误,可挽回。
  • badbit:致命的输入/输出错误,无法挽回。

若在输入输出类里,需要加ios::标识符号。

sync()

通过clear(),我们能确认它的内部标识符,如果输入错误则能重新输入。结合真正的清空数据流方法sync(),请看下例:

#include <iostream>

using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    int a;
    while (true)
    {
        cin >> a;
        if (!cin)  //条件可改写为cin.fail()
        {
            cout << "输入类型错误,请重新输入!" << endl;
            cin.clear(); //复为标志,将cin中的所有标志设置为有效状态
            cin.sync(); //清空流
        }
        else
        {
            cout << a << endl;
            break;
        }
    }
    system("pause");

    return 0;
}

上面的cin默认值为非0,当输入为非整形时,它的状态标识符改为fail(即:0),再用clear()让错误标识改回为非0,可以继续输入,再清空流数据继续输入。如果没有了clear(),则会进入死循环,其过程为我们输入了英文字母,它的状态标识便为fail,当运行到条件判断时,便总是回到错误的条件表示里,并且再也没有办法输入,因为错误的表示关闭了cin,所以会进入死循环。

可以分别注释掉clear()和sync()进行验证。

目录
相关文章
|
存储 关系型数据库 数据库
阿里云中小企业扶持权益介绍,助力企业轻松上云
在数字化浪潮的推动下,中小企业正面临着转型升级的重要关口。阿里云深知中小企业的挑战与机遇,特别推出了一系列中小企业扶持权益,旨在帮助企业以更低的成本、更高的效率拥抱云计算,开启智能时代创业的新范式。
阿里云中小企业扶持权益介绍,助力企业轻松上云
|
小程序 前端开发
解决小程序 scroll-view 里面的image有间距、小程序里面的图片之间有空隙的问题。
解决小程序 scroll-view 里面的image有间距、小程序里面的图片之间有空隙的问题。
|
9月前
|
存储 缓存 固态存储
阿里云服务器2核8G、4核16G、8核32G价格:按量、包年包月收费标准及活动价格参考
2核8G、8核32G、4核16G配置的阿里云服务器处理器与内存比为1:4,这种配比的云服务器一般适用于中小型数据库系统、缓存、搜索集群和企业办公类应用等通用型场景。目前2核8G配置按量收费最低收费标准为0.3375元/小时,4核16G配置的按量收费标准最低为0.675元/小时,8核32G配置的按量收费标准最低为1.35元/小时,云服务器实例规格和配置不同,收费标准与活动价格也不同,本文将为您介绍这三大配置的最新的收费标准、活动价格及选型策略,以供选择参考。
|
传感器 人工智能 搜索推荐
人机融合智能 | 可穿戴计算设备的多模态交互
本文介绍了可穿戴计算设备的多模态交互技术,阐述了以人为中心的设计目标与原则。内容涵盖设备的历史发展、特点及分类,并重点分析手指触控、手部动作、头部和眼睛动作等交互模态。同时探讨支持这些交互的传感器种类与原理,以及未来挑战。通过十个设计原则,强调自然高效、个性化、低认知负荷及隐私保护的重要性,为可穿戴技术的设计提供指导。
658 0
|
机器学习/深度学习 算法 计算机视觉
卷积神经网络(CNN)的工作原理深度解析
【6月更文挑战第14天】本文深度解析卷积神经网络(CNN)的工作原理。CNN由输入层、卷积层、激活函数、池化层、全连接层和输出层构成。卷积层通过滤波器提取特征,激活函数增加非线性,池化层降低维度。全连接层整合特征,输出层根据任务产生预测。CNN通过特征提取、整合、反向传播和优化进行学习。尽管存在计算量大、参数多等问题,但随着技术发展,CNN在计算机视觉领域的潜力将持续增长。
1584 3
|
Java 应用服务中间件 测试技术
深入探索Spring Boot Web应用源码及实战应用
【5月更文挑战第11天】本文将详细解析Spring Boot Web应用的源码架构,并通过一个实际案例,展示如何构建一个基于Spring Boot的Web应用。本文旨在帮助读者更好地理解Spring Boot的内部工作机制,以及如何利用这些机制优化自己的Web应用开发。
476 3
|
缓存 NoSQL Redis
使用Redis实现缓存穿透的解决方案
使用Redis实现缓存穿透的解决方案
|
缓存 安全 数据安全/隐私保护
【Docker专栏】深入理解Docker镜像的构建与推送
【5月更文挑战第7天】本文介绍了Docker镜像的核心作用及基础概念,包括镜像作为容器模板的特性。文章详细阐述了Dockerfile的编写,例如设置基础镜像、工作目录、安装依赖及定义启动命令。通过`docker build`命令构建镜像,并提示了优化构建过程的技巧。此外,还讲解了如何将镜像推送到远程仓库,包括选择仓库、认证、标签和推送镜像的步骤,以及镜像安全性的考虑,如扫描漏洞和遵循最小权限原则。本文旨在帮助读者掌握Docker镜像的构建与推送,以高效管理容器化应用。
803 61
【Docker专栏】深入理解Docker镜像的构建与推送
|
消息中间件 网络协议 C#
C#使用Socket实现分布式事件总线,不依赖第三方MQ
`CodeWF.EventBus.Socket` 是一个轻量级的、基于Socket的分布式事件总线系统,旨在简化分布式架构中的事件通信。它允许进程之间通过发布/订阅模式进行通信,无需依赖外部消息队列服务。
C#使用Socket实现分布式事件总线,不依赖第三方MQ
|
JavaScript Java 关系型数据库
美妆商城系统 SpringBoot + Vue 【毕业设计 资料 + 源码】
这篇文章介绍了一个使用SpringBoot + Vue + Mybatis + Mysql技术栈开发的美妆商城系统,包括系统功能划分、部分页面截图和前后端源码示例,并提供了GitHub上的源码链接。
美妆商城系统 SpringBoot + Vue 【毕业设计 资料 + 源码】