comando shell para obter o tamanho de pixel de uma imagem

43

Existe um comando shell que retorna o tamanho do pixel de uma imagem?

Estou tentando produzir um gif animado a partir de diferentes gifs com tamanhos diferentes usando convert(por exemplo convert -delay 50 1.gif 2.gif -loop 0 animated.gif).

O problema é que o convert simplesmente sobrepõe as imagens usando o tamanho da primeira imagem como o tamanho do gif animado e, como eles têm tamanhos diferentes, o resultado é uma bagunça, com pedaços dos quadros antigos aparecendo nos novos.

azul
fonte
O que é "tamanho de pixel"? Bits por pixel (profundidade) ou contagem de pixels?
Ciro Santilli escreveu

Respostas:

50

encontrei uma solução: identifyparte do pacote imagemagick, faz exatamente o que eu preciso

$ identify color.jpg 
> color.jpg JPEG 1980x650 1980x650+0+0 8-bit DirectClass 231KB 0.000u 0:00.000
azul
fonte
Você poderia explicar por que recebi várias linhas ao executar identify imagename.ico? Like todour.ico[0] ICO 32x32 32x32+0+0 4-bit sRGB 0.000u 0:00.000 todour.ico[1] ICO 16x16 16x16+0+0 4-bit sRGB 0.000u 0:00.000 todour.ico[2] ICO 48x48 48x48+0+0 8-bit sRGB 0.000u 0:00.000
roachsinai 26/02
44

Em vez de analisar a saída identifypelo olho, ou pelos utilitários de texto, você pode usar sua -formatopção para gerar a largura e a altura no formato que melhor lhe convier. Por exemplo:

$ identify -format '%w %h' img.png
100 200
$ identify -format '%wx%h' img.png
100x200

Uma lista das propriedades da imagem que você pode gerar pode ser encontrada nesta página , mas para a pergunta aqui, parece que tudo o que você precisa é %we %h, que fornece a largura e a altura da imagem, respectivamente, em pixels.


A flexibilidade oferecida por me -formatfoi útil para encontrar as maiores imagens em termos de pixels, produzindo %[fx:w*h]várias imagens e classificando a saída.

Convém especificar a -pingopção se estiver processando muitas imagens, usando escapes mais complicados e desejando garantir que o programa não perca tempo carregando as imagens inteiras. Com escapes simples, -pingdeve ser o padrão. Mais informações sobre a escolha entre -pinge +pingpodem ser encontradas aqui .

Dan Getz
fonte
15

você pode simplesmente usar o comando "arquivo" para obter as informações necessárias:

~# file cha_2.png 
cha_2.png: PNG image data, 656 x 464, 8-bit/color RGB, non-interlaced
Pascal Schmiel
fonte
11
Esta não informa o tamanho de outros (não-png) tipos de imagem ... file taylor-swift-money-makers-990.jpg->taylor-swift-money-makers-990.jpg: JPEG image data, JFIF standard 1.01, comment: "CREATOR: gd-jpeg v1.0 (using IJ"
Alex Gray
Não é verdade, pelo menos nas versões recentes do macOS.
Rien333
5

Use identifypara ver os tamanhos:

$ identify color.jpg 
> color.jpg JPEG 1980x650 1980x650+0+0 8-bit DirectClass 231KB 0.000u 0:00.000

Extrair valor via cut | sed, do campo 3:

identify ./color.jpg | cut -f 3 -d " " | sed s/x.*// #width
identify ./color.jpg | cut -f 3 -d " " | sed s/.*x// #height

Atribuir à variável:

W=`identify ./color.jpg | cut -f 3 -d " " | sed s/x.*//` #width
H=`identify ./color.jpg | cut -f 3 -d " " | sed s/.*x//` #height
echo $W
> 1980
echo $H
> 650
Hugolpz
fonte
2

Ambos displaye filesão muito lento, e tem o potencial de trazer até mesmo sistemas bem capaz de joelhos lidar com muitas vários arquivos. Um pequeno teste:

     $ du -h *.png --total | tail -n 1
     9.2M    total

     $ ls -l *.png | wc -l
     107

     $ /usr/bin/time file *.png
-->  0.37user 0.26system 0:06.93elapsed 9%CPU (0avgtext+0avgdata 37232maxresident)k
     22624inputs+0outputs (9major+2883minor)pagefaults 0swaps

     $ /usr/bin/time identify *.png
-->  0.56user 0.22system 0:06.77elapsed 11%CPU (0avgtext+0avgdata 25648maxresident)k
     34256inputs+0outputs (119major+2115minor)pagefaults 0swaps

Ao ler apenas os bytes necessários, esta operação pode ser significativamente acelerada.

     $ /usr/bin/time ./pngsize *.png
-->  0.00user 0.00system 0:00.03elapsed 12%CPU (0avgtext+0avgdata 1904maxresident)k
     0inputs+0outputs (0major+160minor)pagefaults 0swaps

Aqui está o pngsize:

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <err.h>
#define oops(syscall) { printf("error processing %s: ", argv[i]); \
        fflush(0); perror(syscall"()"); continue; }
int main(int argc, char **argv) {
    int fd, i;
    uint32_t h, w;
    if (argc < 2) { printf("%s <pngfile> [pngfile ...]\n", argv[0]); exit(0); }
    for (i = 1; i < argc; i++) {
        if (argc > 2) printf("%s: ", argv[i]);
        if ((fd = open(argv[i], O_RDONLY)) == -1) oops("open");
        if (lseek(fd, 16, SEEK_SET) == -1) oops("lseek");
        if (read(fd, &w, 4) < 1) oops("read");
        if (read(fd, &h, 4) < 1) oops("read");
        printf("%dx%d\n", htonl(w), htonl(h));
        if (close(fd) == -1) oops("close");
    }
    return 0;
}

Esse método é muito mais rápido do que usar uma biblioteca que carrega o PNG para frente, para trás e para os lados apenas para obter o tamanho da imagem: P (considere o código com cuidado antes de alimentar um diretório cheio de PNGs arbitrários, é claro).

Os usos de código inet.h para htonl () para de- endian ize a ordenação byte cabeçalho.

i336_
fonte
0

Você também pode experimentar o GraphicsMagick , que é um fork bem conservado do ImageMagick usado, por exemplo, no Flickr e no Etsy:

$ gm identify a.jpg
a.jpg JPEG 480x309+0+0 DirectClass 8-bit 25.2K 0.000u 0:01

É mais rápido do que a identificação do ImageMagick (nos meus testes cerca de duas vezes).

Jerry Epas
fonte
0

Este foi um trecho útil (não escrevi) que retorna as dimensões para cada png e jpg na pasta:

file ./* | perl -ne '@m = /^.*.jpg|^.*.png|[0-9][0-9]*[ ]?x[ ]?[0-9][0-9]*/g; print "@m\n"'
ArleyM
fonte
0

Para aqueles como eu que querem o tamanho em megapixels:

perl -le "printf \"=> fileName = %s size = %.2f Mpix\n\", \"$fileName\", $(identify -format '%w*%h/10**6' $fileName)"
SebMa
fonte