post on 16 Oct 2018 about 7540words require 26min
CC BY 4.0 (除特别声明或转载文章外)
如果这篇博客帮助到你,可以请我喝一杯咖啡~
项目名称:某科学的超会议议程管理系统(Scientific Meeting Management System) 项目介绍:基于命令行,实现添加(add)、删除(delete)、修改(modify)、查询(query)等议程管理功能。
以下示意图中以SMMS(Scientific Meeting Management System)代表本系统,LOG代表本地缓存文件log.txt
,STACK代表用于议程管理的栈结构(见后)。
sequenceDiagram
USER->>SMMS:输入
LOG->>SMMS:读档
SMMS->>STACK:关键字搜索、操作
SMMS->>LOG:存档
SMMS->>USER:输出
增加会议时只需加入栈顶,时间复杂度为 O(1)。 修改或删除会议时,只需将该元素和栈顶元素交换,然后对栈顶元素进行操作,时间复杂度仍为 O(1)。 清空议程时,只需将栈顶指针指向栈底而无需在数组中擦除数据,时间复杂度仍为 O(1)。
定义会议 n 和 m 相关,当且仅当满足下列任意条件:
- n 和 m 同名
- n 和 m 时间有重合部分,并且地点相同或有相同人员(换言之两场会议是否不能同时举行)
基于会议相关的定义,可以判断一个会议是否能够加入议程;也可以像下面一样进行高效严密的关键字搜索。 例如:
所有函数接口均使用文件流FILE*
,为快速切换文件\屏幕 IO 提供很大便利。
与此相对应的,所有的 IO 均通过函数 fscanf 和 fprintf 实现。
所有的 IO 均写有提示,方便使用。
同文件夹下log.txt
文档中保存了用于测试的数据,运行结果均符合预期。
代码实现分为数据结构和功能函数两个模块,分别对其展开介绍。所有头文件均写有头文件保护。
struct tm
是 c 标准库<time.h>
中定义的用于保存时间的结构体。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#define MAXN 128
typedef struct Meeting
{
int num_of_participants;//参会人数
char name[MAXN],//会议名称
address[MAXN],//会议地址
participants[MAXN][MAXN];//参会人员
struct tm begin,end;//起止时间
} Meeting;
//从in中读入一个Meeting,并将输入时的提示信息输出到out
Meeting getMeeting(FILE *in,FILE *out);
//把m输出到out
void putMeeting(FILE *out,const Meeting *m);
//比较两个时间a和b的先后关系
int tmCmp(const struct tm *a,const struct tm *b);
//判断会议m和n是否冲突
int isRepel(const Meeting *m,const Meeting *n);
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
Meeting getMeeting(FILE *in,FILE *out)
{
Meeting tmp_m;
fprintf(out,"Input the Meeting name.\n");
fscanf(in,"%s",tmp_m.name);
fprintf(out,"Input the Meeting address.\n");
fscanf(in,"%s",tmp_m.address);
fprintf(out,"Input the begin time(year month day hour minute).\n");
fscanf(in,"%d%d%d%d%d",
&tmp_m.begin.tm_year,
&tmp_m.begin.tm_mon,
&tmp_m.begin.tm_mday,
&tmp_m.begin.tm_hour,
&tmp_m.begin.tm_min);
fprintf(out,"Input the end time(year month day hour minute).\n");
fscanf(in,"%d%d%d%d%d",
&tmp_m.end.tm_year,
&tmp_m.end.tm_mon,
&tmp_m.end.tm_mday,
&tmp_m.end.tm_hour,
&tmp_m.end.tm_min);
fprintf(out,"Input the number of participants.\n");
fscanf(in,"%d",&tmp_m.num_of_participants);
for(int i=0; i<tmp_m.num_of_participants; ++i)
{
fprintf(out,"Input the name of participant %d.\n",i);
fscanf(in,"%s",tmp_m.participants[i]);
}
return tmp_m;
}
void putMeeting(FILE *out,const Meeting *m)
{
fprintf(out,"%s %s\n%d %d %d %d %d\n%d %d %d %d %d\n%d",
m->name,
m->address,
m->begin.tm_year,
m->begin.tm_mon,
m->begin.tm_mday,
m->begin.tm_hour,
m->begin.tm_min,
m->end.tm_year,
m->end.tm_mon,
m->end.tm_mday,
m->end.tm_hour,
m->end.tm_min,
m->num_of_participants);
for(int j=0; j< m->num_of_participants; ++j)
fprintf(out," %s",m->participants[j]);
fprintf(out,"\n");
}
int tmCmp(const struct tm *a,const struct tm *b)
{
if(a->tm_year != b->tm_year)
return a->tm_year - b->tm_year;
if(a->tm_mon != b->tm_mon)
return a->tm_mon - b->tm_mon;
if(a->tm_mday != b->tm_mday)
return a->tm_mday - b->tm_mday;
if(a->tm_hour != b->tm_hour)
return a->tm_hour - b->tm_hour;
return a->tm_min - b->tm_min;
}
int isRepel(const Meeting *p,const Meeting *q)
{
if(!strcmp(p->name,q->name))
return 1;
if(tmCmp(&p->end,&q->begin)<0||tmCmp(&p->begin,&q->end)>0)
return 0;
if(!strcmp(p->address,q->address))
return 1;
for(int i=0; i < p->num_of_participants; ++i)
for(int j=0; j < q->num_of_participants; ++j)
if(!strcmp(p->participants[i],q->participants[j]))
return 1;
return 0;
}
(FILE *in,FILE *out)
表示从 in 中获取数据,并通过 out 输出信息。这样的设计既方便了代码的调试,也方便了用户的交互。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
extern int top;
extern Meeting stack[MAXN];
//读入会议,若不产生冲突则加入议程,否则报错
void add(FILE *in,FILE *out);
//按关键字删除会议
void del(FILE *in,FILE *out);
//按关键字修改会议
void modify(FILE *in,FILE *out);
//按关键字查询会议
void query(FILE *in,FILE *out);
//读档
void fin(FILE *in,FILE *out);
//存档
void fout(FILE *in,FILE *out);
//清空
void clear(FILE *in,FILE *out);
//输出提示信息
void help(FILE *in,FILE *out);
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
int top=0;
Meeting stack[MAXN];
void add(FILE *in,FILE *out)
{
stack[top]=getMeeting(in,out);
for(int i=0; i<top; ++i)
if(isRepel(&stack[top],&stack[i]))
{
fprintf(out,"Error: Repel with Meeting %s.\n",stack[i].name);
return;
}
fprintf(out,"Add %s successfully.\n",stack[top++].name);
}
void del(FILE *in,FILE *out)
{
fprintf(out,"Input the key(-1 for no key).\n");
Meeting key=getMeeting(in,out);
for(int i=0; i<top; ++i)
if(isRepel(&key,&stack[i]))
{
fprintf(out,"Delele %s successfully.\n",stack[i].name);
stack[i--]=stack[--top];
}
}
void modify(FILE *in,FILE *out)
{
int t_len=top;
del(in,out);
while(top<t_len)
add(in,out);
}
void query(FILE *in,FILE *out)
{
fprintf(out,"Input the key(-1 for no key).\n");
Meeting key=getMeeting(in,out);
for(int i=0; i<top; ++i)
if(isRepel(&key,&stack[i]))
{
fprintf(out,"Find ");
putMeeting(out,&stack[i]);
}
}
void fin(FILE *in,FILE *out)
{
fprintf(out,"Start input from \'log.txt\'.\n");
FILE *f=fopen("log.txt","r");
if(f==NULL)
{
fprintf(out,"Error:Do not find \'log.txt\'\n");
return;
}
int t_len;
fscanf(f,"%d",&t_len);
while(t_len--)
add(f,out);
fclose(f);
fprintf(out,"Input from \'log.txt\' successfully.\n");
}
void fout(FILE *in,FILE *out)
{
fprintf(out,"Start output to \'log.txt\'.\n");
FILE *f=fopen("log.txt","w");
fprintf(f,"%d\n",top);
for(int i=0; i<top; ++i)
putMeeting(f,&stack[i]);
fclose(f);
fprintf(out,"Output to \'log.txt\' successfully.\n");
}
void clear(FILE *in,FILE *out)
{
top=0;
fprintf(out,"Clear successfully.\n");
}
void help(FILE *in,FILE *out)
{
fprintf(out,"fin : add new meetings from \'log.txt\' to memory if they do not repel.\n");
fprintf(out,"fout : output all the meetings from memory to \'log.txt\'.\n");
fprintf(out,"add : add a new meeting to memory if it does not repel.\n");
fprintf(out,"delete : delete all the meetings which repel with the key.\n");
fprintf(out,"modify : modify all the meetings which repel with the key.\n");
fprintf(out,"clear : clear all the meetings from memory.\n");
fprintf(out,"help : get available instructions.\n");
fprintf(out,"EOF(Ctrl+Z in Windows) : exit.\n");
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
#include<stdio.h>
#include<string.h>
#include"func.h"
int main()
{
FILE *in=stdin,*out=stdout;
fprintf(out,"Scientific Meeting Management System by WuK&LSY\n\n");
fprintf(out,"Input \'fin\' to read from log,or \'help\' to get other available instructions.\n\n");
for(char s[MAXN]; fscanf(in,"%s",s)!=EOF; fprintf(out,"\n"))
{
if(!strcmp(s,"add"))
add(in,out);
else if(!strcmp(s,"delete"))
del(in,out);
else if(!strcmp(s,"modify"))
modify(in,out);
else if(!strcmp(s,"query"))
query(in,out);
else if(!strcmp(s,"fin"))
fin(in,out);
else if(!strcmp(s,"fout"))
fout(in,out);
else if(!strcmp(s,"clear"))
clear(in,out);
else if(!strcmp(s,"help"))
help(in,out);
else fprintf(out,"\'%s\' is not an available instruction,and you can in \'help\' to get available instructions.\n",s);
}
}
Related posts