《C++程序设计原理与实践》笔记 第11章 定制输入/输出

news2025/1/17 6:01:57

在本章中,我们重点关注如何使第10章中介绍的通用iostream框架适配特定的需求和偏好。

11.1 规则性和不规则性

C++标准库的输入/输出部分——iostream库为文本的输入和输出提供了一个统一的、可扩展的框架。

到目前为止,我们将所有输入源视为等价的,有时这是不够的。例如,文件与其他输入源(例如网络连接)的区别是可以按单个字节寻址(而网络连接的字节是流式到达的)(类似于向量和迭代器的区别)。另外,我们假设对象的类型完全决定了其输入和输出格式,但这并不完全正确。例如,我们通常希望在输出浮点数时指定数字个数(精度)。本章会提出一些可以按需求定制输入/输出的方法。

作为程序员,我们更喜欢规则性(regularity):一致地处理所有对象、将所有输入源视为等价、对表示对象的方式强制一个单一的标准能够给出最干净、最简单、最可维护以及通常最高效的代码。然而,程序的存在是为了服务人类,而人类有很强的偏好性(不规则性,irregularity)。因此,作为程序员,我们必须争取在程序复杂性和满足用户偏好之间取得平衡。

11.2 输出格式化

11.2.1 整数输出

整数值可以输出为八进制(octal)、十进制(decimal)和十六进制(hexadecimal)。一个十六进制数恰好表示4个二进制位,2个十六进制数可用于表示一个字节。

可以指定(十进制)数1234以十进制、十六进制或八进制输出:

cout << dec << 1234 << " (decimal)\n"
        << hex << 1234 << "  (hexadecimal)\n"
        << oct << 1234 << " (octal)\n";

将输出

1234 (decimal)
4d2  (hexadecimal)
2322 (octal)

即十进制的1234 = 十六进制的4d2 = 八进制的2322。

其中dechexoct并不输出值,而是告诉输出流任何后续整数值应该以十进制/十六进制/八进制输出,如果未指定则默认为dec。这三个操作是持久的(persist)/“有粘性的(sticky)”,即对每个整数值的输出都生效,直到指定其他进制。像hexoct这种用于改变流的行为的术语称为操纵符(manipulator)。

使用不同进制输出整数

注:

  • dechexoct相当于C标准库printf%d%x%o,但C++的操纵符是持久的,而printf的格式说明符仅对一个参数生效。
  • 这三个操纵符实际上是三个函数,分别设置了输出流对应的格式标志位。经常用到的endl也是这样的函数,其作用就是向输出流写入一个字符'\n'。另外,接受函数指针参数的重载运算符<<就是以流本身为参数调用该函数。其定义大致等价于:
ostream& ostream::operator<<(ios_base& (*pf)(ios_base&)) {
    pf(*this);
    return *this;
}

ios_base& dec(ios_base& base) {
  base.setf(ios_base::dec, ios_base::basefield);
  return base;
}

ios_base& hex(ios_base& base) {
    base.setf(ios_base::hex, ios_base::basefield);
    return base;
}

ios_base& oct(ios_base& base) {
    base.setf(ios_base::oct, ios_base::basefield);
    return base;
}

ostream& endl(ostream& os) {
    os.put('\n');
    return os;
}

因此,以下语句是等价的:

cout << endl;
cout.operator<<(endl);
endl(cout);
cout.put('\n');
cout << hex;
cout.operator<<(hex);
hex(cout);
cout.setf(ios_base::hex, ios_base::basefield);
  • 输入/输出流使用位掩码类型ios_base::fmtflags来表示格式标志位,并通过flags()setf()unsetf()三个成员函数来设置或清除格式标志位。hexshowbaseboolalpha等操纵符就是通过调用setf()来设置对应标志位的辅助函数。

使用其他进制输出数值时默认不显示基底(例如1234的十六进制输出为 “4d2” 而不是 “0x4d2”)。可以使用操纵符showbase显示基底:

cout << dec << 1234 << ' ' << hex << 1234 << ' ' << oct << 1234 << '\n';
cout << showbase;    // show bases
cout << dec << 1234 << ' ' << hex << 1234 << ' ' << oct << 1234 << '\n';

将输出

1234 4d2 2322
1234 0x4d2 02322

即十进制数没有前缀,八进制数带前缀 “0”,十六进制数带前缀 “0x”。这与C++源代码中整数字面值的表示方式是完全一致的(见《C程序设计语言》笔记 第2章 类型、运算符与表达式 2.3节)。

showbase也是持久的。操纵符noshowbase恢复默认行为,即不显示基底。

注:showbase相当于printf格式说明符中的#,即%#x%#o,区别在于showbase是持久的。

小结:整数输出操纵符

操纵符作用
dec使用十进制(默认)
hex使用十六进制
oct使用八进制
showbase显示基底前缀
noshowbase不显示基底前缀(默认)

注:所有的位开关操纵符都定义在头文件<ios>中(包含<iostream>即可自动包含该头文件),完整列表见ios - cplusplus.com和ios - cppreference.com,参数化操纵符定义在头文件<iomanip>中(见11.2.4节)。

11.2.2 整数输入

默认情况下,>>假设数值是十进制表示,也可以使用hexoct指定按十六进制或八进制读取:

使用不同进制输入整数

如果输入

1234 4d2 2322 2322

将输出

1234 1234 1234 1234

在读取整数时,dec不接受前缀,hex接受可选的 “0x” 前缀,oct接受可选的 “0” 前缀(三个操纵符都接受可选的前导0)。例如:

操纵符输入读取的整数值(十进制)
dec“1234”, “01234”, “00001234”1234
hex“4d2”, “0x4d2”, “04d2”, “0x04d2”1234
oct“2322”, “02322”, “00002322”1234

可以使用流的成员函数unsetf()dechexoct对应的标志位全部清除:cin.unsetf(ios_base::basefield),此时输入流处于同时接受三种进制的状态。现在,对于代码

cin >> a >> b >> c >> d;

如果输入

1234 0x4d2 02322 02322

将输出

1234 1234 1234 1234

如果不调用unsetf()函数,则b的输入将会失败,因为 “0x4d2” 不是一个合法的十进制数。

11.2.3 浮点数输出

浮点数输出格式操纵符如下:

操纵符作用
fixed使用固定浮点表示
scientific使用科学记数法表示
defaultfloat选择fixedscientific中更精确的表示(默认)

注:这三个操纵符都是持久的,分别类似于printf%f%e%g

例如:

cout << 1234.56789 << "      (defaultfloat)\n"
        << fixed << 1234.56789 << "  (fixed)\n"
        << scientific << 1234.56789 << " (scientific)\n";

将输出

1234.57      (defaultfloat)
1234.567890  (fixed)
1.234568e+03 (scientific)

11.2.4 精度

默认情况下,defaultfloat格式使用6位有效数字打印浮点值,选择最合适的格式,并按四舍五入规则进行舍入。例如,1234.5678打印为 “1234.57”,1.2345678打印为 “1.23457”,1234567.0打印为 “1.23457e+06”。

使用不同格式输出浮点数

可以使用操纵符setprecision(n)来设置精度(precision):对于defaultfloat是指有效数字位数,对于fixedscientific是指小数点后的数字位数,默认为6。例如:

设置精度

将打印(注意舍入)

1234.57	1234.567890	1.234568e+03
1234.6	1234.56789	1.23457e+03
1234.5679	1234.56789000	1.23456789e+03

注:

  • setprecision()是持久的。
  • 对于fixed格式的浮点数,setprecision(n)相当于printf格式说明符%m.nd中的n;对整数和字符串无效(printf格式说明符中的精度对整数和字符串有效,见《C程序设计语言》笔记 第7章 输入与输出 7.2节)。
  • setprecision()和其他参数化操纵符定义在头文件<iomanip> (I/O manipulators)中,与hexshowbase等位开关操纵符的区别是这些操纵符需要指定一个参数。
  • os << setprecision(n)等价于os.precision(n)

11.2.5 域

对于整数、浮点数和字符串,可以使用操纵符setw(n)精确指定一个值在输出中所占的宽度,这种机制称为(field)。这对于打印表格很有用。例如:

设置域宽度

将打印

12345|12345|   12345|12345|
1234.5|1234.5|  1234.5|1234.5|
abcde|abcde|   abcde|abcde|

注意:

  • setw()不是持久的。
  • setw(n)相当于printf格式说明符%nd中的n
  • 当域宽度小于实际宽度时不会截断,域宽度无效;当域宽度大于实际宽度时填充空格,默认右对齐。可以使用操纵符setfill(c)指定填充字符,leftright指定对齐方式。

11.3 文件打开和定位

从C++的角度,文件是操作系统提供的一个抽象。如10.3节所述,文件就是一个从0开始编号的字节序列。流的属性决定了打开文件后可以执行什么操作,以及操作的含义。

11.3.1 文件打开模式

有多种文件打开模式。默认情况下,ifstream打开的文件用于读,ofstream打开的文件用于写,这满足的大多数常见需求。但是,也可以选择其他打开模式,使用位掩码类型ios_base::openmode表示:

打开模式含义
app(append) 追加模式
ate(at end) 文件尾模式
binary二进制模式
in(input) 读模式
out(output) 写模式
trunc(truncate) 覆盖原有内容

可以在文件流构造函数的文件名参数之后指定打开模式:

ofstream ofs(name1);  // defaults to ios_base::out
ifstream ifs(name2);  // defaults to ios_base::in
ofstream ofs2(name, ios_base::app);  // ofstreams by default include io_base::out
fstream fs(name, ios_base::in | ios_base::out);  // both in and out

其中|是按位或(bitwise OR)运算符,可用于组合多个模式。

打开文件的具体效果取决于操作系统。如果操作系统不能使用某种特定模式打开文件,结果将使流进入非good()状态。以读模式打开文件失败最常见的原因是文件不存在。

注意,如果以写模式打开一个不存在的文件,操作系统会创建一个新文件;而以读模式打开一个不存在的文件则会失败

注:

  • ifstreamofstream的默认打开模式分别是inout,即使指定了其他模式,也会分别自动添加inout
  • appate的区别:app在每次写操作前都定位到文件尾部,因此只能在文件尾部写数据;ate在打开文件后立即定位到文件尾部,但之后可以定位到文件的其他位置(见11.3.3节)。
  • C++文件流的打开模式与C标准库fopen()函数的打开模式对应关系如下:
C++模式C模式含义
inr
outout | truncw写(覆盖原有内容)
appout | appa追加
in | outr+读或写
in | out | truncw+读或写(覆盖原有内容)
in | out | appa+读或追加
in | binaryrb读二进制文件
out | binaryout | trunc | binarywb写二进制文件(覆盖原有内容)
app | binaryout | app | binaryab追加二进制文件
in | out | binaryr+brb+读或写二进制文件
in | out | trunc | binaryw+bwb+读或写二进制文件(覆盖原有内容)
in | out | app | binarya+bab+读或追加二进制文件

来源:https://timsong-cpp.github.io/cppwp/n3337/input.output#tab:iostreams.file.open.modes

11.3.2 二进制文件

默认情况下,iostream以文本模式读写文件,即读写字符序列(将字节按照特定字符集的编码转换为字符)。但是,也可以让istreamostream直接读写字节。这称为二进制I/O (binary I/O),通过以binary模式打开文件实现。

例如,下图是12345分别在文本文件和二进制文件中的表示方式:

文本文件和二进制文件

在文本文件中,使用31 32 33 34 35五个字节来表示 “12345” 这个字符串(字符 ‘1’ 的ASCII码是49,等于十六进制的0x31,以此类推);在二进制文件中,则使用四个字节39 30 00 00来表示32位整数0x00003039(小端顺序),等于十进制的12345。

注:磁盘上只能保存由0和1组成的二进制数据,但通常用十六进制作为简便表示,与二进制的对应关系是:每个十六进制数对应4个二进制位,如下表所示

十六进制数二进制数十六进制数二进制数
0000081000
1000191001
20010A1010
30011B1011
40100C1100
50101D1101
60110E1110
70111F1111

因此,上图所示的两个文件在磁盘上保存的实际数据分别为

31 32 33 34 35 = 00110001 00110010 00110011 00110100 00110101
   39 30 00 00 = 00111001 00110000 00000000 00000000

从这个角度看,文本文件和二进制文件本质上并没有区别,都是二进制字节数据,字节的含义完全是由文件格式人为定义的(如10.3节所述)。

下面是一个读写二进制整数文件的例子:

读写二进制整数文件

这里使用模式ios_base::binary打开二进制文件:

ifstream ifs(iname, ios_base::binary);
ofstream ofs(oname, ios_base::binary);

当我们从面向字符的I/O转向二进制I/O时,不能使用>><<运算符,因为这两个运算符按默认规则将值转换为字符序列(例如,字符串"asdf"转换为字符 ‘a’, ‘s’, ‘d’, ‘f’,整数123转换为字符 ‘1’, ‘2’, ‘3’)。而binary模式告诉流不要试图对字节做任何“聪明”的处理。

在这个例子中,对于int的“聪明”的处理是指用4个字节存储一个int(就像在内存中的表示方式一样),并直接将这些字节写入文件。之后,可以用相同的方式读回这些字节并重组出int

ifs.read(as_bytes(x), sizeof(int));
ofs.write(as_bytes(x), sizeof(int));

istreamread()ostreamwrite()都接受一个地址(这里由as_bytes()函数提供)和字节(字符)数量(这里使用运算符sizeof获得),其中地址指向保存要读/写的值的内存区域的第一个字节。例如,有一个int变量i,其值为1234(用十六进制表示为0x000004d2)则将其写入二进制文件的过程如下图所示:

写二进制文件

首先,通过as_bytes(i)获得指向i的第一个字节的地址p(假设为0xfc40),之后调用ofs.write(p, 4)将从该地址开始的4个字节写入ofs,即write()所做的事仅仅是简单的字节拷贝,read()同理。

注:

  • 从上图中可以看出,内存在本质上与文件一样,都是编号的字节序列。这里的编号叫做地址(address),通过取地址运算符&获得,例如&i;保存地址的变量叫做指针(pointer),例如p。详见17.3节和《C程序设计语言》笔记 第5章 指针与数组。
  • 由于read()write()函数的第一个参数类型是char*,而i的地址&i的类型是int*,因此as_bytes()函数使用reinterpret_cast将其强制转换为char*类型(但指针的值不变),从而将一个int的4个字节视为4个char,见17.8节。

二进制I/O复杂、容易出错。然而,对于某些文件格式必须使用二进制I/O,典型的例子是图片或声音文件。iostream库默认提供的字符I/O可移植、人类可读,而且被类型系统所支持。如果可以选择,尽量使用字符I/O(文本格式)。

11.3.3 在文件中定位

只要可以,最好使用从头到尾读写文件的方式,这是最简单、最不容易出错的方式。很多时候,当你需要修改一个文件,更好的方式是生成一个新的文件。

但是,如果必须“原地”修改文件,可以使用定位(positioning/seek)功能:在文件中选择一个特定的位置(字节编号)进行读写。每个以读模式打开的文件都有一个读位置(read/get position),每个以写模式打开的文件都有一个写位置(write/put position),如下如所示。

读写位置

可以使用istreamostream的以下函数定位读/写位置:

函数作用
tellg()获取当前读位置
seekg()设置读位置(g = “get”)
tellp()获取当前写位置
seekp()设置写位置(p = “put”)

例如:

fstream fs(name);  // open for input and output
if (!fs) error("can't open ", name);

fs.seekg(5);  // move reading position to 5 (the 6th character)
char ch;
fs >> ch;     // read and increment reading position
cout << "character[5] is " << ch << ' (' << int(ch) << ")\n";

fs.seekp(1); // move writing position to 1
fs << 'y';   // write and increment writing position

假设文件test.txt的原始内容为 “abcdefgh”,如上图所示,则上面的程序执行后文件的读写位置如下图所示:

程序执行后的读写位置

其中,运算符>>会使读位置增加读取的字符数,运算符<<会使写位置增加写入的字符数。

注意,如果试图定位到文件结尾之后的位置,结果是未定义的,不同操作系统可能会表现出不同的行为。

11.4 字符串流

可以将一个string作为istream的源或ostream的目标。从字符串读取的istream叫做istringstream,向字符串写入的ostream叫做ostringstream,这两个类定义在头文件 <sstream> 中。例如,istringstream可用于从字符串中提取数值:

字符串转浮点数

如果试图从istringstream的字符串结尾之后读取,istringstream将进入eof()状态。这意味着可以将“标准输入循环”用于istringstream

ostringstream可用于生成格式化字符串(类似于Java的StringBuilder):

ostringstream os;  // stream for composing a message
os << setw(8) << label << ": "
        << fixed << setprecision(5) << temp << unit;
someobject.display(Point(100, 100), os.str());

ostringstream的成员函数str()返回结果字符串。

ostringstream的一个简单应用是拼接字符串:

int seq_no = get_next_number();  // get the number of a log file
ostringstream name;
name << "myfile" << seq_no << ".log";  // e.g., myfile17.log
ofstream logfile(name.str());  // e.g., open myfile17.log

istringstreamostringstream均支持以下操作:

成员函数作用
默认构造函数使用空字符串初始化字符串流
构造函数(s)使用字符串s的拷贝初始化字符串流
str()返回当前内容的拷贝
str(s)将字符串s设置为当前内容,覆盖原有内容

通常情况下,我们用一个字符串来初始化istringstream,然后使用>>从字符串中读取字符。相反,通常用一个空字符串初始化ostringstream,然后用<<向其中填入字符,并使用str()获取结果。

11.5 面向行的输入

11.6 字符分类

11.7 使用非标准分隔符

11.8 还有很多

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/178249.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python Flask 实现 HTML 文件压缩,9 级压缩

本博客详细为你解释 Python Flask 框架下的 HTML 文件压缩内容&#xff0c;其第三方模块也可用在其他框架中。 本案例是基于 Python Flask 进行搭建&#xff0c;所以需要提前搭建一个 Flask 项目环境&#xff0c;有 app.py 文件和 templates/index.html 文件即可。 实现 HTML 文…

微服务架构编码构建

目录 一、约定 > 配置 > 编码 二、IDEA新建project工作空间 &#xff08;一&#xff09;微服务cloud整体聚合父工程Project &#xff08;二&#xff09;父工程POM &#xff08;三&#xff09;Maven工程落地细节复习 三、Rest微服务工程构建 &#xff08;一&#…

【蓝桥云课】进制

对于任意数制RRR的数nnn&#xff0c;都可以表达为n∑i0kaiRia0R0a1R1a2R2...akRkn\sum_{i0}^{k}a_{i}R^{i} a_{0}R^{0}a_{1}R^{1}a_{2}R^{2}...a_{k}R^{k}ni0∑k​ai​Ria0​R0a1​R1a2​R2...ak​Rk 一、十进制转RRR进制 方法&#xff1a;十进制数除RRR取余&#xff0c;余数…

kaggle竞赛 | 计算机视觉 | 数字图像基础操作

目录cv2基本操作图片哈希图像数据扩展无监督数据扩展方法监督式数据扩展方法cv2基本操作 import numpy as np import matplotlib.pyplot as plt import cv2img cv2.imread(cat.jpeg) plt.imshow(img)img cv2.imread(cat.jpeg) img cv2.cvtColor(img, cv2.COLOR_BGR2RGB) pl…

JUC并发编程(2.Java线程)

1.线程运行原理 栈与栈帧 Java Virtual Machine Stacks &#xff08;Java 虚拟机栈&#xff09; 我们都知道 JVM 中由堆、栈、方法区所组成&#xff0c;其中栈内存是给谁用的呢&#xff1f;栈内存是给线程用&#xff0c;每个线程启动后&#xff0c;虚拟机就会为其分配一块栈内…

cmake 05 使用库

本文目标 使用自己写的动态库使用第三方库更新 cm 使用自己的动态库 写一个简单的库 目录结构 F:\2023\code\cmake\calc>tree /f 卷 dox 的文件夹 PATH 列表 卷序列号为 34D2-6BE8 F:. │ CMakeLists.txt │ ├─include │ └─calc │ calc.h │ └─srcc…

2022除夕卖水果

急促的呼吸&#xff0c;急促的爱&#xff0c;急促的吆喝声 来吧 朋友 伸出你的手 还有十分钟&#xff0c;拆拆盒子收摊中 管尝管饱&#xff0c;礼盒散装可打包 红橙黄绿青蓝紫 苹果樱桃小番茄 柠檬枳柑桔橙柚 香蕉龙眼与柠檬 蜜瓜西瓜猕猴桃 提子樱桃火龙果 葡萄甘蔗车厘子 柿子…

ES常用知识点整理第一部分

ES常用知识点整理第一部分引言APICrud APIBulk API批量读取批量查询ES服务器常见错误返回倒排索引分词器中文分词器Search APIURI SearchQuery DSL查询表达式短语搜索Query String 和 Simple Query Stringmapping映射动态映射手动映射多字段特性自定义分词Index TemplateDynami…

【Ubuntu】Nacos 2.1 单机安装

目录Nacos 2.1 单机安装1. 从GitHub下载2.1.0的压缩包2. 解压与配置3. 数据库配置4. 单机启动nacos遇到的一些错误ErrMsg:jmenv.tbsite.netlibstdc.so.6: cannot open shared object file: No such file or directoryCaused by: java.lang.IllegalStateException: No DataSourc…

3DCAD图纸转2D(DXF)图纸通用解决方案文稿

3DCAD图纸转2D(DXF)图纸通用解决方案文稿 本文地址&#xff1a;https://gitee.com/dvaloveu/lovedva/issues/I6B5YC 视频演示(1.25倍速&原速)&#xff1a;Acfun Bilibili 脚本地址&#xff1a;https://gitee.com/dvaloveu/ug-automation/blob/master/ug2caxa/hellowolrd.p…

第二章物理层-第五节:信道的极限容量

文章目录一&#xff1a;相关概念&#xff08;1&#xff09;失真&#xff08;2&#xff09;信道带宽W&#xff08;3&#xff09;波特率&#xff08;Baud&#xff09;二&#xff1a;奈奎斯特定理&#xff08;奈氏准则&#xff09;三&#xff1a;香农定理本节对应视频 【计算机网络…

sql注入绕过(持续更新)

判断存在注入 ab|| 查询表名 select{x table_name}from information_schema.tables where table_schemadatabase# () SQL 中的 substring 函数是用来抓出一个栏位资料中的其中一部分。这个函数的名称在不同的资料库中不完全一样&#xff1a; MySQL: SUBSTR( ), SUBSTRING(…

Coolify系列01- 从0到1超详细手把手教你上手Heroku 和 Netlify 的开源替代方案

什么是Coolify 一款超强大的开源自托管 Heroku / Netlify 替代方案coolLabs是开源、自托管和以隐私为中心的应用程序和服务的统称 为什么使用Coolify 只需单击几下即可托管你的应用、数据库或其他开源服务&#xff0c;等。它是 Heroku 和 Netlify 的一个替代方案。通过 Cool…

springbootWeb常用注解使用

springbootWeb常用注解使用PathVariable 路径变量注解RequestHeader 请求标头注解RequestParam 请求域注解RequestBody 请求体注解ModelAttribute使用1,将其置于方法上&#xff1a;使用2&#xff0c;将其置于方法参数上&#xff1a;CookieValuePathVariable 路径变量注解 可以…

3.4动态规划--最大字段和

要好好学习这个难受难受超级难受的动态规划了&#xff0c;千万不要再沉迷在看剧和玩耍里面了。必须承认最近没有好好学习。 写在前面 最大字段和书上介绍了三种解法&#xff1a;暴力、递归分治、动态规划 递归分治&#xff0c;一分为二&#xff0c;合并的时候有三种情况&…

java容器轻松理解 collection collections(异同篇)

(1)collectionset &#xff08;只能迭代&#xff0c;不可以按下标取值&#xff09;hashset、Treesetlist&#xff08;可以迭代&#xff0c;也可以按下标取值&#xff09;ArrayList、LinkedList、Vector特点&#xff1a;ArrayList:方便随机访问&#xff0c;由数组实现的。所以中…

十大经典排序算法(动态演示+代码)-选择排序与插入排序

选择排序 一、什么是选择排序&#xff1f; 选择排序&#xff08;Selection sort&#xff09;是一种简单直观的排序算法。它的工作原理是&#xff1a;第一次从待排序的中数据元素选出最小&#xff08;或最大&#xff09;的一个元素&#xff0c;存放在序列的起始位置&#xff0c…

行为型模式-策略模式

1.概述 先看下面的图片&#xff0c;我们去旅游选择出行模式有很多种&#xff0c;可以骑自行车、可以坐汽车、可以坐火车、可以坐飞机。 作为一个程序猿&#xff0c;开发需要选择一款开发工具&#xff0c;当然可以进行代码开发的工具有很多&#xff0c;可以选择Idea进行开发&a…

04_iic子系统

总结 iic_client和iic_driver 加入iic总线的思想和paltform总线的玩法一样 把iic设备和驱动注册到iic总线中 构造出字符设备驱动和设备节点供app进行操作 但是iic硬件设备是挂在iic控制器下面的 所以iic控制器也会有自己的驱动和设备树节点 厂家一般都会帮做好 我们写的iic_dr…

离散系统的数字PID控制仿真-2

设计离散PID控制器&#xff0c;各信号的跟踪结果如图所示&#xff0c;其中S代表输入指令信号的类型。通过取余指令 mod实现三角波和锯齿波。当S1时为三角波&#xff0c;S2时为锯齿波&#xff0c;S3时为随机信号。在仿真过程中&#xff0c;如果 D1&#xff0c;则通过 pause命令实…