初探Core ML

文 || 張賀

Core ML是iOS11新推出的機器學(xué)習(xí)使用框架浦旱。在此框架之上還有兩個集成度較高的框架:Vision 和 NLP(分別是圖像處理和文字處理領(lǐng)域的機器學(xué)習(xí)框架)。

圖片來自蘋果官方文檔

CoreML也可以看做一個模型的轉(zhuǎn)換器,可以將一個MLModel格式的模型文件自動生成一些類和方法,可以直接使用這些類去做分析,讓你更簡單是在app使用訓(xùn)練好的模型。蘋果提供了一些已經(jīng)訓(xùn)練好的模型供使用,選擇一個合適自己需求的下載 房官。 戳我去下載
下載模型文件
下載之后將文件拖入工程续滋,拖入工程后會根據(jù)工程類型(swift工程還是OC工程)生成對應(yīng)的一些類和方法翰守。
點擊箭頭會進入MobileNet的頭文件

第一行Mchaine Learning Model 是對模型的一些介紹。
第二行Model Class是自動生成的類和方法疲酌,點擊箭頭可以查看頭文件蜡峰。我使用的模型是MobileNet生成的類有3個MobileNetInput MobileNetOutput MobileNet

//
// MobileNet.h
//
// This file was automatically generated and should not be edited.
//

#import <Foundation/Foundation.h>
#import <CoreML/CoreML.h>
#include <stdint.h>

NS_ASSUME_NONNULL_BEGIN


/// Model Prediction Input Type
/// 輸入模型
API_AVAILABLE(macos(10.13), ios(11.0), watchos(4.0), tvos(11.0))
@interface MobileNetInput : NSObject<MLFeatureProvider>

/// Input image to be classified as color (kCVPixelFormatType_32BGRA) image buffer, 224 pixels wide by 224 pixels high
/// 輸入的是一個CVPixelBufferRef類型的圖片 尺寸是224 * 224 尺寸不對會報錯
@property (readwrite, nonatomic) CVPixelBufferRef image;
- (instancetype)init NS_UNAVAILABLE;
- (instancetype)initWithImage:(CVPixelBufferRef)image;
@end


/// Model Prediction Output Type
/// 輸出模型
API_AVAILABLE(macos(10.13), ios(11.0), watchos(4.0), tvos(11.0))
@interface MobileNetOutput : NSObject<MLFeatureProvider>

/// Probability of each category as dictionary of strings to doubles
/// 所有可能的結(jié)果以及每種結(jié)果的可能百分比
@property (readwrite, nonatomic) NSDictionary<NSString *, NSNumber *> * classLabelProbs;

/// Most likely image category as string value
/// 最有可能的結(jié)果 也就是上面百分比最大的那個
@property (readwrite, nonatomic) NSString * classLabel;
- (instancetype)init NS_UNAVAILABLE;
- (instancetype)initWithClassLabelProbs:(NSDictionary<NSString *, NSNumber *> *)classLabelProbs classLabel:(NSString *)classLabel;
@end


/// Class for model loading and prediction
/// 通過這個類使輸入模型 ---> 輸出模型
API_AVAILABLE(macos(10.13), ios(11.0), watchos(4.0), tvos(11.0))
@interface MobileNet : NSObject
@property (readonly, nonatomic, nullable) MLModel * model;
- (nullable instancetype)initWithContentsOfURL:(NSURL *)url error:(NSError * _Nullable * _Nullable)error;

/**
    Make a prediction using the standard interface
    @param input an instance of MobileNetInput to predict from
    @param error If an error occurs, upon return contains an NSError object that describes the problem. If you are not interested in possible errors, pass in NULL.
    @return the prediction as MobileNetOutput
*/
- (nullable MobileNetOutput *)predictionFromFeatures:(MobileNetInput *)input error:(NSError * _Nullable * _Nullable)error;

/**
    Make a prediction using the convenience interface
    @param image Input image to be classified as color (kCVPixelFormatType_32BGRA) image buffer, 224 pixels wide by 224 pixels high:
    @param error If an error occurs, upon return contains an NSError object that describes the problem. If you are not interested in possible errors, pass in NULL.
    @return the prediction as MobileNetOutput
*/
- (nullable MobileNetOutput *)predictionFromImage:(CVPixelBufferRef)image error:(NSError * _Nullable * _Nullable)error;
@end

NS_ASSUME_NONNULL_END

MobileNet.h的所有代碼都在上面,可以看見使用起來非常簡單朗恳。

你可能需要用到的方法:
UIImage轉(zhuǎn)換成CVPixelBufferRef

/// 下面這個屬性是UIImage的湿颅,你只要UIImage的對象 `image.CGImage`傳進來就行了
/// @property(nullable, nonatomic,readonly) CGImageRef CGImage; 

- (CVPixelBufferRef)GetpixelBufferWithCGImage:(CGImageRef)cgimage
{
    NSDictionary *options = @{
                              (NSString*)kCVPixelBufferCGImageCompatibilityKey : @YES,
                              (NSString*)kCVPixelBufferCGBitmapContextCompatibilityKey : @YES,
                              (NSString*)kCVPixelBufferIOSurfacePropertiesKey: [NSDictionary dictionary]
                              };
    CVPixelBufferRef pxbuffer = NULL;
    
    CGFloat frameWidth = CGImageGetWidth(cgimage);
    CGFloat frameHeight = CGImageGetHeight(cgimage);
    
    CVReturn status = CVPixelBufferCreate(kCFAllocatorDefault,
                                          frameWidth,
                                          frameHeight,
                                          kCVPixelFormatType_32BGRA,
                                          (__bridge CFDictionaryRef) options,
                                          &pxbuffer);
    
    NSParameterAssert(status == kCVReturnSuccess && pxbuffer != NULL);
    
    CVPixelBufferLockBaseAddress(pxbuffer, 0);
    void *pxdata = CVPixelBufferGetBaseAddress(pxbuffer);
    NSParameterAssert(pxdata != NULL);
    
    CGColorSpaceRef rgbColorSpace = CGColorSpaceCreateDeviceRGB();
    
    CGContextRef context = CGBitmapContextCreate(pxdata,
                                                 frameWidth,
                                                 frameHeight,
                                                 8,
                                                 CVPixelBufferGetBytesPerRow(pxbuffer),
                                                 rgbColorSpace,
                                                 (CGBitmapInfo)kCGImageAlphaNoneSkipFirst);
    NSParameterAssert(context);
    CGContextConcatCTM(context, CGAffineTransformIdentity);
    CGContextDrawImage(context, CGRectMake(0,
                                           0,
                                           frameWidth,
                                           frameHeight),
                       cgimage);
    CGColorSpaceRelease(rgbColorSpace);
    CGContextRelease(context);
    
    CVPixelBufferUnlockBaseAddress(pxbuffer, 0);
    
    return pxbuffer;
}

你可能用到的方法,裁剪圖片粥诫。將圖片裁剪成224 * 224 的油航。

#pragma mark -裁剪圖片
//需要傳過來的參數(shù)有 : 圖片 image 和 自定的尺寸
- (UIImage *)image:(UIImage*)image byScalingToSize:(CGSize)targetSize {
    //原始 iamge
    UIImage *sourceImage = image;
    //新的image 用來接收裁剪后的image 開始時為 nil 
    UIImage *newImage = nil;
    UIGraphicsBeginImageContext(targetSize);
    CGRect thumbnailRect = CGRectZero;
    //裁剪后的image 的原點和 裁剪前 的 image 的 原點相同
    thumbnailRect.origin = CGPointZero;
    //裁剪后的image 的寬和 指定的寬 相同
    thumbnailRect.size.width  = targetSize.width;
    //裁剪后的image 的長和 指定的長 相同
    thumbnailRect.size.height = targetSize.height;
    //將原始image 在設(shè)定的 位置上繪制(裁剪)
    [sourceImage drawInRect:thumbnailRect];
    //把裁剪好的 image 放在 新的image 上
    newImage = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    return newImage ;
}

調(diào)用系統(tǒng)相機或從相冊中選擇照片,選擇完成之后

#pragma mark - 圖片識別
/// 我里面的方法就是上面的2個方法做了一點封裝
- (void)RecognizeWithImage:(UIImage *)pImage
{
    MobileNet *mobileNet = [[MobileNet alloc]init];
    //只需要調(diào)用predictionFromImage:error:這一個方法就可以得到結(jié)果了
    MobileNetOutput *output = [mobileNet predictionFromImage:[[pImage ScalingToSize:CGSizeMake(224, 224)] GetpixelBuffer] error:nil];
    _photoName.text = output.classLabel;
    //所有可能的結(jié)果 以及每種結(jié)果的可能性百分比
    NSLog(@"classLabelProbs----------------%@",output.classLabelProbs);
    //可能性最大的那個結(jié)果
    NSLog(@"classLabel----------------%@",output.classLabel);
}

最后附一張識別截圖


最后附一張識別截圖

此文章待更....一些細節(jié)和理論部分有待深入學(xué)習(xí)和理解怀浆。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末谊囚,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子执赡,更是在濱河造成了極大的恐慌镰踏,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,194評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件沙合,死亡現(xiàn)場離奇詭異奠伪,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評論 2 385
  • 文/潘曉璐 我一進店門绊率,熙熙樓的掌柜王于貴愁眉苦臉地迎上來含末,“玉大人,你說我怎么就攤上這事即舌。” “怎么了挎袜?”我有些...
    開封第一講書人閱讀 156,780評論 0 346
  • 文/不壞的土叔 我叫張陵顽聂,是天一觀的道長。 經(jīng)常有香客問我盯仪,道長紊搪,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,388評論 1 283
  • 正文 為了忘掉前任全景,我火速辦了婚禮耀石,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘爸黄。我一直安慰自己滞伟,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,430評論 5 384
  • 文/花漫 我一把揭開白布炕贵。 她就那樣靜靜地躺著梆奈,像睡著了一般。 火紅的嫁衣襯著肌膚如雪称开。 梳的紋絲不亂的頭發(fā)上亩钟,一...
    開封第一講書人閱讀 49,764評論 1 290
  • 那天,我揣著相機與錄音鳖轰,去河邊找鬼清酥。 笑死,一個胖子當(dāng)著我的面吹牛蕴侣,可吹牛的內(nèi)容都是我干的焰轻。 我是一名探鬼主播,決...
    沈念sama閱讀 38,907評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼昆雀,長吁一口氣:“原來是場噩夢啊……” “哼鹦马!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起忆肾,我...
    開封第一講書人閱讀 37,679評論 0 266
  • 序言:老撾萬榮一對情侶失蹤荸频,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后客冈,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體旭从,經(jīng)...
    沈念sama閱讀 44,122評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,459評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了和悦。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片退疫。...
    茶點故事閱讀 38,605評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖鸽素,靈堂內(nèi)的尸體忽然破棺而出褒繁,到底是詐尸還是另有隱情,我是刑警寧澤馍忽,帶...
    沈念sama閱讀 34,270評論 4 329
  • 正文 年R本政府宣布棒坏,位于F島的核電站,受9級特大地震影響遭笋,放射性物質(zhì)發(fā)生泄漏坝冕。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,867評論 3 312
  • 文/蒙蒙 一瓦呼、第九天 我趴在偏房一處隱蔽的房頂上張望喂窟。 院中可真熱鬧,春花似錦央串、人聲如沸磨澡。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽钱贯。三九已至,卻和暖如春侦另,著一層夾襖步出監(jiān)牢的瞬間秩命,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評論 1 265
  • 我被黑心中介騙來泰國打工褒傅, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留弃锐,地道東北人。 一個月前我還...
    沈念sama閱讀 46,297評論 2 360
  • 正文 我出身青樓殿托,卻偏偏與公主長得像霹菊,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子支竹,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,472評論 2 348

推薦閱讀更多精彩內(nèi)容