Перекрывающиеся экземпляры Haskell и функции типов

У меня есть следующий класс типов, который моделирует SQL-подобную оптимизацию запросов:

class OptimizableQuery q where
  type Optimized q :: *
  optimize :: q -> Optimized q

instance Query q => OptimizableQuery q where
  type Optimized q = q
  optimize q = q

instance (Query q, OptimizableQuery q) => OptimizableQuery (Select (Select q p) p) where
  type Optimized (Select (Select q p) p) = Select (Optimized q) p
  optimize (Select (Select q _) p) = Select (optimize q) p

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

1 ответ

Решение

Недопустимо иметь перекрывающиеся экземпляры с семействами типов. Подробности см. В руководстве GHC"Перекрытие экземпляров синонимов типов".

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

Другие вопросы по тегам