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<stdio.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
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)<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 <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>

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.txt redup.c redup2 redup2.c text.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处理就用这个处理。


0 条评论

发表评论

您的电子邮箱地址不会被公开。 必填项已用*标注