Python 3: преобразование изображения в оттенки серого

Я пытаюсь выполнить упражнение в книге Джона Зелле "Программирование на Python: введение в информатику". Я скачал специальный графический пакет для его книги ( graphics.py, который находится на связанном веб-сайте). Вопрос гласит следующее:

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

Возможно, вы захотите вернуться и просмотреть объект Image из графической библиотеки (раздел 4.8.4). Основная идея для преобразования изображения состоит в том, чтобы пройти его по пикселям и преобразовать каждое из цветов в соответствующий оттенок серого. Серый пиксель создается путем установки его красной, зеленой и синей составляющих, чтобы они имели одинаковую яркость. Так, color_rgb(0, 0, 0) черный, color_rgb(255, 255, 255) белый, и color_rgb(127, 127, 127) серый "на полпути" между ними. Вы должны использовать средневзвешенное значение исходных значений RGB для определения яркости серого. Вот псевдокод для алгоритма в градациях серого [по какой-то причине отступ с четырьмя пробелами не работает при предварительном просмотре]:

for each row in the image:  
    for each column in the image:  
        r, g, b = get pixel information for current row and column  
        brightness = int(round(0.299r + 0.587g + 0.114b))  
    update the image # to see progress row by row

Примечание: операции с пикселями в Image класс довольно медленный, поэтому вы можете использовать относительно небольшие изображения (не 12 мегапикселей) для тестирования вашей программы.


Я работал над этим часами. Это последняя версия моего кода:

# grayscale.py

from graphics import *

def main():  
    infileName = input("File name: ")  
    outfileName = input("Save to: ")

    image = Image(Point(100,100), infileName)
    width = image.getWidth()
    height = image.getHeight()
    win = GraphWin("rgb")
    image.draw(win)

    row = 0
    column = 0

    win.getMouse()

    for row in range(200):
        for column in range(200):
            r, g, b = image.getPixel(row, column)
            brightness = int(round(0.299 * r + 0.587 * g + 0.114 * b))
            image.setPixel(row, column, color_rgb(brightness, brightness, brightness))
            win.update()

    win.getMouse()
    win.close()

main()

Наконец-то я понял, что Python не выдаёт мне сообщений об ошибках. Но все, что делает программа, это загружает изображение, делает пару щелчков мышью, а затем закрывается. Я вводил входной файл как U:\My Pictures\yay.gif, а выходной файл как U:\My Pictures\yay2.gif. Но я только что искал свой компьютер и U: \ My Pictures \ yay2.gif не существует. Что я делаю неправильно? Кстати, это НЕ для класса - у меня нет инструктора, чтобы спросить.

Может быть, я должен следить за постом вместо этого. Я добавил функцию сохранения, но получил изображение с полутоновым окном 200x200 и остальным цветом. Итак, вот несколько строк, которые я изменил:

win = GraphWin("rgb", width, height)
for row in range(height):
    for column in range(width):

И я получаю следующее сообщение об ошибке: Traceback (последний вызов последним):
Файл "C:\Python31\grayscale.py", строка 31, в
главный()
Файл "C:\Python31\grayscale.py", строка 22, в основном
r, g, b = image.getPixel(строка, столбец)
Файл "C:\Python31\lib\graphics.py", строка 810, в getPixel
значение = self.img.get(x,y)
Файл "C: \ Python31 \ lib \ tkinter_init_.py", строка 3301, в get
вернуть self.tk.call (self.name, 'get', x, y)
_tkinter.TclError: pyimage1 get: координаты вне диапазона

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

4 ответа

Вы не сохраняете изображение. Обратите внимание, что переменная outfileName никогда не используется. Вы должны передать его какой-то функции сохранения. бросил быстрый взгляд на graphics.py, думаю, это то, что вам нужно:

РЕДАКТИРОВАТЬ

Чтобы решить проблему с преобразованием только одного угла, сделайте это, вместо этого ваш старый код преобразовывал только первые 200x200 пикселей. Я также изменил диапазон на xrange, не обязательно, но более эффективно.

for row in xrange(height):
        for column in xrange(windth):

Я знаю, что это старый пост, но. Я думаю, что ваша проблема - это фактическое изображение, которое вы используете. Измените разрешение изображения до 200 пикселей на дюйм.

Лучше, чтобы все было просто. Программа просит сохранить файл, который был забыт в конце, и использовать getMouse(), чтобы получить "текущий пиксель", который вы используете getX() и getY(). Вот лучший пример с использованием graphics.py

grayscale.py

from graphics import*

print("This program grayscales images\n")# intro 

def grayscale(): # Obtaining file
    infile = input('In what file is the image?: ')
    outfile = input('What file do you want it saved?: ')

#   Using image file to get dynamics of window. You don't have to create a window either to obtain it. 

    file = Image(Point(0,0), infile)
    width = file.getWidth()
    height = file.getHeight()

#   Getting center point of photo
    cWidth = width / 2
    cHeight = height / 2
    cen = Point(cWidth, cHeight)

#   Ready for window
    image = Image(cen,infile)
    win = GraphWin("Grayscale Image", width, height)
    image.draw(win)


#   Getting current Pixel for loop
    p = win.getMouse()
    x = p.getX()
    y = p.getY()

    for x in range(height):
        for y in range(width):
            r, g, b = image.getPixel(x, y)
            gray = int(round(.299*r + .587*g + .114*b))
            image.setPixel(x,y, color_rgb(gray, gray, gray))
        # i accidentally mixed my x and y variable and it spit
        # out a two-headed cyclops tee hee 
            win.update()

# Click to exit 
    image.save(outfile)
    win.getMouse()
    win.close()

grayscale()

Я не вижу никакого смысла outfileName после того, как он установлен - другими словами, вы, кажется, никогда не сохраняете результат в этот файл (или любой другой файл, в этом отношении).

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