Печатные комнаты, но доступ только один раз
Я возился со способом напечатать номера этой проблемы. Я использовал символы рисования прямоугольников из расширенного ASCII для проверки этого, поэтому я понимаю, что это ограничивает область, в которой будет компилироваться мой код.
Краткое объяснение проблемы, если вы не хотите читать там:
Комната определяется как 4-битное целое число, такое что
- Самый важный бит показывает, есть ли в комнате западная стена
- Следующий, наиболее важный момент, имеет ли комната северную стену
- Немаловажно, имеет ли комната восточную стену
- Наименее значимый бит показывает, имеет ли комната южную стену
Например, комната, представленная 0b1101, будет иметь западную, северную и южную стены:
┌─
│
└─
Есть дополнительные ограничения:
- Наружные стены комнат всегда будут иметь стену
- Внутренние стены всегда будут выражены в обеих комнатах (пример 0b1101 имеет стену на юг, поэтому в комнате под ней должен быть установлен бит, следующий за наиболее значимым, указывающий на северную стену)
- Там никогда не будет больше, чем
numeric_limits<int>::max()
номера
Моя проблема - выбор символа пересечения. Мой алгоритм перебора должен иметь доступ к каждой комнате дважды (за исключением комнат в первом ряду / столбце.) Есть ли способ найти пересечения только с одним доступом к памяти на комнату?
Если вы хотите увидеть мой код для справки; это принимает в:
-
vector<char>
информации о комнате -
size_t
давая ширину ряда -
vector<int>
с метками для каждой комнаты (это может быть установленоvector(size(testValues), -17)
просто напечатать структуру комнаты без меток
string printArray(const vector<char>& testValues, const size_t width, const vector<int>& indexes) {
if (empty(testValues)) {
return string();
} else {
string result;
auto prevLine = "\xC9\xCD"s;
prevLine.reserve(2U * (1U + width));
for (auto i = 0U; i + 1 < width; ++i) {
if ((testValues[i] & 0b10) != 0) {
prevLine += "\xD1\xCD"s;
} else {
prevLine += "\xCD\xCD"s;
}
}
prevLine += "\xBB\n"s;
result.reserve(size(prevLine) * (1U + 2U * size(testValues) / width));
for (auto i = 0U; i < size(testValues) - width; ++i) {
const auto x = i % width;
const auto isBottomSet = (testValues[i] & 0b1) != 0;
if (x == 0) {
result += (prevLine + '\xBA') + static_cast<char>('0' + indexes[i]);
prevLine = isBottomSet ? "\xC7\xC4"s : "\xBA "s;
}
if (x + 1U == width) {
result += "\xBA\n"s;
prevLine += isBottomSet ? "\xB6\n"s : "\xBA\n"s;
} else {
const auto isRightSet = (testValues[i] & 0b10) != 0;
const size_t index = static_cast<int>(isRightSet) << 3 | testValues[i + width + 1] & 0b100 | (testValues[i + width + 1] & 0b1000) >> 2 | static_cast<int>(isBottomSet);
// MSB: isAboveIntersectionSet
// isRightOfIntersectionSet
// isBelowIntersectionSet
// LSB: isLeftOfIntersectionSet
constexpr const char* getIntersection[] = { " ", // 0b0
" ", // 0b1
" ", // 0b10
"\xBF ", // 0b11
" \xC4", // 0b100
"\xC4\xC4", // 0b101
"\xDA\xC4", // 0b110
"\xC2\xC4", // 0b111
" ", // 0b1000:
"\xD9 ", // 0b1001
"\xB3 ", // 0b1010
"\xB4 ", // 0b1011
"\xC0\xC4", // 0b1100
"\xC1\xC4", // 0b1101
"\xC3\xC4", // 0b1110
"\xC5\xC4" }; // 0b1111
result += { isRightSet ? '\xB3' : ' ', static_cast<char>('0' + indexes[i + 1]) };
prevLine += getIntersection[index];
}
}
result += (prevLine + '\xBA') + static_cast<char>('0' + indexes[size(testValues) - width]);
prevLine = "\xC8\xCD"s;
for (auto i = size(testValues) - width; i + 1 < size(testValues); ++i) {
if ((testValues[i] & 0b10) != 0) {
result += { '\xB3', static_cast<char>('0' + indexes[i + 1]) };
prevLine += "\xCF\xCD"s;
} else {
result += { ' ', static_cast<char>('0' + indexes[i + 1]) };
prevLine += "\xCD\xCD"s;
}
}
return result + "\xBA\n"s + prevLine + '\xBC';
}
}
Если вы заинтересованы в простом тесте для этого, вы можете сделать:
const vector<char> rooms = { 0b1101, 0b110, 0b1101, 0b110, 0b1100, 0b101, 0b110,
0b1110, 0b1001, 0b110, 0b1011, 0b1010, 0b1111, 0b1010,
0b1000, 0b101, 0b11, 0b1110, 0b1011, 0b1110, 0b1010,
0b1011, 0b1101, 0b101, 0b1, 0b101, 0b11, 0b1011 };
const vector<int> indexes = { 1, 1, 2, 2, 3, 3, 3,
1, 1, 1, 2, 3, 5, 3,
1, 1, 1, 6, 3, 6, 3,
1, 6, 6, 6, 6, 6, 3 };
cout << printArray(rooms, width, indexes) << endl;
Чтобы запустить это, вам нужно перейти на http://webcompiler.cloudapp.net/ и вставить все в.