如何从用户输入中获取某些值并忽略其他值

How to get certain values from user input and ignore others

提问人: 提问时间:4/23/2019 更新时间:4/24/2019 访问量:65

问:

有没有办法只从这个字符串输入中获取数字:

h8 f7

如果尝试 fscanf.但是,可能我用错了。

int positionOfTheKnight = 0, finalDestination = 0;

fscanf("%*s%d %*s%d", &positionOfTheKnight, &finalDestination);

cout << positionOfTheKnight << " " << finalDestination;

它显示以下错误:

cpp|11|error: cannot convert 'const char*' to 'FILE* {aka _iobuf*}' for argument '1' to 'int fscanf(FILE*, const char*, ...)'|
C++ 输入 IOSTREAM

评论

1赞 Achal 4/23/2019
编译器错误消息是正确的。 第一个参数是错误的,它应该是类型。fscanf()FILE*
0赞 4/23/2019
我真的很感谢你的话。但是,有什么方法可以获得我想要的输入吗?有或没有 FScanf 的帮助?我是 c++ 的初学者,请原谅。
1赞 Scheff's Cat 4/23/2019
如果你想将 fscanf() 与标准输入通道一起使用,你的第一个参数必须是 。或者,您可以使用从标准输入中读取的内容。顺便说一句,它被标记为 c++:为什么不是 iostreamstdinscanf()
0赞 4/23/2019
Scheff,我已经更新了标签。你能给我举个例子,说明如何在 scanf 的帮助下获取这些输入吗?
0赞 Scheff's Cat 4/24/2019
如果您想解决某人的评论,请以 .这将生成该人的通知。@

答:

2赞 Scheff's Cat 4/24/2019 #1

我做了一个小样本来演示阅读 OP 示例的多种方法:

#include <cstdio>
#include <iostream>
#include <string>

int rcToInt(char c, char r)
{
  if (c < 'a' || c > 'h' || r < '1' || r > '8') {
    std::cerr << "Wrong input!\n";
    return -1;
  }
  return (c - 'a') * 8 + (r - '1'); // convert characters to index
}

int main()
{
  { // using scanf
    std::cout << "Using scanf():\n";
    char r0, c0, r1, c1;
    if (scanf(" %c%c %c%c ", &c0, &r0, &c1, &r1) < 4) {
      std::cerr << "Reading move with scanf() failed!\n";
    } else {
      const int pos0 = rcToInt(c0, r0), pos1 = rcToInt(c1, r1);
      if (pos0 >= 0 && pos1 >= 0) {
        std::cout << "Got " << c0 << r0 << " -> " << c1 << r1 << '\n';
      }
    }
  }
  { // Yet another way to use scanf():
    std::cout << "Using scanf():\n";
    char c0[2], r0[2], c1[2], r1[2];
    if (scanf("%1[a-h]%1[1-8] %1[a-h]%1[1-8] ", c0, r0, c1, r1) < 4) {
      std::cerr << "Reading move with scanf() failed!\n";
    } else {
      const int pos0 = rcToInt(*c0, *r0), pos1 = rcToInt(*c1, *r1);
      if (pos0 >= 0 && pos1 >= 0) {
        std::cout << "Got " << c0 << r0 << " -> " << c1 << r1 << '\n';
      }
    }
  }
  { // using std::cin
    std::cout << "Using operator>>():\n";
    char r0, c0, r1, c1;
    if (!(std::cin >> c0 >> r0 >> c1 >> r1)) {
      std::cerr << "Reading move with operator>>() failed!\n";
    } else {
      const int pos0 = rcToInt(c0, r0), pos1 = rcToInt(c1, r1);
      if (pos0 >= 0 && pos1 >= 0) {
        std::cout << "Got " << c0 << r0 << " -> " << c1 << r1 << '\n';
      }
    }
  }
  // There is still the [ENTER] in input queue which must be consumed:
  std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
  // ...before proceeding
  { // using std::cin
    std::cout << "Using std::getline():\n";
    std::string line;
    if (!std::getline(std::cin, line) || line.size() < 5) {
      std::cerr << "Reading move with std::getline() failed!\n";
    } else {
      const char c0 = line[0], r0 = line[1], c1 = line[3], r1 = line[4];
      const int pos0 = rcToInt(c0, r0), pos1 = rcToInt(c1, r1);
      if (pos0 >= 0 && pos1 >= 0) {
        std::cout << "Got " << c0 << r0 << " -> " << c1 << r1 << '\n';
      }
    }
  }
  // done
  return 0;
}

输出:

Using scanf():
h8 f7
Got h8 -> f7
Using scanf():
f7 h8
Got f7 -> h8
Using operator>>():
h8 f7
Got h8 -> f7
Using std::getline():
f7 h8
Got f7 -> h8

Live Demo on coliru

所有四个备选方案都遵循基本思想,将输入读取为字符,然后将它们转换为 [0, 63] 范围内的相应索引。

虽然可能表示字符,但它也是一个整数值 - 特别是最小的可用整数类型。在(C 和)C++ 中,这没有区别。字符存储为整数值(假定为 ASCII 代码))。 以及表示相同的值。因此,如果合适,可以使用字符常量进行算术计算(如上面的例子所示)。char'1'33'1'33rcToInt()

  1. if (scanf(" %c%c %c%c ", &c0, &r0, &c1, &r1) < 4) {

    • 格式化程序以空格 () 开头,以使用可选的待处理空格。
    • %c存储一个字符。因此,必须提供一个参数(指向足够的存储空间)。char*
    • 格式化程序以空格 () 结尾,以使用终止符。ENTER
    • 选中 的返回值以授予将分配所有 4 个参数。scanf()
  2. if (scanf(" %1[a-h]%1[1-8] %1[a-h]%1[1-8] ", c0, r0, c1, r1) < 4) {

    • 格式化程序以空格 () 开头和结尾,以使用周围的空格(如上例所示)。
    • %1[a-h]读取长度为 1 的字符串,该字符串只能由字符 、 、 ...组成。必须为额外的字节提供存储,因为该格式化程序将始终存储额外的 0 终止符 ()。因此,在这种情况下,declations。abh\0char c0[2], r0[2], c1[2], r1[2];
    • %1[1-8]与上面的字符类似,,...,。请注意,数字被读取为字符。128
  3. if (!(std::cin >> c0 >> r0 >> c1 >> r1)) {

    • 使用流输入运算符读取四个变量。
      标头为此提供了各种重载的 s(例如)。
      chariostreamoperator>>std::istream& operator>>(std::istream&, char&)
    • 空格(在读取之前出现)被默认设置跳过。c1std::cin
  4. if (!std::getline(std::cin, line) || line.size() < 5) {

    • 将一整行读成 a(适当增长)。std::string
    • 分析其中的字符。std::string

3.和 4.将读取带有 的行,将读取行包装在 a 中以使用输入流运算符解析它。std::getline()std::istringstream


对不起,如果我在上面的代码中交换了行和列。我不是国际象棋专家,也不知道通常的转换。