2016-10-14 1 views
1

У меня есть некоторые проблемы с python 3.x. В python 2.x. Я мог бы использовать replace attr в filter obj, но теперь я не могу использовать это. Вот часть моего кода:AttributeError: объект 'filter' не имеет атрибута 'replace' в Python 3

def uniq(seq): 
    seen = {} 
    return [seen.setdefault(x, x) for x in seq if x not in seen] 

def partition(seq, n): 
    return [seq[i : i + n] for i in xrange(0, len(seq), n)] 

def PlayFairCipher(key, from_ = 'J', to = None): 
    if to is None: 
     to = 'I' if from_ == 'J' else '' 

    def canonicalize(s): 
     return list(filter(str.isupper, s.upper()).replace(from_, to)) 

    m = partition(uniq(canonicalize(key + ascii_uppercase)), 5) 

    enc = {} 

    for row in m: 
     for i, j in product(xrange(5), repeat=2): 
      if i != j: 
       enc[row[i] + row[j]] = row[(i + 1) % 5] + row[(j + 1) % 5] 

    for c in zip(*m): 
     for i, j in product(xrange(5), repeat=2): 
      if i != j: 
       enc[c[i] + c[j]] = c[(i + 1) % 5] + c[(j + 1) % 5] 

    for i1, j1, i2, j2 in product(xrange(5), repeat=4): 
     if i1 != i2 and j1 != j2: 
      enc[m[i1][j1] + m[i2][j2]] = m[i1][j2] + m[i2][j1] 

    def sub_enc(txt): 
     lst = findall(r"(.)(?:(?!\1)(.))?", canonicalize(txt)) 
     return ''.join(enc[a + (b if b else 'X')] for a, b in lst) 

    return sub_enc 

Но когда компилируется, я получаю это:

AttributeError: 'filter' object has no attribute 'replace' 

Как я могу это исправить?

+0

плюс ваш код полон 'xrange' .. –

+0

Я не запускаю это, я запускаю фильтр (str.isupper, s.upper()). Replace (from_, to) "в python 2, и это работает хорошо. но как насчет в python 3? @vaultah –

+0

Я решил проблему с xrange. это часть моего кода, как я уже сказал. @ Jean-FrançoisFabre –

ответ

2

in python 2, filter возвращает строку, если строка передается как входной.

filter(...) filter(function or None, sequence) -> list, tuple, or string

Return those items of sequence for which function(item) is true. If function is None, return the items that are true. If sequence is a >tuple or string, return the same type, else return a list.

Чтобы смоделировать это поведение в Python 3 вобще

"".join(filter(str.isupper, s.upper())) 

преобразовать итерацию в строку, то вы можете выполнить замену

1

Я думаю, что вы можете использовать список понимание:

[c.replace(from_, to) for c in s.upper() if c.isupper()] 

это то, что вы хотите? Там много кода, поэтому мне может быть что-то не хватает