TypeScript: Тип Any
В этом уроке мы разберем специальный тип any
, который добавлен в TypeScript.
Польза типа any
Тип any
используется там, где проверка типов не нужна, либо когда TypeScript не может вывести тип данных автоматически:
// В таком случае тип будет any[],
// так как TypeScript не может вывести тип содержимого,
// ведь его еще нет
const items = [];
// Можно добавлять все что угодно
items.push(1);
items.push('code-basics');
any
превращает TypeScript в JavaScript, так как данные с таким типом перестают проверяться:
// Ошибка возникнет только во время запуска js-кода
let value: any = 5;
value.toString(); // ok
value(); // ok, но будет ошибка при запуске js-кода
value.trim(); // ok, но будет ошибка при запуске js-кода
value = 'wow'; // ok
Тем не менее, any
полезен во многих случаях. Например, когда нужно перевести проект из JavaScript в TypeScript. В этом случае сначала все типы объявляются как any
, а затем переписываются на нужные.
Также any
используют для работы с библиотеками JavaScript из TypeScript кода, у которых нет описанных типов. В остальных случаях any
нужно избегать, так как теряется весь смысл использования языка TypeScript.
Рассмотрим первый случай подробнее.
Из JavaScript в TypeScript
Возьмем для примера код, который считает количество слов в предложении:
const sentence = 'table cat table dog dog apple table';
const words = sentence.split(' ');
const initial = {};
const result = words.reduce((acc, word) => {
acc[word] = Object.hasOwn(acc, word) ? acc[word] + 1 : 1;
return acc;
}, initial);
// { table: 3, cat: 1, dog: 2, apple: 1 }
Компилятор TypeScript такой код не пропустит. Он укажет, что объект, который находится в константе initial
, не содержит ключей со строковым типом:
No index signature with a parameter of type 'string' was found on type '{}'.
4 acc[word] = Object.hasOwn(acc, word) ? acc[word] + 1 : 1;
Так происходит, потому что структура объекта задает его тип во время определения. Также в процессе работы структура не может меняться. Но в коде выше изначально объект вообще пустой, а по мере работы он заполняется ключами динамически.
Правильно задавать тип в ситуации с динамическими ключами мы научимся позже. А пока сделаем код рабочим с помощью any
. Для этого нужно определить объект с явным указанием типа:
const sentence = 'table cat table dog dog apple table';
const words = sentence.split(' ');
const initial: any = {}; // Указали тип как any
const result = words.reduce((acc, word) => {
acc[word] = Object.hasOwn(acc, word) ? acc[word] + 1 : 1;
return acc;
}, initial);
TypeScript больше не показывает ошибку компиляции, что с одной стороны хорошо. Но с другой — сама проверка допустимости действий над этим объектом отключена. Если в дальнейшем обратиться к несуществующему свойству в этом объекте, TypeScript не укажет на ошибку.
Выводы
В этом уроке мы научились работать с типом any
. Также мы узнали, для чего и в каких случаях он используется.
Задание
Реализуйте функцию getParams()
, которая принимает на вход строку запроса (query string) и возвращает параметры в виде объекта:
getParams('per=10&page=5');
// { per: '10', page: '5' }
getParams('name=hexlet&count=3&order=asc');
// { name: 'hexlet', count: '3', order: 'asc' }
Эту задачу лучше всего решать через метод reduce()
.
Упражнение не проходит проверку — что делать? 😶
Если вы зашли в тупик, то самое время задать вопрос в «Обсуждениях». Как правильно задать вопрос:
- Обязательно приложите вывод тестов, без него практически невозможно понять что не так, даже если вы покажете свой код. Программисты плохо исполняют код в голове, но по полученной ошибке почти всегда понятно, куда смотреть.
В моей среде код работает, а здесь нет 🤨
Тесты устроены таким образом, что они проверяют решение разными способами и на разных данных. Часто решение работает с одними входными данными, но не работает с другими. Чтобы разобраться с этим моментом, изучите вкладку «Тесты» и внимательно посмотрите на вывод ошибок, в котором есть подсказки.
Мой код отличается от решения учителя 🤔
Это нормально 🙆, в программировании одну задачу можно выполнить множеством способов. Если ваш код прошел проверку, то он соответствует условиям задачи.
В редких случаях бывает, что решение подогнано под тесты, но это видно сразу.
Прочитал урок — ничего не понятно 🙄
Создавать обучающие материалы, понятные для всех без исключения, довольно сложно. Мы очень стараемся, но всегда есть что улучшать. Если вы встретили материал, который вам непонятен, опишите проблему в «Обсуждениях». Идеально, если вы сформулируете непонятные моменты в виде вопросов. Обычно нам нужно несколько дней для внесения правок.
Кстати, вы тоже можете участвовать в улучшении курсов: внизу есть ссылка на исходный код уроков, который можно править прямо из браузера.