Почему Qt использует целочисленный тип со знаком для своих контейнерных классов? - PullRequest
10 голосов
/ 17 февраля 2011

Вопрос понятен.

Интересно, почему они даже подумали, что это будет удобно, поскольку явно отрицательные значения не могут использоваться в контейнерах, которые будут использоваться с ними (см., Например, QList документы ).

Я думал, что они хотели разрешить это для какой-то сумасшедшей формы индексации, но это кажется неподдерживаемым?

Он также генерирует тонну (правильных) предупреждений компилятора о приведении и сравнении подписанных / неподписанных типов (в MSVC).

По какой-то причине он просто кажется несовместимым с STL ...

Ответы [ 2 ]

10 голосов
/ 17 февраля 2011

Хотя я глубоко сочувствую линии рассуждений Криса, я не соглашусь здесь (по крайней мере частично, я играю адвоката дьявола). Нет ничего плохого в использовании неподписанных типов для размеров, и это может быть даже полезно в некоторых обстоятельствах.

Крис оправдывает подписанные типы размеров тем, что они естественным образом используются в качестве индексов массивов, и вы можете захотеть сделать арифметику с индексами массивов, и эта арифметика может создавать временные значения, которые являются отрицательными.

Это нормально, и арифметика без знака не создает проблем при условии, что вы будете правильно интерпретировать свои значения при выполнении сравнений. Поскольку поведение переполнения целых чисел без знака полностью определено, временные переполнения в отрицательном диапазоне (или в огромные положительные числа) не вносят никаких ошибок, если они исправлены перед выполнением сравнения.

Иногда поведение переполнения является даже желательным, поскольку поведение переполнения арифметики без знака делает определенные проверки диапазона выразимыми как одно сравнение, которое в противном случае потребовало бы двух сравнений. Если я хочу проверить, находится ли x в диапазоне [a,b] и все ли значения без знака, я могу просто сделать:

if (x - a < b - a) {
}

Это не работает со знаковыми переменными; такие проверки диапазона довольно распространены для размеров и смещений массива.

Я упоминал ранее, что выгода в том, что арифметика переполнения имеет определенные результаты. Если ваша арифметика индекса переполняет подписанный тип, поведение определяется реализацией; нет способа сделать вашу программу переносимой. Используйте неподписанный тип, и эта проблема исчезнет. По общему признанию это относится только к огромным смещениям, но это касается некоторых применений.

В основном, возражения против неподписанных типов часто завышены. Реальная проблема заключается в том, что большинство программистов на самом деле не думают о точной семантике кода, который они пишут, и для небольших целочисленных значений подписанные типы ведут себя в большей степени в соответствии с их интуицией. Однако размеры данных растут довольно быстро. Когда мы имеем дело с буферами или базами данных, мы часто выходим за пределы диапазона «small», и переполнение со знаком гораздо более проблематично для правильной обработки, чем переполнение без знака. Решение заключается не в том, чтобы «не использовать неподписанные типы», а в том, чтобы «тщательно продумать код, который вы пишете, и убедиться, что вы его понимаете».

5 голосов
/ 17 февраля 2011

Потому что, на самом деле, вы обычно хотите выполнять арифметику с индексами, что означает, что вы можете захотеть создать временные значения, которые являются отрицательными.Это явно болезненно, когда базовый тип индексации без знака.

Единственное подходящее время для использования чисел без знака - с арифметикой модуля.Использование «unsgined» в качестве некоторого спецификатора контракта «число в диапазоне [0 ...» просто неуклюже и слишком грубо, чтобы быть полезным.

Подумайте: какой тип следует использовать для представления идеичто число должно быть положительным целым числом от 1 до 10?Почему 0 ... 2 ^ x более специальный диапазон?

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...