C语言家教记录(八)

C语言家教记录(八)

  • 导语
  • 指针的高级应用
    • 动态分配、使用、释放
    • 链表
    • 指向指针的指针
    • 指向函数的指针
  • 流和文件
    • 文件操作
    • 变量和格式化io
  • 总结和复习

导语

本次授课的内容如下:指针的高级应用,流和文件

辅助教材为 《C语言程序设计现代方法(第2版)》

指针的高级应用

动态分配、使用、释放

malloc,calloc(一般不用),realloc(一般不用)
null 空指针,也可以用0

null的使用

if(p==NULL)
if(!p)
if(p!=NULL)
if(p)

malloc的使用

p=malloc(n+1);//不常用
p=(char*)malloc(n+1);//常用,malloc的分配单位是字节

int* a;
a=malloc(n*sizeof(int));
for(int i=0;i<n;i++)a[i]=0;

calloc的使用

a=calloc(n,sizeof(int));//自动初始化
struct point { int x, y; } *p;
p = calloc(1, sizeof(struct point));//x,y都为0

realloc的使用,规则见书

realloc(p,n*sizeof(int));

释放存储空间,例子见书

p=malloc();
q=malloc();
free(p);
p=q;

char*p=malloc(...);
free(p);
strcpy(p,"abc");//报错,悬空指针问题

示例程序

char *concat(const char *s1, const char *s2)
{
 char *result ;
 result = malloc(strlen(s1) + strlen(s2) + 1);
 if (result == NULL) {
 printf("Error: malloc failed in concat\n");
 exit (EXIT_FAILURE);
 }
 strcpy(result, s1);
 strcat(result, s2);
 return result;
}
p=concat("abs","def");

链表

概念和例子见书

struct node
{
	int value;
	struct node *next;
};//链表的节点定义
struct node* first =NULL;

//节点的创建,图见书
struct node* p;
p=malloc(sizeof(struct node));
(*p).value=10;
//等价于
p->value=10;

示例程序

解释插入链表的实现,解释图

struct node *add_to_list(struct node *list, int n)
{
 struct node *new_node;
 new_node = malloc(sizeof(struct node));
 if (new_node == NULL) {
 printf("Error: malloc failed in add_to list\n");
 exit(EXIT_FAILURE);
 }
 new_node->value = n;
 new_node->next = list;
 return new_node;
}

搜索链表,用书上例子解释

struct node *search_list(struct node *list, int n)
{
 struct node *p;
 for (p = list; p != NULL; p = p->next)
 if (p->value == n)
 return p;
 return NULL;
}

struct node *search_list(struct node *list, int n)
{
 while (list != NULL && list->value != n)
 list = list->next;
 return list;
}

删除节点,根据书上例子

struct node *delete_from_list(struct node *list, int n)
{
 struct node *cur, *prev;
 for (cur = list, prev = NULL;
 cur != NULL && cur->value != n;
 prev = cur, cur = cur->next)
 ;
 if (cur == NULL)
 return list; /* n was not found */
 if (prev == NULL)
 list = list->next; /* n is in the first node */
 else
 prev->next = cur->next; /* n is in some other node */
 free (cur);
 return list;
}

指向指针的指针

如果不进行返回,添加结点可能会失效,见书上例子

void_add_to_list(struct node **list, int n)
{
 struct node *new_node;
 new_node = malloc(sizeof(struct node));
 if (new_node == NULL) {
 printf("Error: malloc failed in add_to_list\n");
 exit(EXIT_FAILURE);
 }
 new_node->value = n;
 new_node->next = *list;
 *list = new_node;
}

指向函数的指针

void (*pf)(int);
pf=f;//赋值
(*pf)(i);//调用
pf(i);//调用
//pf 可以指向任何带有int 型形式参数并且返回void 型值的函数

double integrate(double (*f)(double), double a, double b);
double integrate(double f(double), double a, double b);//等价

result=integrate(sin,0.0,PI/2);

integrate内可以调用
y=(*f)(x);

qsort函数

void qsort(void *base, size_t nmemb, size_t size,
 int (*compar) (const void *, const void *));//函数原型

qsort(inventory, num_parts, sizeof(struct part), compare_parts);//调用格式

int compare_parts(const void *p, const void *q)
{
 const struct part *p1 = p;
 const struct part *q1 = q;
 if (p1->number < q1->number)
 return -1;
 else if (p1->number == q1->number)
 return 0;
 else
 return 1;
}
//等价于
int compare_parts(const void *p, const void *q)
{
 return strcmp(((struct part *) p)->name,
 ((struct part *) q)->name);
}

流和文件

stdin,stdout,stderr简单介绍

简单介绍二进制文件

简单介绍输入输出重定向和命令行

文件操作

fopen,fclose,freopen等
tmpfile,fflush,remove,rename跳过

FILE *fopen(const char * restrict filename, const char * restrict
 mode);//函数原型

fp=fopen("a.txt","r");//路径和模式,运行之后输入从键盘改成文件
//解释参数r w a r+ w+ a+,参数可以组合

int fclose(FILE *stream);//关闭文件

FILE *freopen(const char * restrict filename, const char * restrict mode, FILE * restrict stream);//重定向

if (freopen("foo","w", stdout) == NULL) {
 /* error; foo can't be opened */
}//最好先关闭再重定向

//解释从命令行获取文件名
int main(int argc, char *argv[])
{
 ...
}

示例程序

#include 
#include 
#define FILE_NAME "example.dat"
int main(void)
{
 FILE *fp;
 fp = fopen(FILE_NAME, "r");
 if (fp == NULL) {
 printf("Can't open %s\n", FILE_NAME);
 exit(EXIT_FAILURE);
 }
 ...
 fclose(fp);
 return 0;
}

#include 
#include 
int main(int argc, char *argv[])
{
 FILE *fp;
 if (argc != 2) {
 printf("usage: canopen filename\n");
 exit (EXIT_FAILURE);
 }
 if ((fp = fopen(argv[1], "r")) == NULL) {
 printf("%s can't be opened\n", argv[1]);
 exit (EXIT_FAILURE);
 }
 printf("%s can be opened\n", argv[1]);
 fclose(fp);
 return 0;
}

变量和格式化io

简介fprintf和printf,见书

简介fscanf和scanf,见书

简介fputc,putc,putchar,fgets,gets

sprintf,snprintf,针对字符数组的输出

sscanf,针对字符数组的输入

文件定位和块输入输出跳过,简单勾一下

示例程序

#include 
#include 
int main(int argc, char *argv[])
{
 FILE *source_fp, *dest_fp;
 int ch;
 if (argc != 3) {
 fprintf(stderr, "usage: fcopy source dest\n");
 exit(EXIT_FAILURE);
 }
 if ((source_fp = fopen(argv[1], "rb")) == NULL) {
 fprintf (stderr, "Can't open %s\n", argv[1]);
 exit(EXIT_FAILURE);
 }
 if ((dest_fp = fopen(argv[2], "wb")) == NULL) {
 fprintf (stderr, "Can't open %s\n", argv[2]);
 fclose(source_fp);
 exit(EXIT_FAILURE);
 }
 while ((ch = getc(source_fp)) != EOF)
 putc (ch, dest_fp);
 fclose(source_fp);
 fclose(dest_fp);
 return 0;
}

总结和复习

本次授课讲述第17章和第22章内容,关键点:指针的高级应用和输入输出

你可能感兴趣的:(C语言家教记录,c语言,windows,开发语言)