Самый простой способ - это создать три массива для сложных случаев и использовать простую функцию, например:
// convertToRoman:
// In: val: value to convert.
// res: buffer to hold result.
// Out: n/a
// Cav: caller responsible for buffer size.
void convertToRoman (unsigned int val, char *res) {
char *huns[] = {"", "C", "CC", "CCC", "CD", "D", "DC", "DCC", "DCCC", "CM"};
char *tens[] = {"", "X", "XX", "XXX", "XL", "L", "LX", "LXX", "LXXX", "XC"};
char *ones[] = {"", "I", "II", "III", "IV", "V", "VI", "VII", "VIII", "IX"};
int size[] = { 0, 1, 2, 3, 2, 1, 2, 3, 4, 2};
// Add 'M' until we drop below 1000.
while (val >= 1000) {
*res++ = 'M';
val -= 1000;
}
// Add each of the correct elements, adjusting as we go.
strcpy (res, huns[val/100]); res += size[val/100]; val = val % 100;
strcpy (res, tens[val/10]); res += size[val/10]; val = val % 10;
strcpy (res, ones[val]); res += size[val];
// Finish string off.
*res = '\0';
}
Это будет обрабатывать любое целое число без знака, хотя большие числа будут иметь огромное количество M
символов спереди, и вызывающий должен убедиться, что их буфер достаточно велик.
Как только число было уменьшено ниже 1000, это простой поиск по 3 таблицам, по одному на сотни, десятки и единицы. Например, возьмем случай, когда val
равно 314
.
val/100
в этом случае будет 3
, поэтому поиск в массиве huns
даст CCC
, тогда val = val % 100
даст вам 14
для поиска tens
.
Тогда val/10
будет 1
в этом случае, поэтому поиск в массиве tens
даст X
, тогда val = val % 10
даст вам 4
для поиска ones
.
Тогда val
будет 4
в этом случае, поэтому поиск массива ones
даст IV
.
Это дает вам CCCXIV
для 314
.
Версия с проверкой переполнения буфера является простым шагом:
// convertToRoman:
// In: val: value to convert.
// res: buffer to hold result.
// Out: returns 0 if not enough space, else 1.
// Cav: n/a
int convertToRoman (unsigned int val, char *res, size_t sz) {
char *huns[] = {"", "C", "CC", "CCC", "CD", "D", "DC", "DCC", "DCCC", "CM"};
char *tens[] = {"", "X", "XX", "XXX", "XL", "L", "LX", "LXX", "LXXX", "XC"};
char *ones[] = {"", "I", "II", "III", "IV", "V", "VI", "VII", "VIII", "IX"};
int size[] = { 0, 1, 2, 3, 2, 1, 2, 3, 4, 2};
// Add 'M' until we drop below 1000.
while (val >= 1000) {
if (sz-- < 1) return 0;
*res++ = 'M';
val -= 1000;
}
// Add each of the correct elements, adjusting as we go.
if (sz < size[val/100]) return 0;
sz -= size[val/100];
strcpy (res, huns[val/100]);
res += size[val/100];
val = val % 100;
if (sz < size[val/10]) return 0;
sz -= size[val/10];
strcpy (res, tens[val/10]);
res += size[val/10];
val = val % 10;
if (sz < size[val) return 0;
sz -= size[val];
strcpy (res, ones[val]);
res += size[val];
// Finish string off.
if (sz < 1) return 0;
*res = '\0';
return 1;
}
хотя в этот момент вы могли бы подумать о том, чтобы перестроить обработку сотен, десятков и единиц в отдельную функцию, поскольку они очень похожи. Я оставлю это как дополнительное упражнение.