2017-2018-1 20155320 《信息安全系统设计基础》第十四周学习总结

本文总结了《信息安全系统设计基础》第十章系统级I/O的学习内容,涵盖了Unix I/O、文件操作、文件元数据读取、目录内容读取等方面的知识,并通过具体实例加深理解。

2017-2018-1 20155320 《信息安全系统设计基础》第十四周学习总结

参考老师提供的教材内容导读

本周的内容是要找出全书你认为学得最差的一章,深入重新学习一下

我决定学习第十章——系统级I/O,因为对相关内容感觉再应用方面并未十分熟练。

教材学习内容总结

10.1 Unix I/O

  • Linux文件:m个字节的序列
  • 所有的I/O设备都被模型化为文件,所有的输入和输出都被当作对相应文件的读和写来执行
  • Linux内核引出一个简单、低级的应用接口,称为Unix I/O,这使得所有的输入和输出都能以一种统一且一致的方式来执行

    10.2 文件

  • 每个Linux文件都有一个类型来表明它在系统中的角色:
    普通文件包含任意数据。

    目录是包含一组链接的文件,其中每个链接都将一个文件名映射到一个文件,这个文件可能是另一个目录。每个目录至少含有两个条目:'.'是到该目录自身的链接,'..'是到目录层次结构中父目录的链接。

    套接字是用来与另一个进程进行跨网络通信的文件。

  • 用cd命令来修改shell中的当前工作目录
  • 路径名有两种形式:
    绝对路径名以一个斜杠开始,表示从当前工作目录开始的路径

    相对路径名以文件名开始,表示从当前工作目录开始的路径

    10.3 打开和关闭文件

  • 进程通过open函数来打开一个已存在的文件或者创建

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int open(char *filename, int flags, mode_t mode);
若成功则返回新的文件描述符,若出错则为1
  1. 参数解析:

    返回值:类型为int型,返回的是描述符数字,总是在进程中当前没有打开的最小描述符。如果出错,返回值为-1.

    filename:文件名

    flags:指明进程打算如何访问这个文件,可以取的值见下:

以下值可以用“或”连接起来。
O_RDONLY | 只读
---|---
O_WRONLY | 只写
O_RDWR |可读可写
O_CREAT |文件不存在,就创建新文件
O_TRUNC|如果文件存在,就截断它
O_APPEND|写操作前设置文件位置到结尾处

mode参数指定了新文件的访问权限位

image

  • close函数
#include <unistd.h>

int close(int fd);

1.参数解析:

返回值:成功返回0,出错返回-1

关闭一个已经关闭的描述符会出错

fd:即文件的描述符。

10.4 读和写文件

-读 read

#include <unistd.h>

ssize_t read(int fd, void *buf, size_t n);//返回有符号值

成功则返回读的字节数,EOF返回0,出错返回-1。返回值为有符号数。
  1. 参数解析:

    fd:文件描述符

    buf:存储器位置

    n:最多从当前文件位置拷贝n个字节到存储器位置buf

-写 write

#include <unistd.h>

ssize_t write(int fd, void *buf, size_t n);

成功则返回写的字节数,出错返回-1。返回值为有符号

(2)参数解析:

fd:文件描述符

buf:存储器位置

n:最多从存储器位置buf拷贝n个字节到当前文件位置

PS:read和write在正常情况下返回值是实际传送的字节数量。

#include "csapp.h"
int main(void)
{
char c;
while(read(STDIN_FILENO,&c,1)!=0)
write(STDOUT_FILENO,&c,1);
exit(0);
}
  • 该程序使用read和write调用一次一个字节地从标准输入复制到标准输出。

1071521-20171219230926928-1062190673.png

  • 不足值

不足值指在某些情况下,read和write传送的字节比应用程序要求的要少,原因如下:

读的时候遇到EOF

从终端读文本行

读和写socket

10.5 用RIO包健壮地读写

  • RIO:I/O包,会自动为你处理不足值
  • RIO的无缓冲的输入输出函数
#include "csapp.h"

ssize_t rio_readn(int fd, void *usrbuf, size_t n);
ssize_t rio_writen(int fd, void *usrbuf, size_t n);

rio_readn成功则返回传送的字节数,EOF为0(一个不足值),出错为-1
rio_writen成功则返回传送的字节数,出错为-1,没有不足值。
  1. 参数:

    fd:文件描述符

    usrbuf:存储器位置

    n:传送的字节数

  • read遇到EOF只能返回一个不足值,write不会返回不足值。
  • 被中断时函数会重启read或write
  • RIO的带缓冲的输入函数
ssize_t rio_readnb(rio_t *rp, void *usrbuf, size_t n);
ssize_t rio_readlineb(rio_t *rp, void *usrbuf, size_t maxlen);
  • 可以高效的从文件中读取文本行和二进制数据。

  • 文本行就是一个由换行符结尾的ASCII码字符序列。换行符数字值为0x0a.rio_readlineb函数从内部读缓冲区拷贝一个文本行,当缓冲区为空时,会自动地调用read重新填满缓冲区。rio_readn带缓冲区的版本:rio_readnb。

  • rio_readlineb从rp读出一个文本行(包括换行符)并把它存到usrbuf,并用空字符结束这个文本行。最多读maxlen-1个字节,剩下一个给结尾处的空字符。

  • rio_readnb最多读n个字节。

  • 课本实例

#include "csapp.h"
int main(int argc,char **argv)
{
int n;
rio_t rio;
char buf[MAXLINE];
rio_readinitb(&rio,STDIN_FILENO);
while((n=rio_readlineb(&rio,buf,MAXLINE))!=0)
rio_writen(STDOUT_FILENO,buf,n);
}
  • rio_readinitb从内部读缓冲区复制一个文本行,当缓冲区变空使,会自动地调用read重新填满缓冲区。以上程序一次一行地从标准输入复制一个文本文件到标准输出。

1071521-20171220111855928-809553083.png

10.6读取文件元数据

  • 通过调用stat和fstat函数,检索文件的元数据。
#include <unistd.h>
#include<sys/stat.h>
int stat(const char *filename,struct stat *buf);
int fstat(int fd,struct stat *buf);
返回:若成功则为0,若出错则为-1;
  • stat的数据结构

1071521-20171220113802646-485956197.png

st_size成员包含了文件的字节数大小,st_mode成员则编码了文件访问许可位和文件类型。

- st_mode成员的文件类型:
S_ISREG(m)普通文件
S_ISDIR(m)目录文件
S_ISSOCK(m)网络套接字
  • 课本实例——展示了如何使用宏和stat函数来读取和解释一个问价的
#include "csapp.h"
int main(int argc,char **argv)
{
struct stat stat;
char *type,*readok;
Stat(argv[1],&stat);
if(S_ISREG(stat.st_mode))
  type = "regular";
else if(S_ISDIR(stat.st_mode))
  type = "directory";
else
  type = "other";
if((stat.st_mode & S_IRUSR))
  readok = "yes";
else
  readok = "no";
printf("type:%s,read:%s\n",type,readok);
}

1071521-20171220204827350-332852966.png

10.7读取目录内容

  • 应用程序可以用readdir系列函数来读取目录内容
#include<sys/types.h>
#include<dirent.h>
DIR *opendir(const char *name);
返回:若成功,则为处理的指针;若出错,则为NULL
  • 函数opendir以路径名为参数,返回指向目录流的指针。
#include<dirent.h>
struct dirent *readdir(DIR *dirp);
返回:若成功,则为指向下一个目录项的指针,若没有更多的目录项或出错,则为NULL
  • dirent指向流,每个目录项都是一个结构

1071521-20171220214603943-890497814.png

  • 函数closedir关闭流并释放其所有的资源。
#include<dirent.h>
int closedir(DIR *dirp);
返回:成功为0,错误为-1

10.8 共享文件

  • 用三个相关的数据结构表示打开的文件:

    • 描述符表:每个打开的描述符表项指向文件表中的一个表项。

    • 文件表:打开的文件的集合是由一张文件表表示的,所有的进程共享这张表。包括文件位置、引用计数(当前指向该表项的描述符表项数),指向v-node表的指针。

    • v-node表:包含stat结构中大多数信息。

  • 打开文件的内核数据结构:

image

  • 共享了同一个磁盘文件:

image

  • 子父进程共享文件
    子进程有一个父进程描述表符副本,所以他们共享打开文件的集合。注意,在内核删除相应文件表表项之前,子父进程必须都关闭他们的描述符。
    image

练习

  • 10.1下面程序的输出是什么?
#include "csapp.h"
int main()
{
int fd1,fd2;
fd1=open("foo.txt",O_RDONLY,0);
close(fd1);
fd2=open("baz.txt",O_RDONLY,0);
printf("fd2=%d\n",fd2);
exit(0);
}
  • 结果为:Unix进程生命周期开始时,打开的描述符赋给了stdin(描述符0)、stdout((描述符1)和stderr(描述符2)。open函数总是返回最低的未打开的描述符,所以第一次调用open会返回描述符3,调用close函数会释放描述符3。最后对open的调用会返回描述符3,因此程序的翰出是“fd2=3”。当文件不存在是,输出“fd2=-1”

1071521-20171223095646740-1227092851.png

  • 10.2 假设磁盘文件foobar.txt由6个ASCII码字符“foobar"组成,那么下列程序的输出是什么?
#include "csapp.h"
int main()
{
int fd1,fd2;
char c;
fd1=open("foobar.txt",O_RDONLY,0);
fd2=open("foobar.txt",O_RDONLY,0);
read(fd1,&c,1);
read(fd2,&c,1);
printf("c=%c\n",c);
exit(0);
}
  • fd2读操作读取foobar.txt的第一个字节

1071521-20171223102834459-1850774783.png

  • 10.3 就像前面那样,假设磁盘文件foobar.txt由6个ASCII码字符“foobar"组成,那么下列程序的输出是什么?
#include "csapp.h"
int main()
{
int fd;
char c;
fd = open("foobar.txt",O_RDONLY,0);
if(fork()==0)
{
read(fd,&c,1);
exit(0);
}
wait(NULL);
read(fd,&c,1);
printf("c=%c\n",c);
exit(0);
}
  • 描述符fd在父子进程中都指向同一个打开文件表表项,当子进程读取文件的第一个字节时,文件位置加1。因此,父进程会读取第二个字节,也就是o

1071521-20171223104255178-1564453437.png

  • 10.4 如何用dup2将标准输入重定向到描述符5?
  • dup2(5,0)
  • 10.5 假设磁盘文件foobar.txt由6个ASCII码字符“foobar"组成,那么下列程序的输出是什么?
#include "csapp.h"
int main()
{
int fd1,fd2;
char c;
fd1=open("foobar.txt",O_RDONLY,0);
fd2=open("foobar.txt",O_RDONLY,0);
read(fd2,&c,1);
dup2(fd2,fd1);
read(fd1,&c,1);
printf("c=%c\n",c);
exit(0);
}
  • 将fdl重定向到了fd2,输出实际上是c=o

1071521-20171223111733662-1173663195.png

教材学习中的问题和解决过程

  • 问题1:
  • 问题1解决方案:

代码调试中的问题和解决过程

  • 问题1:在起初在编译时用gcc xx.c -o xx出现错误
  • 问题1解决方案:后来发现只要加一个-lpthread就行了,用命令gcc xx.c -o xx -lpthread

上周考试错题总结

  • Y86-64中(ABE)指令没有访存操作.

A .
rrmovl

B .
irmovq

C .
rmmovq

D .
pushq

E .
jXX

F .
ret
解析:jXX无访存操作

  • 16
    ( 多选题 | 1 分)
    有关磁盘操作,说法正确的是(ACD)

A .
对磁盘扇区的访问时间包括三个部分中,传送时间最小。

B .
磁盘以字节为单位读写数据

C .
磁盘以扇区为单位读写数据

D .
读写头总处于同一柱面

  • 有关RAM的说法,正确的是(ACDEG)

A .
SRAM和DRAM掉电后均无法保存里面的内容。

B .
DRAM将一个bit存在一个双稳态的存储单元中

C .
一般来说,SRAM比DRAM快

D .
SRAM常用来作高速缓存

E .
DRAM将每一个bit存储为对一个电容充电

F .
SRAM需要不断刷新

G .
DRAM被组织为二维数组而不是线性数组

解析:静态RAM(SRAM)比动态RAM(DRAM)更快。

代码托管

1071521-20171224163403662-1785119167.png

结对及互评

本周结对学习情况

  • 20155326
    • 结对照片

    • 结对学习内容

      • 第十章内容

其他(感悟、思考等,可选)

感觉I/O这块的内容一直都是重点而且无处不在,但是自己一直在这一块的学习上感觉存在不足,借这次机会正好再次认真学习了一遍。

学习进度条

代码行数(新增/累积)博客量(新增/累积)学习时间(新增/累积)重要成长
目标5000行30篇200小时
第一周5/51/115/15
第二周1/223/38
第三周206/3271/328/66
第四周206/3271/410/77
第五周285/5681/520/97主要学习了汇编及反汇编的相关知识
第六周160/6833/820/107
第七周/2/1020/127第四章学习内容和第二次实验
第八周2/1222/149第十一章、第十二章
第九周408/ 25823/1521/170第六章内容、第三次实验、课后pwd的实现
第十一周174 / 30353/1820/200第九章内容、实验四
第十三周741/37762/2220/220第十二章再学习
第十四周87/38631/2320/240第十章再学习

尝试一下记录「计划学习时间」和「实际学习时间」,到期末看看能不能改进自己的计划能力。这个工作学习中很重要,也很有用。
耗时估计的公式
:Y=X+X/N ,Y=X-X/N,训练次数多了,X、Y就接近了。

参考:软件工程软件的估计为什么这么难软件工程 估计方法

转载于:https://www.cnblogs.com/ljq1997/p/8098558.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值