Решатель кроссвордов с использованием библиотеки ограничений - PullRequest
0 голосов
/ 29 ноября 2018

enter image description here

Мне нужно решить эту проблему, как показано на рисунке.я попробовал код, приведенный ниже

    # -*- coding: utf-8 -*-
"""
Spyder Editor
This is a temporary script file.
"""
from constraint import *

problem = Problem()

problem.addVariable('oneACROSS',['HOSES','LASER','SAILS','SHEET','STEER'])

problem.addVariable('fourACROSS',['HEEL','HIKE','KEEL','KNOT','LINE'])

problem.addVariable('sevenACROSS',['AFT','ALE','EEL','LEE','TIE'])

problem.addVariable('eightACROSS',['HOSES','LASER','SAILS','SHEET','STEER'])

problem.addVariable('twoDOWN',  ['HOSES','LASER','SAILS','SHEET','STEER'])

problem.addVariable('threeDOWN',  ['HOSES','LASER','SAILS','SHEET','STEER'])

problem.addVariable('fiveDOWN',['HEEL','HIKE','KEEL','KNOT','LINE'])

problem.addVariable('sixDOWN',['AFT','ALE','EEL','LEE','TIE'])

problem.addConstraint(lambda oneACROSS,twoDOWN : oneACROSS[3]==twoDOWN[1])

problem.addConstraint(lambda oneACROSS,threeDOWN : oneACROSS[5]==threeDOWN[1]) 

problem.addConstraint(lambda fourACROSS,twoDOWN : fourACROSS[2]==twoDOWN[3])

problem.addConstraint(lambda fourACROSS,fiveDOWN : fourACROSS[3]==fiveDOWN[1])

problem.addConstraint(lambda threeDOWN,fourACROSS : fourACROSS[4]==threeDOWN[3])

problem.addConstraint(lambda twoDOWN,sevenACROSS : sevenACROSS[1]==twoDOWN[4])

problem.addConstraint(lambda sevenACROSS,fiveDOWN : sevenACROSS[2]==fiveDOWN[2]) 

problem.addConstraint(lambda threeDOWN,sevenACROSS : sevenACROSS[3]==threeDOWN[4])

problem.addConstraint(lambda sixDOWN,eightACROSS : eightACROSS[1]==sixDOWN[2])

problem.addConstraint(lambda twoDOWN,eightACROSS : eightACROSS[3]==twoDOWN[5])

problem.addConstraint(lambda fiveDOWN,eightACROSS : eightACROSS[4]==fiveDOWN[3])

problem.addConstraint(lambda threeDOWN,eightACROSS : eightACROSS[5]==threeDOWN[5])

solution=problem.getSolutions()

print(solution)

Но он дает мне эту ошибку

runfile ('C: /Users/aliya/.spyder-py3/temp.py', wdir ='C: /Users/aliya/.spyder-py3') Трассировка (последний последний вызов):

Файл "", строка 1, в runfile ('C: /Users/aliya/.spyder-py3/temp.py ', wdir =' C: /Users/aliya/.spyder-py3 ')

Файл "C: \ ProgramData \ Anaconda3 \ lib \ site-packages \ spyder_kernels \ customize \ spydercustomize.py", строка 668, в исполняемом файле исполняемого файла (имя файла, пространство имен)

Файл" C: \ ProgramData \ Anaconda3 \ lib \ site-packages \ spyder_kernels \ customize \ spydercustomize.py ", строка 108, в исполняемом файле exec (compile (f.read (), filename, 'exec'), пространство имен)

Файл "C: /Users/aliya/.spyder-py3/temp.py", строка 50, в решении = проблема.getSolutions ()

Файл "C: \ ProgramData \ Anaconda3 \ lib \ site-packages \ constraint__init __. py", строка 271, в getSolutions возвращает self._solver.getSolutions (домены, ограничения, vconstraints)

Файл "C: \ ProgramData \ Anaconda3 \ lib \ site-packages"\ constraint__init __. py ", строка 567, в возвращаемом списке getSolutions (self.getSolutionIter (домены, ограничения, vconstraints))

Файл" C: \ ProgramData \ Anaconda3 \ lib \ site-packages \ constraint__init __. py "строка 544, в getSolutionIter, если не ограничение (переменные, домены, назначения, pushdomains):

Файл "C: \ ProgramData \ Anaconda3 \ lib \ site-packages \ constraint__init __. py", строка 991, в call self.forwardCheck (переменные, домены, назначения)

Файл "C: \ ProgramData \ Anaconda3 \ lib \ site-packages \ constraint__init __. Py", строка 935, в forwardCheck, если не self(переменные, домены, назначения):

Файл "C: \ ProgramData \ Anaconda3 \ lib \ site-packages \ constraint__init __. py", строка 993, в вызов , возврат self._func (* parms)

Ошибка типа: lambda () принимает 2 позиционных аргумента, но было дано 8

Любая помощь по этому поводу!

Ответы [ 3 ]

0 голосов
/ 29 ноября 2018

Синтаксический анализатор Python запрещает именование переменных таким образом, для разбора чисел и переменных по отдельности, так как присвоение имени переменной 1e1 создаст хаос - это число 10.0 или переменная 1e1?

"Питон, пожалуйста, выведите для меня 1e1!"- «Почему это 10.0? Я хранил там 100!»

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

Поэтому измените

problem.addConstraint(lambda 1ACROSS,2DOWN: 1ACROSS in range((1,1),(1,5)) and 2DOWN in range ((1,3),(5,3)) and 1ACROSS[2]==2DOWN[0])
problem.addConstraint(lambda 3DOWN,1ACROSS: 3DOWN in range((1,5),(5,5)) and 1ACROSS[4]==3DOWN[0])
problem.addConstraint(lambda 4ACROSS,5DOWN: 4ACROSS in range((3,2),(3,5)) and 5DOWN in range ((3,4),(6,4)) and 4ACROSS[1]==2DOWN[2] and 4ACROSS[2]==5DOWN[0] and 4ACROSS[3]==3DOWN[2])
problem.addConstraint(lambda 6DOWN,8ACROSS: 6DOWN in range((4,1),(6,1)) and 8ACROSS in range((5,1),(5,5)) and 6DOWN[1]==8ACROSS[0])

На:

problem.addConstraint(lambda ACROSS,DOWN: ACROSS in range((1,1),(1,5)) and DOWN in range ((1,3),(5,3)) and ACROSS[2]==DOWN[0])
problem.addConstraint(lambda DOWN,ACROSS: DOWN in range((1,5),(5,5)) and ACROSS[4]==DOWN[0])
problem.addConstraint(lambda ACROSS,DOWN: ACROSS in range((3,2),(3,5)) and DOWN in range ((3,4),(6,4)) and ACROSS[1]==DOWN[2] and ACROSS[2]==DOWN[0] and ACROSS[3]==DOWN[2])
problem.addConstraint(lambda DOWN,ACROSS: DOWN in range((4,1),(6,1)) and ACROSS in range((5,1),(5,5)) and DOWN[1]==ACROSS[0])
0 голосов
/ 05 декабря 2018

Привет, дорогой кроссворд, это очень сложная проблема. У меня есть решение этого задания, но оно реализовано в библиотеке ortools от Google.Ниже приведены требования для использования этой библиотеки: Требования:

**Library:** ortools by google

**Requirements for this library:**


**1-you have to install/update your pip forcefully by 2 commands:**
    1- "conda config --add channels conda-forge"
    2- "conda insatll pip=18.0"


**2-you have to update or install protobuf library version=3.6.1 using following command:**
    1- "pip insatll protobuf=3.6.1"


**3-if protobuf cannot istall following libraries with it you have to also install them:**
   1- six 1.11.1 (Query: pip install six=1.11.0).
   2- setuptools 40.6.2 (Query: pip install setuptools=40.6.2)
   3- ortools 6.10.6025 (Query: pip install ortools=6.10.6025)
**"pywrapcp"** this function will help for the creation of the solver.
**.IntVar() **function is used to add the varriable and domains
**.Add()** function is used to add constraints.

Ниже приведен код этой проблемы:

    ## Author: Ahsan Azeem
## Solution of Crossword using ortools library
from ortools.constraint_solver import pywrapcp
def main():
  solver = pywrapcp.Solver("Problem")
  alpha = "_abcdefghijklmnopqrstuvwxyz"
  a = 1
  b = 2
  c = 3
  d = 4
  e = 5
  f = 6
  g = 7
  h = 8
  i = 9
  j = 10
  k = 11
  l = 12
  m = 13
  n = 14
  o = 15
  p = 16
  q = 17
  r = 18
  s = 19
  t = 20
  u = 21
  v = 22
  w = 23
  x = 24
  y = 25
  z = 26
  num_words = 15
  word_len = 5
  AA = [
      [h, o, s, e, s],  # HOSES
      [l, a, s, e, r],  # LASER
      [s, a, i, l, s],  # SAILS
      [s, h, e, e, t],  # SHEET
      [s, t, e, e, r],  # STEER
      [h, e, e, l, 0],  # HEEL
      [h, i, k, e, 0],  # HIKE
      [k, e, e, l, 0],  # KEEL
      [k, n, o, t, 0],  # KNOT
      [l, i, n, e, 0],  # LINE
      [a, f, t, 0, 0],  # AFT
      [a, l, e, 0, 0],  # ALE
      [e, e, l, 0, 0],  # EEL
      [l, e, e, 0, 0],  # LEE
      [t, i, e, 0, 0]  # TIE
  ]
  num_overlapping = 12
  overlapping = [
      [0, 2, 1, 0],  # s
      [0, 4, 2, 0],  # s
      [3, 1, 1, 2],  # i
      [3, 2, 4, 0],  # k
      [3, 3, 2, 2],  # e
      [6, 0, 1, 3],  # l
      [6, 1, 4, 1],  # e
      [6, 2, 2, 3],  # e
      [7, 0, 5, 1],  # l
      [7, 2, 1, 4],  # s
      [7, 3, 4, 2],  # e
      [7, 4, 2, 4]  # r
  ]
  n = 8
  A = {}
  for I in range(num_words):
    for J in range(word_len):
      A[(I, J)] = solver.IntVar(0, 26, "A(%i,%i)" % (I, J))
  A_flat = [A[(I, J)] for I in range(num_words) for J in range(word_len)]
  E = [solver.IntVar(0, num_words, "E%i" % I) for I in range(n)]
  solver.Add(solver.AllDifferent(E))
  for I in range(num_words):
    for J in range(word_len):
      solver.Add(A[(I, J)] == AA[I][J])
  for I in range(num_overlapping):
    # This is what I would do:
    # solver.Add(A[(E[overlapping[I][0]], overlapping[I][1])] ==  A[(E[overlapping[I][2]], overlapping[I][3])])
    # But we must use Element explicitly
    solver.Add(
        solver.Element(
            A_flat, E[overlapping[I][0]] * word_len + overlapping[I][1]) ==
        solver.Element(
            A_flat, E[overlapping[I][2]] * word_len + overlapping[I][3]))
  solution = solver.Assignment()
  solution.Add(E)
  db = solver.Phase(E + A_flat,
                    solver.INT_VAR_SIMPLE,
                    solver.ASSIGN_MIN_VALUE)
  solver.NewSearch(db)
  num_solutions = 0
  while solver.NextSolution():
    print(E)
    print_solution(A, E, alpha, n, word_len)
    num_solutions += 1
  solver.EndSearch()
  print()
  print("num_solutions:", num_solutions)
  print("failures:", solver.Failures())
def print_solution(A, E, alpha, n, word_len):
  for ee in range(n):
    print("%i: (%2i)" % (ee, E[ee].Value()), end=' ')
    print("".join(["%s" % (alpha[A[ee, ii].Value()]) for ii in range(word_len)]))
if __name__ == "__main__":
  main()

надеюсь, что вы получили ответ

0 голосов
/ 29 ноября 2018

Во всех строках, начинающихся с problem.addConstraint, вы используете имена переменных, начинающиеся с цифры, например, 1ACROSS и 2DOWN.Это не разрешено в Python.Попробуйте назвать их ONEACROSS и TWODOWN или VAR1ACROSS и VAR2DOWN, например, и это должно работать лучше.

...