C++ 的cout.tellp()和cout.seekp()语法介绍

简介: 无论是使用 cout 输出普通数据,用 cout.put() 输出指定字符,还是用 cout.write() 输出指定字符串,数据都会先放到输出流缓冲区,待缓冲区刷新,数据才会输出到指定位置(屏幕或者文件中)。值得一提的是,当数据暂存于输出流缓冲区中时,我们仍可以对其进行修改。ostream 类中提供有 tellp() 和 seekp() 成员方法,借助它们就可以修改位于输出流缓冲区中的数据。C++ tellp()成员方法首先,tellp() 成员方法用于获取当前输出流缓冲区中最后一个字符所在的位置,其语法格式如下:streampos tellp();显然,tellp()

无论是使用 cout 输出普通数据,用 cout.put() 输出指定字符,还是用 cout.write() 输出指定字符串,数据都会先放到输出流缓冲区,待缓冲区刷新,数据才会输出到指定位置(屏幕或者文件中)。

值得一提的是,当数据暂存于输出流缓冲区中时,我们仍可以对其进行修改。ostream 类中提供有 tellp() seekp() 成员方法,借助它们就可以修改位于输出流缓冲区中的数据。

C++ tellp()成员方法

首先,tellp() 成员方法用于获取当前输出流缓冲区中最后一个字符所在的位置,其语法格式如下:

streampos tellp();

显然,tellp() 不需要传递任何参数,会返回一个 streampos 类型值。事实上,streampos 是 fpos 类型的别名,而 fpos 通过自动类型转换,可以直接赋值给一个整形变量(即 short、int 和 long)。也就是说,在使用此函数时,我们可以用一个整形变量来接收该函数的返回值。

注意,当输出流缓冲区中没有任何数据时,该函数返回的整形值为 0;当指定的输出流缓冲区不支持此操作,或者操作失败时,该函数返回的整形值为 -1。

在下面的样例中,实现了借助 cout.put() 方法向 test.txt 文件中写入指定字符,由于此过程中字符会先存入输出流缓冲区,所以借助 tellp() 方法,我们可以实时监控新存入缓冲区中字符的位置。

举个例子:

#include <iostream> //cin 和 cout
    #include <fstream> //文件输入输出流
    int main() {
        //定义一个文件输出流对象
        std::ofstream outfile;
        //打开 test.txt,等待接收数据
        outfile.open("test.txt");
        const char * str = "http://c.biancheng.net/cplus/";
        //将 str 字符串中的字符逐个输出到 test.txt 文件中,每个字符都会暂时存在输出流缓冲区中
        for (int i = 0; i < strlen(str); i++) {
            outfile.put(str[i]);
            //获取当前输出流
            long pos = outfile.tellp();
            std::cout << pos << std::endl;
        }
        //关闭文件之前,刷新 outfile 输出流缓冲区,使所有字符由缓冲区流入test.txt文件
        outfile.close();
        return 0;
    }

注意,此例中涉及到了文件操作的相关知识,初学者仅需借助注释了解程序的执行脉络即可,不需要研究具体实现细节。有关文件操作,后续章节会做详细讲解。

读者可自行运行此程序,其输出结果为 1~29。这意味着,程序中每次向输出流缓冲区中放入字符时,pos 都表示的是当前字符的位置。比如,当将 str 全部放入缓冲区中时,pos 值为 29,表示的是最后一个字符 '/' 位于第 29 个位置处。

C++ seekp()成员方法

seekp() 方法用于指定下一个进入输出缓冲区的字符所在的位置。

举个例子,假设当前输出缓冲区中存有如下数据:

http://c.biancheng.net/cplus/

借助 tellp() 方法得知,最后一个 '/' 字符所在的位置是 29。此时如果继续向缓冲区中存入数据,则下一个字符所在的位置应该是 30,但借助 seekp() 方法,我们可以手动指定下一个字符存放的位置。

比如通过 seekp() 指定下一个字符所在的位置为 23,即对应 "cplus/" 部分中字符 'c' 所在的位置。此时若再向缓冲区中存入 "python/",则缓冲区中存储的数据就变成了:

http://c.biancheng.net/python/

显然,新的 "python/" 覆盖了原来的 "cplus/"。seekp() 方法有如下 2 种语法格式:

//指定下一个字符存储的位置 ostream& seekp (streampos pos); //通过偏移量间接指定下一个字符的存储位置    ostream& seekp (streamoff off, ios_base::seekdir way);

其中,各个参数的含义如下:

  • pos:用于接收一个正整数;、
  • off:用于指定相对于 way 位置的偏移量,其本质也是接收一个整数,可以是正数(代表正偏移)或者负数(代表负偏移);
  • way:用于指定偏移位置,即从哪里计算偏移量,它可以接收表 1 所示的 3 个值。

同时,seekp() 方法会返回一个引用形式的 ostream 类对象,这意味着 seekp() 方法可以这样使用:

  • cout.seekp(23) << "当前位置为:" << cout.tellp();
#include <iostream> //cin 和 cout
    #include <fstream> //文件输入输出流
    using namespace std;
    int main() {
        //定义一个文件输出流对象
        ofstream outfile;
        //打开 test.txt,等待接收数据
        outfile.open("test.txt");
        const char * str = "http://c.biancheng.net/cplus/";
        //将 str 字符串中的字符逐个输出到 test.txt 文件中,每个字符都会暂时存在输出流缓冲区中
        for (int i = 0; i < strlen(str); i++) {
            outfile.put(str[i]);
            //获取当前输出流
        }
        cout << "当前位置为:" << outfile.tellp() << endl;
        //调整新进入缓冲区字符的存储位置
        outfile.seekp(23);  //等价于:
                            //outfile.seekp(23, ios::beg);
                            //outfile.seekp(-6, ios::cur);
                            //outfile.seekp(-6, ios::end);
        cout << "新插入位置为:" << outfile.tellp() << endl;
        const char* newstr = "python/";
        outfile.write("python/", 7);
        //关闭文件之前,刷新 outfile 输出流缓冲区,使所有字符由缓冲区流入test.txt文件
        outfile.close();
        return 0;
    }
相关文章
|
2月前
|
Java C# C++
C++ 11新特性之语法甜点1
C++ 11新特性之语法甜点1
33 4
|
2月前
|
编译器 C++ 容器
C++ 11新特性之语法甜点2
C++ 11新特性之语法甜点2
30 1
|
2月前
|
存储 算法 编译器
C++ 11新特性之语法甜点4
C++ 11新特性之语法甜点4
28 0
|
2月前
|
安全 C++ 容器
C++ 11新特性之语法甜点3
C++ 11新特性之语法甜点3
36 0
|
3月前
|
编译器 C++ 容器
C++语言的基本语法
想掌握一门编程语言,第一步就是需要熟悉基本的环境,然后就是最重要的语法知识。 C++ 程序可以定义为对象的集合,这些对象通过调用彼此的方法进行交互。现在让我们简要地看一下什么是类、对象,方法、即时变量。 对象 - 对象具有状态和行为。例如:一只狗的状态 - 颜色、名称、品种,行为 - 摇动、叫唤、吃。对象是类的实例。 类 - 类可以定义为描述对象行为/状态的模板/蓝图。 方法 - 从基本上说,一个方法表示一种行为。一个类可以包含多个方法。可以在方法中写入逻辑、操作数据以及执行所有的动作。 即时变量 - 每个对象都有其独特的即时变量。对象的状态是由这些即时变量的值创建的。 完整关键字
|
4月前
|
Java 编译器 程序员
C++中的语法知识虚继承和虚基类
**C++中的多继承可能导致命名冲突和数据冗余,尤其在菱形继承中。为解决这一问题,C++引入了虚继承(virtual inheritance),确保派生类只保留虚基类的一份实例,消除二义性。虚继承通过`virtual`关键字指定,允许明确访问特定路径上的成员,如`B::m_a`或`C::m_a`。这样,即使基类在继承链中多次出现,也只有一份成员副本,简化了内存布局并避免冲突。虚继承应在需要时提前在继承声明中指定,影响到从虚基类派生的所有后代类。**
|
4月前
|
编译器 C++ 开发者
C++一分钟之-属性(attributes)与属性语法
【7月更文挑战第3天】C++的属性(attributes)自C++11起允许附加编译器指令,如`[[nodiscard]]`和`[[maybe_unused]]`,影响优化和警告。注意属性放置、兼容性和适度使用,以确保代码清晰和可移植。示例展示了如何使用属性来提示编译器处理返回值和未使用变量,以及利用编译器扩展进行自动清理。属性是提升代码质量的工具,但应谨慎使用。
145 13
|
5月前
|
编译器 程序员 C++
C++一分钟之-属性(attributed)与属性语法
【6月更文挑战第28天】C++的属性为代码添加元数据,帮助编译器理解意图。C++11引入属性语法`[[attribute]]`,但支持取决于编译器。常见属性如`nodiscard`提示检查返回值,`maybe_unused`防止未使用警告。问题包括兼容性、过度依赖和误用。使用属性时需谨慎,确保团队共识,适时更新以适应C++新特性。通过示例展示了`nodiscard`和`likely/unlikely`的用法,强调正确使用属性能提升代码质量和性能。
89 13
|
5月前
|
编译器 C语言 C++