Подразделы

Другие разделы

Дата и время

19/03/2024 14:40:21

Авторизация

Имя:
Пароль:
Зарегистрироваться
Восстановить пароль
 

print2302. (множества) Множество на массиве

print(множества) Множество на массиве

Ограничения: время – 1s/2s, память – 32MiB Ввод: input.txt или стандартный ввод Вывод: output.txt или стандартный вывод copy
Послать решение Blockly Посылки Темы Где Обсудить (0)

Структура "множество" определена в коде.
Элементы множества содержатся в массиве.
Предполагается, что в множестве нет дубликатов!
Требуется реализовать операции над множествами:
Объединение(union), пересечение(cross) и разность (difference)

Используйте следующий шаблон программы:
#include <iostream>
#include <string>

#define MAXSETSIZE 100
#define DIGIT 0
#define COMMA 2
#define DEFIS 3
#define BLANK 4
#define OTHER 5

using namespace std;

//структура множества на массиве
struct SETINARRAY {
	int m[MAXSETSIZE]; // массив, хранящий элементы множества
	int n; // действительное число эл-тов в множестве
};

// Прототипы функций
void Union(SETINARRAY &A, SETINARRAY &B, SETINARRAY &Res); // Res = A union B
void Cross(SETINARRAY &A, SETINARRAY &B, SETINARRAY &Res);// Res = A cross B.
void Difference(SETINARRAY &A, SETINARRAY &B, SETINARRAY &Res); // Разность Res = A \ B.
short SymClass(char z); //Классификатор символов
int StringToIntArray(string &p, int *r); // разбор строки в массив целых чисел
string ArrayToString(int *p, int np); // для вывода

int main() { // Точка входа в программу
	SETINARRAY X, Y, Z;
	string s;
	cin >> s;
	X.n = StringToIntArray(s, X.m);
	cin >> s;
	Y.n = StringToIntArray(s, Y.m);

	Union(X, Y, Z);
	cout << "\n" + ArrayToString(Z.m, Z.n);
	Cross(X, Y, Z);
	cout << "\n" + ArrayToString(Z.m, Z.n);
	Difference(X, Y, Z);
	cout << "\n" + ArrayToString(Z.m, Z.n);
	return 0;
}

void Union(SETINARRAY &A, SETINARRAY &B, SETINARRAY &Res) {
	// НАПИШИТЕ САМИ
	// функция вычисляет объединение множеств a,b
	// результат помещается в *Res
	// предполагается что множества не содержат дубликатов
}

void Cross(SETINARRAY &A, SETINARRAY &B, SETINARRAY &Res) {
	// НАПИШИТЕ САМИ
	// функция вычисляет пересечение множеств a,b
	// результат помещается в *Res
}

void Difference(SETINARRAY &A, SETINARRAY &B, SETINARRAY &Res) {
	// НАПИШИТЕ САМИ
	// функция вычисляет разность множеств a,b
	// результат помещается в *Res
}

short SymClass(char z) {
	if (z >= '0' && z <= '9') {
		return DIGIT;
	}
	if (z == ',') {
		return COMMA;
	}
	if (z == '-') {
		return DEFIS;
	}
	if (z == ' ') {
		return BLANK;
	}
	return OTHER;
}

int StringToIntArray(string &p, int *r) {
	int k, p1 = -1, p2 = -1;
	char buf[20];
	short sClass;
	int i = 0; // номер символа набираемого числа
	int n = 0; // число страниц (текущая длина p)
	int LastPunkt = -1; // последний символ из DEFIS или COMMA
						// -1 значит никакого

	if (!p.size()) {
		return 0;
	}
	for (int j = 0; j < p.size(); j++) {
		sClass = SymClass(p[j]);
		switch (sClass) {
		case DIGIT:
			buf[i++] = p[j];
			break;
		case COMMA:
			buf[i] = '\0';
			if (LastPunkt == -1 || LastPunkt == COMMA) {
				p1 = atoi(buf);
				r[n++] = p1;
			}
			if (LastPunkt == DEFIS) {
				p2 = atoi(buf);
				if (p1 == -1 || p2 == -1) {
					return -1;
				}
				if (p2<p1) {
					return -1;
				}
				for (k = p1; k <= p2; k++) {
					r[n++] = k;
				}
				p1 = p2 = -1;
			}
			i = 0;
			LastPunkt = COMMA;
			break;
		case DEFIS:
			buf[i] = '\0';
			p1 = atoi(buf);
			p2 = -1;
			LastPunkt = DEFIS;
			i = 0;
			break;
		case BLANK:
			break;
		default:
			return -1;
		} // switch
	}
	buf[i] = '\0';
	if (LastPunkt == -1 || LastPunkt == COMMA) {
		p1 = atoi(buf);
		r[n++] = p1;
	}
	if (LastPunkt == DEFIS) {
		p2 = atoi(buf);
		if (p1 == -1 || p2 == -1) {
			return -1;
		}
		for (k = p1; k <= p2; k++) {
			r[n++] = k;
		}
	}
	// сортировка
	bool b = true;
	int t;
	while (b) {
		b = false;
		for (i = 0; i<n - 1; i++) {
			if (r[i]>r[i + 1]) {
				t = r[i];
				r[i] = r[i + 1];
				r[i + 1] = t;
				b = true;
			}
		}
	}
	return n;
}

string ArrayToString(int *p, int np) {
	if (!np) {
		return "Empty";
	}
	int i, cnt = 0;
	string s = to_string(p[0]);
	bool DiffOne, SerStart = false;
	for (i = 1; i<np; i++) {
		DiffOne = (p[i] - p[i - 1] == 1);
		if (!SerStart && DiffOne) {
			SerStart = true;
		}
		if (DiffOne) {
			cnt++;
			if (i == np - 1) {
				if (cnt>1) {
					s += "-";
				}
				else {
					s += ",";
				}
				s += to_string(p[i]);
			}
		}
		else {
			if (cnt <= 1) {
				s += ",";
				s += to_string(p[i]);
			}
			else {
				s += "-";
				s += to_string(p[i - 1]);
				s += ",";
				s += to_string(p[i]);
				cnt = 0;
			}
		}
	}
	return s;
}


Ввод
2 строки в специальном формате: между целочисленными значениями разрешается использовать запятые, или дефис. Другие знаки использовать нельзя.
Дефис свидетельствует о вхождении диапазона в множество и левое значение от дефиса должно быть меньше правого (1-3 корректно, 3-1 – не корректно).
Входные строки оканчиваются числами.
Например, корректный ввод множества "1-3,5,7-9" означает {1, 2, 3, 5, 7, 8, 9}.
Функция преобразования строки помещает отсортированные элементы.
Вывод
3 строки:
Результат объединения множеств
Результат пересечения множеств
Результат разности множеств

Пример ввода

1-3
2

Пример вывода

1-3
2
1,3
loading