実験内容
高度:
開いた
クリエイトとオープンの関係
コード
// from:https://www.cnblogs.com/cxy1616/p/6063463.html
#include <stdio.h>//标准输入输出
#include <stdlib.h>//C标准函数库
#include <unistd.h>//Unix类系统定义符号常量
#include <fcntl.h>//定义了很多宏和open,fcntl函数原型
#define BUFFERSIZE 4096//定义存储器容量
#define COPYMODE 0644// 八进制,权限rwx,0表示八进制
void oops(const char *,const char *);
int main(int argc, char *argv[])
{
int in_fd, out_fd, n_chars;//三个描述符值
char buf[BUFFERSIZE];//存储器位置
/*cp的参数有两个,分别是要复制的文件,和目的目录,这样一共应该是有三个操作数
所以要先检查argc的值是否为三,如果不是,返回标准错误*/
if (argc != 3)
{
fprintf(stderr, "usage: %s source destination\n", *argv);
exit(1);
}
/*检查cp的第一个参数,要复制的文件,用open打开,in_fd为open返回的描述符
如果返回-1,代表打开失败,提示错误*/
if ((in_fd = open(argv[1], O_RDONLY)) == -1)
{
oops("Cannot open ", argv[1]);
}
/*检查cp的第二个参数,复制的目的地址,用create在目的地址创建新文件,out_fd为open返回的描述符
如果返回-1,代表创建失败,提示错误*/
if ((out_fd = creat(argv[2], COPYMODE)) == -1)
oops("Cannot creat", argv[2]);
/*cp指令的动作就是读取一个文件的内容到存储器,在新的地址创建空白文件,再从存储器将内容写入新文件。
这里判断复制是否成功:
如果能读取顺利,而读取的位数和写的位数不同,是写错误;
如果读取失败,是读错误。*/
while ((n_chars = read(in_fd, buf, BUFFERSIZE)) > 0)
if (write(out_fd, buf, n_chars) != n_chars)
oops("Write error to ", argv[2]);
if (n_chars == -1)
oops("Read error from ", argv[1]);
/*这里执行的是关闭文件的动作,in_fd和out_fd两个文件描述符
所指向的文件只要有一个关闭错误,就提示关闭错误。*/
if (close(in_fd) == -1 || close(out_fd) == -1)
oops("Error closing files", "");
}
/*这个是用来输出错误信息的函数*/
void oops(const char *s1,const char *s2)
{
fprintf(stderr, "Error: %s", s1);
perror(s2);//把一个描述性错误消息输出到标准错误 stderr
exit(1);
}
高度なコード
// from:https://www.cnblogs.com/cxy1616/p/6063463.html
#include <stdio.h>//标准输入输出
#include <stdlib.h>//C标准函数库
#include <unistd.h>//Unix类系统定义符号常量
#include <fcntl.h>//定义了很多宏和open,fcntl函数原型
#include <sys/stat.h>
#define BUFFERSIZE 4096//定义存储器容量
#define COPYMODE 0644// 八进制,权限rwx,0表示八进制
void oops(const char *,const char *);
int main(int argc, char *argv[])
{
int in_fd, out_fd, n_chars;//三个描述符值
char buf[BUFFERSIZE];//存储器位置
/*cp的参数有两个,分别是要复制的文件,和目的目录,这样一共应该是有三个操作数
所以要先检查argc的值是否为三,如果不是,返回标准错误*/
if (argc != 3)
{
fprintf(stderr, "usage: %s source destination\n", *argv);
exit(1);
}
/* 判断源文件是否存在 */
/*检查cp的第一个参数,要复制的文件,用open打开,in_fd为open返回的描述符
如果返回-1,代表打开失败,提示错误*/
if ((in_fd = open(argv[1], O_RDONLY)) == -1)
{
oops("Cannot open ", argv[1]);
}
else
{
/* 判断 源文件是否是目录 */
struct stat s;
stat(argv[1], &s);
if(s.st_mode & S_IFDIR)
{
// oops("源文件是目录 ", argv[1]);
printf("Error: 源文件是目录: %s\n", argv[1]);
exit(1);
}
}
/*检查cp的第二个参数,复制的目的地址,用create在目的地址创建新文件,out_fd为open返回的描述符
如果返回-1,代表创建失败,提示错误*/
/*
如果文件事先已经存在,
open(pathname, O_RDWR | O_CREAT,0666); 打开成功,返回一个大于0的fd
open(pathname, O_RDWR | O_CREAT | O_EXCL, 0666); 打开失败,返回-1
O_EXCL表示的是:如果使用O_CREAT时文件存在,就返回错误信息,它可以测试文件是否存在。
*/
if ((out_fd = open(argv[2], O_RDWR | O_CREAT | O_EXCL, 0666)) == -1)
{
printf("目标文件已存在,是否覆盖1 or 合并2?");
char op[2];
scanf("%s", op);
if(op[0] == '1') out_fd = open(argv[2], O_RDWR | O_TRUNC, 0666);
else out_fd = open(argv[2], O_RDWR | O_APPEND, 0666);
}
else
{
if ((out_fd = creat(argv[2], COPYMODE)) == -1)
oops("Cannot creat", argv[2]);
}
/*cp指令的动作就是读取一个文件的内容到存储器,在新的地址创建空白文件,再从存储器将内容写入新文件。
这里判断复制是否成功:
如果能读取顺利,而读取的位数和写的位数不同,是写错误;
如果读取失败,是读错误。*/
while ((n_chars = read(in_fd, buf, BUFFERSIZE)) > 0)
if (write(out_fd, buf, n_chars) != n_chars)
oops("Write error to ", argv[2]);
if (n_chars == -1)
oops("Read error from ", argv[1]);
/*这里执行的是关闭文件的动作,in_fd和out_fd两个文件描述符
所指向的文件只要有一个关闭错误,就提示关闭错误。*/
if (close(in_fd) == -1 || close(out_fd) == -1)
oops("Error closing files", "");
}
/*这个是用来输出错误信息的函数*/
void oops(const char *s1,const char *s2)
{
fprintf(stderr, "Error: %s", s1);
perror(s2);//把一个描述性错误消息输出到标准错误 stderr
exit(1);
}