OC 日志
在本项目中,想要实现一类功能,即启动后,一切输入输出流包括错误流全部自动的写入到一个文件中,在百度了各种OC的方法,基本上都是设置导出到一个指定目录,然后调用C语言的freopen进行重定向,使用C语言的freopen重定向和对NSLog的重定向设置方法有一个区别,就是freopen除了NSLog,C语言的输出方式printf也可以做到,而在串口交互的时候,由于传递进来的直接就是字符流,因此必须使用类似freopen的方法进行兼容操作。
objective-c
//OC代码,重定向,其实就是自己处理了一下文件路径然后调用了C语言
freopen([logFilePath cStringUsingEncoding:NSASCIIStringEncoding],"a+", stdout);
但是一旦进行了重定向,原本输出给控制台端的交互显示就全部消失了,这样就将程序编程了盲打一般,搜索了半天由于,现在解决这个问题做的如下笔记:
freopen
首先来翻阅freopen的文档
FILE *freopen(const char *filename, const char *mode, FILE *stream)
传入参数
名称
解释
mode
是文件的访问模式,同fopen()函数,主要有【”r”,”w”,”a”,”r+”,”w+”,”a+”】,具体意思不做解释
stream
指向FILE对象的指正,该FILE对象表示了要被重新打开的流
filename
是需要打开传入的文件地址
传出参数
如果文件可以成功打开,则函数返回一个指针,否则返回NULL。
其内部实现代码为:
FILE* Freopen(const char *fileName, const char *type, FILE *stream){
FILE *fileFp = fopen(fileName, type);
int fd1 = fileno(fileFp);
int fd2 = fileno(stream);
if(dup2(fd1, fd2) < 0) return NULL;
else return stream;
}
利用fileno取出文件描述符,对于fileName采用新创建的方法(也难怪需要提前判断文件是否存在,否则则删除文件重新创建),这里主要调用到了dup2这个函数,有关dup,dup2这两个函数,搜索到的笔记又如下:
dup
dup的定义相当简单
int dup(int oldfd);
dup用来复制参数oldfd所指的文件描述符。当复制成功时,返回最小的尚未被使用过的文件描述符,若有错误则返回-1.错误代码存入errno中。返回的新文件描述符和参数oldfd指向同一个文件,这两个描述符共享同一个数据结构,共享所有的锁定,读写指针和各项全现或标志位。
而调用dup也等同于调用fcntl(oldfd, F_DUPFD, 0);这个方法(后面介绍)。
一个代码示例如下:
#include
#include
#include
#include
int main(int argc, char* argv[])
{
int fd=open("text.txt", O_CREAT|O_RDWR|O_TRUNC, S_IRUSR|S_IWUSR);
if(fd < 0)
{
printf("Open Error!\n");
return 0;
}
int fd2=dup(fd);
if(fd2<0)
{
printf("Error!\n");
return 0;
}
char buf[1000];
int n;
//接受键盘输入,并将其存入buf所指向的缓存中
while((n=read(STDIN_FILENO, buf,1000)) > 0 )
{
printf("%s\n", buf);
//将buf所指向的缓存区的n个字节的数据写入到由文件描述符fd2所指示的文件中
if(write(fd2, buf, n)
{
printf("Write Error!!\n");
return 0;
}
}
return 0;
}
测试输出代码如下(GCC编译,非Clang)
(base) mt-007deMac-mini:Test mt-007$ ./redup
xxxxxxxxman
xxxxxxxxman
^C
(base) mt-007deMac-mini:Test mt-007$ cat text.txt
xxxxxxxxman
(base) mt-007deMac-mini:Test mt-007$
可以看到fd将描述指向了text.txt,而fd2经过dup操作,相当于做了一次弱拷贝,以后我们对fd2进行的操作就相当于对fd进行的操作一样,此时fd这个标识符也依旧存在。
dup2
dup2的定义是在dup1的基础上发生的改变
int dup2(int oldfd, int newfd);
dup2与dup区别是dup2可以用参数newfd指定新文件描述符的数值。若参数newfd已经被程序使用,则系统就会将newfd所指的文件关闭,若newfd等于oldfd,则返回newfd,而不关闭newfd所指的文件。dup2所复制的文件描述符与原来的文件描述符共享各种文件状态。共享所有的锁定,读写位置和各项权限或flags等。
若dup2调用成功则返回新的文件描述符,调用失败则返回-1.
一个测试代码如下:
#include
#include
#include
#include
#include
int main()
{
int oldfd;
int fd;
int t;
char *buf="This is a test!!!!\n";
if((oldfd=open("orther.txt",O_RDWR|O_CREAT,0644))==-1) {
printf("open error\n");
exit(-1);
}
fd=dup2(oldfd,fileno(stdout));
if(fd==-1) {
printf("dup2 error\n");
exit(-1);
}
printf("dup2的返回值:%d\n",fd);
t=strlen(buf);
//本应该写入到stdout的信息,但是标准输出已经重定向到目标文件中,故向标准输出写的数据将会写到目标文件中。
if(write(fileno(stdout),buf,t)!=t) {
printf("write error!\n");
exit(-1);
}
close(fd);
return 0;
}
可以结合下面的编译内容,可以看得出dup2主要用来重定向,将oldFild直接取消,而将原有的字节流全部传递给newFile。
(base) mt-007deMac-mini:Test mt-007$ ./redup2
(base) mt-007deMac-mini:Test mt-007$ ls
orther.txtredup.credup2redup2.ctext.txt
(base) mt-007deMac-mini:Test mt-007$ cat orther.txt
This is a test!!!!
(base) mt-007deMac-mini:Test mt-007$
特殊备注
有的时候运行C语言的FILE操作无法读取文件,但是使用OC的操作调用C的代码就可以读取,这大概率是因为权限问题,一般而言文件路径地址该用OC的NSString处理就用这个处理。
Post Views:
145