usaa24/cv7/program.c
2024-11-08 11:09:32 +01:00

137 lines
3.8 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_LINE_LENGTH 100
typedef struct Node {
char text[MAX_LINE_LENGTH];
struct Node *yes;
struct Node *no;
} Node;
// Создает новый узел с заданным текстом
Node* create_node(const char* text) {
Node *node = (Node*)malloc(sizeof(Node));
if (!node) {
fprintf(stderr, "Memory allocation error.\n");
exit(1);
}
strncpy(node->text, text, MAX_LINE_LENGTH);
node->yes = NULL;
node->no = NULL;
return node;
}
// Создает базу знаний, если файл не существует
void create_default_knowledge_base() {
FILE *file = fopen("knowledge_base.txt", "w");
if (!file) {
perror("Failed to create the knowledge base file");
exit(1);
}
fprintf(file, "Rastie to na strome?\n");
fprintf(file, "*Jablko\n");
fprintf(file, "Rastie to pod zemou?\n");
fprintf(file, "*Mrkva\n");
fprintf(file, "*Šalát\n");
fclose(file);
}
// Считывает базу знаний из файла и создает дерево
Node* parse_knowledge_base(FILE *file) {
char line[MAX_LINE_LENGTH];
if (!fgets(line, sizeof(line), file) || line[0] == '\n') {
return NULL;
}
line[strcspn(line, "\n")] = 0; // Удаляем символ новой строки
Node *node;
if (line[0] == '*') {
node = create_node(line + 1); // Создаем узел-ответ
} else {
node = create_node(line); // Создаем узел-вопрос
node->yes = parse_knowledge_base(file); // Рекурсивно читаем ветку "да"
node->no = parse_knowledge_base(file); // Рекурсивно читаем ветку "нет"
}
return node;
}
// Подсчитывает количество листовых узлов (продуктов)
int count_products(Node *node) {
if (!node) return 0;
if (!node->yes && !node->no) return 1; // Листовой узел - это продукт
return count_products(node->yes) + count_products(node->no);
}
// Запускает экспертную систему
void run_expert_system(Node *node) {
while (node) {
if (!node->yes && !node->no) {
printf("*%s\n", node->text);
printf("Koniec\n");
return;
}
printf("Odpovedajte 'a' pre prvu moznost alebo 'n' pre druhu moznost.\n");
printf("%s\n", node->text);
char answer;
if (scanf(" %c", &answer) != 1) {
printf("Koniec\n");
return;
}
if (answer == 'a') {
node = node->yes;
} else if (answer == 'n') {
node = node->no;
} else {
printf("Nespravny vstup.\nKoniec\n");
return;
}
}
}
// Освобождает выделенную память для дерева
void free_tree(Node *node) {
if (node) {
free_tree(node->yes);
free_tree(node->no);
free(node);
}
}
int main() {
FILE *file = fopen("knowledge_base.txt", "r");
if (!file) {
// Если файла нет, создаем его
create_default_knowledge_base();
file = fopen("knowledge_base.txt", "r");
if (!file) {
perror("Failed to open the knowledge base file");
return 1;
}
}
Node *root = parse_knowledge_base(file);
fclose(file);
if (!root) {
printf("Báza znalostí sa nedá načítať.\n");
return 1;
}
printf("Expert z bufetu to vie.\n");
int product_count = count_products(root);
printf("Pozna %d druhov ovocia a zeleniny.\n", product_count);
run_expert_system(root);
free_tree(root);
return 0;
}