74

サードパーティから提供されたURLから画像を読み込んでいます。URLにはファイル拡張子(またはファイル名)はありません(あいまいなURLであるため)。これから(NSDataの形式で)データを取得し、UIImageにロードして、正常に表示できます。

このデータをファイルに保存したいと思います。しかし、データがどの形式(PNG、JPG、BMP)であるかわかりませんか?JPGだと思いますが(ウェブからの画像なので)、プログラムで確実に見つける方法はありますか?StackOverflowとドキュメントを調べましたが、何も見つかりませんでした。

TIA。


編集:本当にファイル拡張子が必要ですか?私はそれを外部ストレージ(Amazon S3)に永続化していますが、iOSまたはブラウザー(どちらも拡張機能なしでデータを解釈するのに問題がないようです)のコンテキストで常に使用されることを考えると、おそらくこれは問題ではありません。

4

11 に答える 11

145

画像ファイルにNSDataがある場合は、最初のバイトを確認することでコンテンツタイプを推測できます。

+ (NSString *)contentTypeForImageData:(NSData *)data {
    uint8_t c;
    [data getBytes:&c length:1];

    switch (c) {
    case 0xFF:
        return @"image/jpeg";
    case 0x89:
        return @"image/png";
    case 0x47:
        return @"image/gif";
    case 0x49:
    case 0x4D:
        return @"image/tiff";
    }
    return nil;
}
于 2011-02-18T14:20:03.017 に答える
29

wl。の回答を改善して、署名に基づいて画像のMIMEタイプを予測するためのはるかに拡張された正確な方法を次に示します。このコードは、主にphpのext / standard/image.cに触発されました。

- (NSString *)mimeTypeByGuessingFromData:(NSData *)data {

    char bytes[12] = {0};
    [data getBytes:&bytes length:12];

    const char bmp[2] = {'B', 'M'};
    const char gif[3] = {'G', 'I', 'F'};
    const char swf[3] = {'F', 'W', 'S'};
    const char swc[3] = {'C', 'W', 'S'};
    const char jpg[3] = {0xff, 0xd8, 0xff};
    const char psd[4] = {'8', 'B', 'P', 'S'};
    const char iff[4] = {'F', 'O', 'R', 'M'};
    const char webp[4] = {'R', 'I', 'F', 'F'};
    const char ico[4] = {0x00, 0x00, 0x01, 0x00};
    const char tif_ii[4] = {'I','I', 0x2A, 0x00};
    const char tif_mm[4] = {'M','M', 0x00, 0x2A};
    const char png[8] = {0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a};
    const char jp2[12] = {0x00, 0x00, 0x00, 0x0c, 0x6a, 0x50, 0x20, 0x20, 0x0d, 0x0a, 0x87, 0x0a};


    if (!memcmp(bytes, bmp, 2)) {
        return @"image/x-ms-bmp";
    } else if (!memcmp(bytes, gif, 3)) {
        return @"image/gif";
    } else if (!memcmp(bytes, jpg, 3)) {
        return @"image/jpeg";
    } else if (!memcmp(bytes, psd, 4)) {
        return @"image/psd";
    } else if (!memcmp(bytes, iff, 4)) {
        return @"image/iff";
    } else if (!memcmp(bytes, webp, 4)) {
        return @"image/webp";
    } else if (!memcmp(bytes, ico, 4)) {
        return @"image/vnd.microsoft.icon";
    } else if (!memcmp(bytes, tif_ii, 4) || !memcmp(bytes, tif_mm, 4)) {
        return @"image/tiff";
    } else if (!memcmp(bytes, png, 8)) {
        return @"image/png";
    } else if (!memcmp(bytes, jp2, 12)) {
        return @"image/jp2";
    }

    return @"application/octet-stream"; // default type

}

上記の方法では、次の画像タイプが認識されます。

  • image/x-ms-bmp(bmp)
  • image/gif(gif)
  • image/jpeg(jpg、jpeg)
  • image/psd(psd)
  • image/iff(iff)
  • image/webp(webp)
  • image/vnd.microsoft.icon(ico)
  • image/tiff(tif、tiff)
  • image/png(png)
  • image/jp2(jp2)

UIImage残念ながら、カプセル化されたビットマップデータにアクセスできないため、インスタンスからこの種の情報を取得する簡単な方法はありません。

于 2014-12-28T18:59:23.937 に答える
15

@TaiLeのSwift3のソリューションは、データ全体をバイト配列に割り当てることです。画像が大きいと、クラッシュする可能性があります。このソリューションは、1バイトを割り当てるだけです。

import Foundation

public extension Data {
    var fileExtension: String {
        var values = [UInt8](repeating:0, count:1)
        self.copyBytes(to: &values, count: 1)

        let ext: String
        switch (values[0]) {
        case 0xFF:
            ext = ".jpg"
        case 0x89:
            ext = ".png"
        case 0x47:
            ext = ".gif"
        case 0x49, 0x4D :
            ext = ".tiff"
        default:
            ext = ".png"
        }
        return ext
    }
}
于 2017-04-06T09:18:13.720 に答える
9

URLから画像を取得している場合は、おそらくHTTP応答ヘッダーを調べることができます。Content-Typeヘッダーには何か便利なものが含まれていますか?(ブラウザはおそらく画像を正しく表示できるので、コンテンツタイプが適切に設定されている場合にのみ表示できると思います)

于 2010-11-10T17:44:54.943 に答える
8

Swift3バージョン:

let data: Data = UIImagePNGRepresentation(yourImage)!

extension Data {
    var format: String {
        let array = [UInt8](self)
        let ext: String
        switch (array[0]) {
        case 0xFF:
            ext = "jpg"
        case 0x89:
            ext = "png"
        case 0x47:
            ext = "gif"
        case 0x49, 0x4D :
            ext = "tiff"
        default:
            ext = "unknown"
        }
        return ext
    }
}
于 2016-10-29T08:01:32.113 に答える
4

受け入れられた答えの代替は、で画像のUTIをチェックすることimage I/O frameWorkです。UTIから画像タイプを実現できます。これを試して:

CGImageSourceRef imgSrc = CGImageSourceCreateWithData((CFDataRef)data, NULL);
NSString *uti = (NSString*)CGImageSourceGetType(imgSrc);
NSLog(@"%@",uti);

たとえば、GIF画像のUTIは「com.compuserve.gif」、PNG画像のUTIは「public.png」ですがimage I/O frameWork、認識されない画像からUTIを達成することはできません。

于 2017-03-29T07:32:18.990 に答える
3

@ccoroomに基づく私の改善されたソリューション

//  Data+ImageContentType.swift

import Foundation

extension Data {  
    enum ImageContentType: String {
        case jpg, png, gif, tiff, unknown

        var fileExtension: String {
            return self.rawValue
        }
    }

    var imageContentType: ImageContentType {

        var values = [UInt8](repeating: 0, count: 1)

        self.copyBytes(to: &values, count: 1)

        switch (values[0]) {
        case 0xFF:
            return .jpg
        case 0x89:
            return .png
        case 0x47:
           return .gif
        case 0x49, 0x4D :
           return .tiff
        default:
            return .unknown
        }
    }
}

いくつかの使用例:

//load some image
do {
    let imageData = try Data(contentsOf: URL(string: "https://myServer/images/test.jpg")!)
} catch {
    print("Unable to load image: \(error)")
}

//content type check
guard [Data.ImageContentType.jpg,
       Data.ImageContentType.png].contains(imageData.imageContentType) else {
    print("unsupported image type")
            return
        }

//set file extension
let image = "myImage.\(imageData.imageContentType.fileExtension)" //myImage.jpg
于 2019-08-05T15:38:06.487 に答える
3

から画像タイプを取得するにはUIImage、基になるQuartz画像データからタイプ識別子(UTI)を取得できます。

extension UIImage {
    var typeIdentifier: String? {
        cgImage?.utType as String?
    }
}

URLから画像タイプ識別子を取得するには、URLがローカルリソースを指しているかどうかによって異なります。

extension URL {
    // for local resources (fileURLs)
    var typeIdentifier: String? { (try? resourceValues(forKeys: [.typeIdentifierKey]))?.typeIdentifier }
    // for non local resources (web) you can get it asyncronously
    func asyncTypeIdentifier(completion: @escaping ((String?, Error?) -> Void)) {
        var request = URLRequest(url: self)
        request.httpMethod = "HEAD"
        URLSession.shared.dataTask(with: request) { _ , response , error in
            completion((response as? HTTPURLResponse)?.mimeType, error)
        }.resume()
    }
}

let imageURL = URL(string: "https://i.stack.imgur.com/varL9.jpg")!
imageURL.asyncTypeIdentifier { typeIdentifier, error in
    guard let typeIdentifier = typeIdentifier, error == nil else { return }
    print("typeIdentifier:", typeIdentifier)
}
于 2020-05-16T05:11:18.287 に答える
1

既知の画像形式ごとに署名チェックを実装します。これは、PNGデータに対してそれを行う簡単なObjective-C関数です。

// Verify that NSData contains PNG data by checking the signature

- (BOOL) isPNGData:(NSData*)data
{
  // Verify that the PNG file signature matches

  static const
  unsigned char   png_sign[8] = {137, 80, 78, 71, 13, 10, 26, 10};

  unsigned char   sig[8] = {0, 0, 0, 0, 0, 0, 0, 0};

  if ([data length] <= 8) {
    return FALSE;
  }

  [data getBytes:&sig length:8];

  BOOL same = (memcmp(sig, png_sign, 8) == 0);

  return same;
}
于 2011-06-09T20:46:19.070 に答える
1

それが本当にあなたにとって重要であるならば、私はあなたがバイトストリームを調べる必要があると信じています。JPEGはバイトFFD8で始まります。PNGは89504E 47 0D 0A1A0Aで始まります。BMPに同様のヘッダーがあるかどうかはわかりませんが、2010年にWeb上でそれらに遭遇する可能性はあまりないと思います。

しかし、それはあなたにとって本当に重要ですか?それを未知の画像として扱い、Cocoa Touchに任せてはいけませんか?

于 2010-11-10T17:45:49.823 に答える
1

NSDataの画像タイプを確認するためのライブラリを作成しました。

https://github.com/sweetmandm/ImageFormatInspector

于 2015-03-26T15:15:00.993 に答える