11 4
tail的c语言简单实现

下面的实现是来自K&R-C的习题实现:

编写程序tail,将输入中的n行打印出来。默认情况下,n的值为10,但可通过一个可选参数改变n的值,因此命令 tail -n 将打印其输入的最后n行。无论输入或n的值是否合理,该程序都应该能正常运行,编写的程序要充分利用存储空间

下面的例子只简单实现一次输入输出,并不实现系统 tail -f 一样的监控文件的实时输出

tail.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MAXLINE 1000

void tail(int n);
static int mygetline(char *s, int len);

int main(int argc, char *argv[]) {
  int n = 10;
  char *arg;
  while (--argc > 0) {
    arg = *(++argv);
    if (*arg == '-') {
      n = atoi(++arg);
    }
  }
  tail(n);
}

void tail(int n) {
  printf("show %d lines: \n", n);
  int count = 0;
  char *lines[n];

  char s[MAXLINE];
  int w;
  char *p;

  while ((w = mygetline(s, MAXLINE)) != 0) {
    p = malloc(w);
    if (p == NULL) {
      printf("memory alloc error");
      return;
    } else {
      strcpy(p, s);

      if (count >= n) {
        free(lines[count % n]);
      }
      lines[count % n] = p;
      count++;
    }
  }

  int start = 0;
  int end = n;

  if (count >= n) {
    start = count;
    end = count + n;
  }

  for (; start < end; start++) {
    printf("%s\n", lines[start % n]);
    free(lines[start % n]);
  }
}

static int mygetline(char *s, int len) {
  static char c;
  if (c == EOF) {
    return 0;
  }

  int i = 0;

  while ((c = getchar()) != '\n' && c != EOF && i < len - 1) {
    *(s + i++) = c;
  }

  if (c == '\n') {
    *(s + i) = '\n';
  }

  *(s + i) = '\0';
  return i;
}

运行例子:

cc tail.c && ./a.out -10 < input.txt