IO进程线程day1(2023.7.25)

2023-11-07

一、Xmind整理:

什么是IO:

文件IO函数与标准IO函数: 

二、课上练习:

练习1:标准IO函数的简单示例

scanf:
if(OS == Linux)
{
    去调用Linux的文件IO
    read();
}  
else if(OS == windows)
{
    去调用windows的文件IO
    fread();   
}

练习2:FILE结构体

struct _IO_FILE {

  char* _IO_buf_base;   缓冲区的起始地址
  char* _IO_buf_end;    缓冲区的结尾地址
  

  int _fileno;          文件描述符,在文件IO的时候再讲解。
}

缓冲区大小:尾地址-首地址。

练习3:fopen

功能:打开一个文件

原型:

#include <stdio.h>

FILE *fopen(const char *pathname, const char *mode);

参数:

 char *pathname:指定要打开的文件路径以及名字;
 char *mode:以什么方式打开文件;
       r      以读的方式打开文件; 流在文件开头位置---> 若要读取文件,则从开头开始读取
              若文件不存在,打开失败。

       r+     以读写的方式打开文件; 流在文件开头位置---> 若要读写取文件,则从开头开始操作
              若文件不存在,打开失败.

       w      以写的方式打开文件,---> 若要写取文件,则从开头开始操作
              若文件不存在,则会创建文件,并打开;
              若文件存在,则清空文件,并打开; 

       w+     以读写的方式打开文件,---> 若要读写取文件,则从开头开始操作
              若文件不存在,则会创建文件,并打开;
              若文件存在,则清空文件,并打开; 

       a      以写的方式打开文件,---> 若要写取文件,则从结尾开始操作
              若文件不存在,则会创建文件,并打开;
              若文件存在,则文件的流在文件结尾,不清空;

       a+     以读写方式打开文件,
              若文件不存在,则会创建文件,并打开;
              若文件存在,则流的初始位置:
              1. 若开始是读文件,则流在文件开头
              2. 若开始是写文件,则流在文件结尾。

返回值:

成功,返回FILE *类型指针;
失败,返回NULL,同时更新errno;     
errno:本质上是一个整型数,不同的错误会更新不同的errno;    定义在errno.h这个头文件中
所有情况:/usr/include/asm-generic   errno.h  errno-base.h
  

打开一个文件,以读的方式 

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
int main(int argc, const char *argv[])
{
	//1.抄函数名
	//2.考虑参数列表(数据类型一一匹配)
	//3.考虑返回值,定义相同类型的变量接收
	//打开一个文件,以读的方式
	FILE* fp=fopen("./1.txt","w");
	if(NULL==fp)
	{
	//	printf("errno=%d\n",errno);//2:文件不存在 13:文件权限不足
	//	printf("fopen函数运行失败\n");
	    perror("fopen");
		return -1;
	}
	printf("fopen函数运行成功,文件打开成功\n");

	if(fclose(fp)<0)
	{
		perror("fclose");
		return -1;
	}
	printf("fclose function success\n");
	return 0;
}

练习4:perror

功能:根据errno,打印对应的错误信息

原型:

#include <stdio.h>
void perror(const char *s);

参数:

char *s:用于提示的字符串;

练习5:fclose

功能:关闭指定的文件; 释放资源,例如缓冲区空间。若不关闭,则可能会造成内存泄漏

原型:

#include <stdio.h>
int fclose(FILE *stream);

参数:

FILE *stream:指定要关闭的文件对应的流指针;

返回值: 

成功,返回0;
失败,返回EOF,同时更新errno; # define EOF (-1) 

 打开文件,以写的方式

	if(fclose(fp)<0)
	{
		perror("fclose");
		return -1;
	}
	printf("fclose function success\n");
	return 0;

练习6:fprintf

功能:将数据格式化输出到指定文件中

原型:

#include <stdio.h>
int printf(const char *format, ...);     
printf("hello world %d %c %f \r\n\t", 1, 'a', 10.1);
int fprintf(FILE *stream, const char *format, ...);

参数:

FILE *stream:流指针,指定要输出到哪个文件中,就填对应文件的流指针;
char *format:格式化字符串:字符,占位符,转义字符;
...:不定参数,不定数据个数,不定数据类型。

返回值: 

成功,返回被打印的字符个数;
失败,返回负数,

 打开文件,以写的方式

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main(int argc, const char *argv[])
{
	//打开文件,以写的方式
	FILE* fp=fopen("./1.txt","w");
	if(NULL==fp)
	{
		perror("fopen");
		return -1;
	}
	printf("fopen success\n");
	
	//将hello world输出到fprintf.txt文件中
	//printf("hello world %d\n",12);
	fprintf(fp,"hello world %d\n",12);

	if(fclose(fp)<0)
	{
		perror("fclose");
		return -1;
	}
	printf("fclose function success\n");
	return 0;
}

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main(int argc, const char *argv[])
{
	//打开文件,以写的方式
	FILE* fp=fopen("./1.txt","a+");
	if(NULL==fp)
	{
		perror("fopen");
		return -1;
	}
	/*
	   printf("fopen success\n");

	//将hello world输出到fprintf.txt文件中
	//printf("hello world %d\n",12);
	fprintf(fp,"hello world %d\n",12);
	*/
	int arr[3]={1,2,3};
	for(int i=0;i<3;i++)
	{
		printf("%d ",arr[i]);  //指定打印到终端
		fprintf(fp,"%d",arr[i]);  //指定打印到fp对应的文件中
	}
	putchar(10);

	if(fclose(fp)<0)
	{
		perror("fclose");
		return -1;
	}
	printf("fclose function success\n");
	return 0;
}

练习7:fscanf

功能:从指定文件中格式化读取数据; scanf fscanf的%s %d默认不识别空格,\n, \t

          %c形式获取单个字符会识别空格,换行。

原型:

#include <stdio.h>
int scanf(const char *format, ...);
int fscanf(FILE *stream, const char *format, ...);

参数:

FILE *stream:流指针,从哪个文件中读取,填对应的流指针;
char *format:格式化字符串:字符,占位符,转义字符;
...:不定参数,不定数据个数,不定数据类型。

返回值: 

成功,返回成功读取的数据个数;
=EOF,文件读取完毕
=EOF,函数运行失败,同时更新errno;

注意: 

1.%s %d不识别空格 \n \t,若想要获取到上述字符,需要使用%c形式

2.读写操作后,文件偏移量会自动偏移,所以若想要从头读取数据,就需要想办法将偏移量修改到开头位置

        a.将文件关闭重新开

        b.fseek ftell

3.printf ,scanf怎么使用 ,fprintf fscanf就怎么使用。

 打开文件,以写的方式

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main(int argc, const char *argv[])
{
	//打开文件
	FILE* fp=fopen("./2.txt","a+");
	if(NULL==fp)
	{
		perror("fopen");
		return -1;
	}
	printf("以读的方式打开文件成功\n");
	//指定从fp对应的文件中读取数据
	char arr[100]="";
	char arr1[100]="";
	int a=0;
	fscanf(fp,"%s%s%d",arr,arr1,&a);
	printf("%s %s %d\n",arr,arr1,a);
	//关闭文件
	if(fclose(fp)<0)
	{
		perror("fclose");
		return -1;
	}
	printf("fclose function success\n");

	return 0;
}

用fprintf与fscanf函数实现:文件拷贝,例如将1.txt的内容拷贝到2.txt中:

#include <stdio.h>

int main(int argc, const char *argv[])
{
    //以读的方式打开源文件
    FILE* fp_r = fopen("./01_fopen.c", "a+");
    if(NULL == fp_r)
    {
        printf("__%d__\n", __LINE__);
        perror("fopen");
        return -1;
    }

    //以写的方式打开目标文件
    FILE* fp_w = fopen("copy.c", "w+");
    if(NULL == fp_w)
    {
        printf("__%d__\n", __LINE__);
        perror("fopen");
        return -1;
    }

    char c = 0;
    while(fscanf(fp_r, "%c", &c) != EOF)
    {
        //写文件
        fprintf(fp_w, "%c", c);
    }

    printf("拷贝成功\n");
    //此时文件偏移量在文件的结尾,如果想要从copy.c文件中读取数据
    //则需要将文件描述符重新偏移打破文件开头位置
    //所以需要关闭文件重新打开,将偏移量重新改回到开头位置
                                                                
    //关闭文件
    fclose(fp_w);
    fclose(fp_r);

    //从fp_w中读取数据,然后打印到终端
    fp_w = fopen("copy.c", "r");
    while(fscanf(fp_w, "%c", &c) != EOF)
    {
        printf("%c", c);
    }
    fclose(fp_w);

    return 0;
}

练习8:fputc

功能:将单个字符打印到指定的文件中

原型:

#include <stdio.h>
int fputc(int c, FILE *stream);
int putchar(int c);     putchar('a'); putchar(97); putchar(10); putchar('\n');

参数:

int c:指定要输出的字符对应的字符形式或者整型形式,填'a', 也可以填97;
FILE *stream:流指针,想要打印到哪个文件中就填哪个文件对应的流指针;

返回值: 

成功,返回成功输出字符对应的整型形式;
失败,返回EOF;

注意: 

Linux操作系统,默认以\n结尾,若用编辑器保存检测到没有以\n结尾,则会自动补充'\n'

unix操作系统,默认以\r\n结尾,

windows操作系统,默认以\r结尾。

练习9:fgetc

功能:从指定文件中读取单个字符

原型:

#include <stdio.h>
int fgetc(FILE *stream);

参数:

FILE *stream:指定要从哪个文件中读取

返回值: 

成功,返回成功读取到的字符对应的整型形式;
当文件读取完毕或者函数运行失败,返回EOF;
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main(int argc, const char *argv[])
{
	FILE* fp=fopen("./3.txt","r");

	if(NULL==fp)
	{
		perror("fopen");
		return -1;
	}
	printf("打开成功\n");
	fputc('a',fp);	
	fputc('b',fp);	
	fputc('c',fp);	
	fputc('\n',fp);	
	fclose(fp);
	fp=fopen("./3.txt","r");
	if(NULL==fp)
	{
		perror("fopen");
		return -1;
	}
	printf("fopen success\n");
	int c=0;
	while(1)
	{
		c=fgetc(fp);
		if(EOF==c)
			break;
		printf("%c %d\n",c,c);
	}
	fclose(fp);
	return 0;
}

三、课后作业:

1.实现登录功能。自定义一个usr.txt,手动输入账户密码,格式如下:账户 密码

例如: zhangsan 12345

            lisi abcde

            wangwu abc123

需求如下:从终端获取账户密码,与文件中的账户密码比较

                  若终端输入的账户不存在,则输出账户不存在

                  若终端输入的账户存在,但是密码不正确,则输出密码错误

                  若账户密码均正确,则输出登录成功

附加题:

                  实现注册功能,注册的账户密码存储在上一题的文件中。需求:不能重复注册。

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main(int argc, const char *argv[])
{
	FILE* fp=fopen("./usr.txt","r+");
	if(NULL==fp)
	{
		perror("fopen");
		return -1;
	}
	printf("fopen success\n");
	char name[20];
	char password[20];
	char aname[20];
	char apassword[20];
	int c;
	//从终端获取账户名与密码
	printf("请输入账户名与密码:");
	scanf("%s %s",name,password);
	//比较
	while(1)
	{
		if(fscanf(fp,"%s %s",aname,apassword)==EOF)
		{
			printf("%s 账户不存在\n",name);
			fclose(fp);
			char a;
			printf("Do you want to register:");
			scanf(" %c",&a);
			if('Y' == a || 'y' == a)
			{
				printf("name:");
				scanf("%s",name);
				fp = fopen("usr.txt","r");
				while(strcmp(name,aname)!=0)
				{
					c = fscanf(fp,"%s",aname);
					if(EOF == c)
						break;			
				}
				if(strcmp(name,aname)==0)
				{
					printf("账户已存在!\n");
					return -1;
				}
				else
				{
					printf("password:");
					scanf("%s",password);
					fclose(fp);
					fp = fopen("usr.txt","a");
					fprintf(fp,"%s %s\n",name,password);
				}
			}
			if(EOF == c)
				break;
		}
		if(strcmp(name,aname)==0)
		{
			printf("%s 账户已存在\n", name);
			if(strcmp(password,apassword)==0)
			{
				printf("%s 登录成功\n",name);
			}
			else
			{
				printf("%s 密码错误\n",name);
			}
			break;
		}
	}
	if(fclose(fp)<0)
	{
		perror("fclose");
		return -1;
	}
	printf("fclose function success\n");
	return 0;
}

2.文件拷贝,例如将1.txt的内容拷贝到2.txt中

   要求用fgetc计算一个文件有多少个字节

   用fgetc计算一个文件有几行? 计算\n个数

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main(int argc, const char *argv[])
{
	FILE *fp = fopen("./1.txt","r");
	FILE *fp_t = fopen("./2.txt","w");
	if(NULL == fp)
	{
		perror("fopen");
		return -1;
	}
	char c;
	int count=0,line=0;
	while((c = fgetc(fp)) != EOF)
	{
		count++;
		if(c == '\n')
			line++;
		fputc(c,fp_t);
	}
	printf("文件有%d字节,有%d行\n",count,line);
	fclose(fp);
	fclose(fp_t);
	return 0;
}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

IO进程线程day1(2023.7.25) 的相关文章

  • 华为od寻找关键钥匙

    题目描述 小强正在参加 密室逃生 游戏 当前关卡要求找到符合给定 密码K 升序的不重复小写字母组成 的箱子 并给出箱子编号 箱子编号为1 N 每个箱子中都有一个字符串s 字符串由大写字母 小写字母 数字 标点符号 空格组成 需要在这些字符串
  • 服务机器人研究报告:产业链分析及投资前景预测

    服务机器人研究报告 产业链分析及投资前景预测 2017 08 31 08 37中国投资咨询网 服务机器人在国内发展的阻力远远小于工业机器人 一方面 服务机器人是国内公司与国外公司差距较小的领域 国内企业的服务机器人往往能针对特定市场进行开发
  • 用stm32f103点灯的一些模式

    关于stm32点灯 因为使用的芯片就母版可能不是一样的 这里我使用的是stm32f103c8t6来进行讲解 首先要点亮一个灯 就需要知道灯是一个外设 就涉及到gpio的一些知识点 第一步就是要对板子原理图上的硬件连接进行一定的了解 下面这是
  • 堆排序HeapSort

    堆排序的最好最坏平均时间复杂度都为nlogn adjustment执行次数为logn for循环次数为3 2n 空间复杂度为O 1 没有使用额外空间 include
  • 文盘Rust -- tonic-Rust grpc初体验

    gRPC 是开发中常用的开源高性能远程过程调用 RPC 框架 tonic 是基于 HTTP 2 的 gRPC 实现 专注于高性能 互操作性和灵活性 该库的创建是为了对 async await 提供一流的支持 并充当用 Rust 编写的生产系
  • 如何保护移动应用程序安全–移动应用程序安全检查表

    Security has always been a major concern for businesses And this concern is even greater when it comes to mobile apps 安全

随机推荐

  • 全程 AIGC 游戏创作,2小时开发微信小游戏!(完整ChatGPT调教流程)

    给 AI 打工 2小时开发一款3D小游戏可行么 源码下载地址见文末 最近 科技发展是日新月异 如果说 Stable Diffusion 和 Mid Journey 只是卷死了美术 我们还在庆幸研发不能被替代 那么 3 月份出来的 GPT4
  • react-router-dom的基本使用

    React Router Dom 1 React Router react router是一个核心库 我们在做PC端时要用react router dom 而在移动端我们要用React Router Native 2 React Route
  • 无极超分辨率

    文章目录 前言 一 Meta SR 1 1 Meta SR A Magnification Arbitrary Network for Super Resolution 1 2 Location Projection 1 3 Weight
  • Android Update Engine 分析(二十一)Android A/B 更新过程

    0 背景 早期 Android A B 系统升级在 Android 7 1 版本推出时 参考文档十分有限 也就是 Android 官方大概有两三个页面介绍文档 我的第一篇 A B 系统分析文章 Android A B System OTA分
  • php使用ecc算法进行签名,ECDSA签名算法(ECC椭圆曲线算法3)

    现在有一个场景 Alice想要用私钥签名一个数据 Bob想要使用Alice的公钥验证这个签名 只有Alice能够进行计算签名然后得到签名 每个人都能验证签名值 首先Alice和Bob拥有相同的椭圆曲线参数 算法被签名称之为ECDSA 是DS
  • 关于OpenGL纹理尺寸的坑 - 图像行偏移,出现异色条纹

    学习OpenGL时想简单创建一个纹理 但马上就出现错误 错误效果如下 原图如下 由于之前没有碰过这种问题 费了好大一番功夫才找到问题所在 原始图片尺寸为210 220 OpenGL版本与教程一致 为3 3 把像素值打印出来观察之后发现 传入
  • Spring 特性

    目录 核心特性 数据存储 Web 技术 Web Servlet 技术栈 Spring 1 4的唯一支持 Web Reactive 技术栈 Spring 5引入 技术整合 Integration 测试 Testing Spring 模块化设计
  • DeformableDetr论文简介+mmdet源码解读

    文章目录 前言 一 论文解读 1 1 研究问题 1 2 可形变注意力模块 1 3 拓展到多层特征图 二 mmdet源码讲解 2 1 图像特征提取 2 2 生成mask和位置编码 2 3 送入Transformer 2 3 1 Transfo
  • 机器学习之朴树贝叶斯②——调库实现

    文章目录 多项式朴素贝叶斯 MultinomialNB 高斯朴素贝叶斯 GaussianNB 多项式朴素贝叶斯 MultinomialNB sklearn naive bayes MultinomialNB alpha 1 0 fit pr
  • 为什么HashMap线程不安全?

    我们都知道 HashMap 是线程不安全的 那 HashMap 为什么线程不安全 JDK1 8 还有这些问题吗 如何解决这些问题呢 本文将对该问题进行解密 多线程下扩容死循环 JDK1 7中的 HashMap 使用头插法插入元素 在多线程的
  • 如何在深度学习中处理图像数据?

    深度学习在图像处理领域取得了重大的突破 可以用于图像分类 目标检测 图像生成等各种任务 处理图像数据的关键是将图像转换为适合深度学习模型处理的形式 下面是处理图像数据的一般步骤 1 数据准备 收集和整理用于训练的图像数据集 数据集应包含图像
  • MySQL基础篇——第10章 DDL(数据定义):创建和管理表

    MySQL基础篇 第10章 DDL 数据定义 创建和管理表 1 基础知识 1 1 一条数据存储的过程 存储数据是处理数据的第一步 只有正确地把数据存储起来 我们才能进行有效的处理和分析 MySQ的数据存储过程 创建数据库 确认字段 创建数据
  • 基于Matlab实现图像目标边界描述

    图像目标边界描述是图像处理中的一个重要问题 边界描述可以用于目标检测和识别 图像分割等应用 Matlab提供了强大的图像处理工具箱 可以方便地实现图像目标边界描述 本文介绍一种基于边缘检测的图像目标边界描述方法 并提供一个简单的案例源码 文
  • 自定义Webpack配置

    自定义Webpack配置 1 初始化并创建要被打包的文件 2 命令行配置 3 配置文件配置 1 初始化并创建要被打包的文件 首先创建文件夹webpack demo 随便起一个 用来演示打包过程 在该文件夹下终端运行命令 对项目进行初始化操作
  • java静态编译 动态编译_Java代码的静态编译和动态编译中的问题比较

    两种技术都需要谨慎选择编译的方法以实现最高的性能 对动态编译器而言 编译器自身作出决策 而对于静态编译器 由开发人员作出选择 让 JIT 编译器选择编译的方法是不是优点很难说 取决于编译器在给定情形中推断能力的好坏 在大多数情况下 我们认为
  • Nvidia 显卡 Failed to initialize NVML Driver/library version mismatch 错误解决方案

    目录 问题复现 问题原因 问题分析 解决方案 问题复现 nvidia smi gt Failed to initialize NVML Driver library version mismatch 问题原因 出现这个问题的原因是nvidi
  • Windows修改java环境变量不生效

    Windows修改java环境变量不生效 1 运行 regedit 2 HKEY LOCAL MACHINE SOFTWARE JavaSof t Java Runtime Environment 3 删除不需要的
  • Qt中Q_D宏及d指针

    原文标题 d指针在Qt上的应用及实现 原文链接 http blog csdn net rabinsong article details 9474859 正文 Qt为了使其动态库最大程度上实现二进制兼容 引入了d指针的概念 那么为什么d指针
  • 全新服务器安装redis步骤大全

    如安装过程中报错可查看另外一篇文章 https blog csdn net Cjava math article details 108832966 第一步 下载安装包 访问https redis io download 到官网进行下载 这
  • IO进程线程day1(2023.7.25)

    一 Xmind整理 什么是IO 文件IO函数与标准IO函数 二 课上练习 练习1 标准IO函数的简单示例 scanf if OS Linux 去调用Linux的文件IO read else if OS windows 去调用windows的