2014-08-29 95 views
4

如何使用比特率壓縮視頻?如何使用比特率在ios中壓縮視頻?

我想下面的代碼來壓縮視頻,但它不是工作,因爲它給了我這樣的錯誤

******終止應用程序由於未捕獲的異常「NSInvalidArgumentException」,原因是:「* - [AVAssetReader startReading]不能再被稱爲閱讀後已經開始」 ****

 - (void) imagePickerController: (UIImagePickerController *) picker 
      didFinishPickingMediaWithInfo: (NSDictionary *) info 
    { 


     // Handle movie capture 
     NSURL *movieURL = [info objectForKey: 
          UIImagePickerControllerMediaURL]; 

     NSData *data = [NSData dataWithContentsOfURL:movieURL]; 

     NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,  NSUserDomainMask, YES); 
     NSString *documentsDirectory = [paths objectAtIndex:0]; 
     NSString *tempPath = [documentsDirectory stringByAppendingFormat:@"/vid1.mp4"]; 

     BOOL success = [data writeToFile:tempPath atomically:NO]; 

     if (success) 
     { 
         NSLog(@"VIdeo Successfully written"); 
     } 
     else 
     { 
         NSLog(@"VIdeo Wrting failed"); 
     } 


     NSURL *uploadURL = [NSURL fileURLWithPath:[[NSTemporaryDirectory() stringByAppendingPathComponent:@"1234"] stringByAppendingString:@".mp4"]]; 

     // Compress movie first 
     [self convertVideoToLowQuailtyWithInputURL:movieURL outputURL:uploadURL]; 
    } 




- (void)convertVideoToLowQuailtyWithInputURL:(NSURL*)inputURL 
             outputURL:(NSURL*)outputURL 
    { 
     //setup video writer 
     AVAsset *videoAsset = [[AVURLAsset alloc] initWithURL:inputURL options:nil]; 

     AVAssetTrack *videoTrack = [[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0]; 

     CGSize videoSize = videoTrack.naturalSize; 

     NSDictionary *videoWriterCompressionSettings = [NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithInt:1250000], AVVideoAverageBitRateKey, nil]; 

     NSDictionary *videoWriterSettings = [NSDictionary dictionaryWithObjectsAndKeys:AVVideoCodecH264, AVVideoCodecKey, videoWriterCompressionSettings, AVVideoCompressionPropertiesKey, [NSNumber numberWithFloat:videoSize.width], AVVideoWidthKey, [NSNumber numberWithFloat:videoSize.height], AVVideoHeightKey, nil]; 

     AVAssetWriterInput* videoWriterInput = [AVAssetWriterInput 
               assetWriterInputWithMediaType:AVMediaTypeVideo 
               outputSettings:videoWriterSettings]; 

     videoWriterInput.expectsMediaDataInRealTime = YES; 

     videoWriterInput.transform = videoTrack.preferredTransform; 

     AVAssetWriter *videoWriter = [[AVAssetWriter alloc] initWithURL:outputURL fileType:AVFileTypeQuickTimeMovie error:nil]; 

     [videoWriter addInput:videoWriterInput]; 

     //setup video reader 
     NSDictionary *videoReaderSettings = [NSDictionary dictionaryWithObject:[NSNumber numberWithInt:kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange] forKey:(id)kCVPixelBufferPixelFormatTypeKey]; 

     AVAssetReaderTrackOutput *videoReaderOutput = [[AVAssetReaderTrackOutput alloc] initWithTrack:videoTrack outputSettings:videoReaderSettings]; 

     AVAssetReader *videoReader = [[AVAssetReader alloc] initWithAsset:videoAsset error:nil]; 

     [videoReader addOutput:videoReaderOutput]; 

     //setup audio writer 
     AVAssetWriterInput* audioWriterInput = [AVAssetWriterInput 
               assetWriterInputWithMediaType:AVMediaTypeAudio 
               outputSettings:nil]; 

     audioWriterInput.expectsMediaDataInRealTime = NO; 

     [videoWriter addInput:audioWriterInput]; 

     //setup audio reader 
     AVAssetTrack* audioTrack = [[videoAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0]; 

     AVAssetReaderOutput *audioReaderOutput = [AVAssetReaderTrackOutput assetReaderTrackOutputWithTrack:audioTrack outputSettings:nil]; 

     AVAssetReader *audioReader = [AVAssetReader assetReaderWithAsset:videoAsset error:nil]; 

     [audioReader addOutput:audioReaderOutput]; 

     [videoWriter startWriting]; 

     //start writing from video reader 
     [videoReader startReading]; 

     [videoWriter startSessionAtSourceTime:kCMTimeZero]; 

     dispatch_queue_t processingQueue = dispatch_queue_create("processingQueue1", NULL); 

     [videoWriterInput requestMediaDataWhenReadyOnQueue:processingQueue usingBlock: 
     ^{ 

      while ([videoWriterInput isReadyForMoreMediaData]) 
      { 

       CMSampleBufferRef sampleBuffer; 

       if ([videoReader status] == AVAssetReaderStatusReading && 
        (sampleBuffer = [videoReaderOutput copyNextSampleBuffer])) 
       { 

        [videoWriterInput appendSampleBuffer:sampleBuffer]; 
        CFRelease(sampleBuffer); 
       } 

       else 
       { 
        [videoWriterInput markAsFinished]; 

        if ([videoReader status] == AVAssetReaderStatusCompleted) 
        { 
          [audioReader startReading]; 

          [videoWriter startSessionAtSourceTime:kCMTimeZero]; 

          dispatch_queue_t processingQueue = dispatch_queue_create("processingQueue2", NULL); 

          [audioWriterInput requestMediaDataWhenReadyOnQueue:processingQueue usingBlock:^{ 

           while (audioWriterInput.readyForMoreMediaData) 
           { 
            CMSampleBufferRef sampleBuffer; 

            if ([audioReader status] == AVAssetReaderStatusReading && 
             (sampleBuffer = [audioReaderOutput copyNextSampleBuffer])) { 

             [audioWriterInput appendSampleBuffer:sampleBuffer]; 
             CFRelease(sampleBuffer); 
            } 

            else 
            { 

             [audioWriterInput markAsFinished]; 

             if ([audioReader status] == AVAssetReaderStatusCompleted) 
             { 
              [videoWriter finishWritingWithCompletionHandler:^() 
              { 
               NSLog(@"Output URl : %@",outputURL); 
              }]; 
             } 
            } 
           } 

          } 
           ];      
        } 
       } 
      } 
     } 

     ]; 


    } 
+0

請幫我解決這個問題,儘快 – Dipen 2014-08-29 08:11:28

+0

退房這樣的回答: https://stackoverflow.com/questions/27075391/need-assistance-regarding-video-compression-in-ios/27098739#27098739 – 2014-11-24 06:04:18

回答

2

您可以使用下面的參數來壓縮視頻的質量。

  • AVAssetExportPresetLowQuality
  • AVAssetExportPresetMediumQuality
  • AVAssetExportPresetHighestQuality

代碼:

- (void)CompressVideo 
{ 
    if(firstAsset !=nil) 
    { 
     //Create AVMutableComposition Object.This object will hold our multiple AVMutableCompositionTrack. 
     AVMutableComposition* mixComposition = [[AVMutableComposition alloc] init]; 

     //  http://stackoverflow.com/questions/22715881/merge-video-files-with-their-original-audio-in-ios 

     //VIDEO TRACK 
     AVMutableCompositionTrack *firstTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeVideo preferredTrackID:kCMPersistentTrackID_Invalid]; 
     [firstTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, firstAsset.duration) ofTrack:[[firstAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] atTime:kCMTimeZero error:nil]; 

     //For Audio Track inclusion 
     //============================================================================================ 
     NSArray *arr = [firstAsset tracksWithMediaType:AVMediaTypeAudio]; 
     AVMutableCompositionTrack *audioTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid]; 
     [audioTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, firstAsset.duration) ofTrack:[arr lastObject] atTime:kCMTimeZero error:nil]; 
     //=============================================================================================== 

     AVMutableVideoCompositionInstruction *MainInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction]; 
     MainInstruction.timeRange = CMTimeRangeMake(kCMTimeZero, firstAsset.duration); 

     //FIXING ORIENTATION// 
     AVMutableVideoCompositionLayerInstruction *FirstlayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:firstTrack]; 
     AVAssetTrack *FirstAssetTrack = [[firstAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0]; 
     UIImageOrientation FirstAssetOrientation_ = UIImageOrientationUp; 
     BOOL isFirstAssetPortrait_ = NO; 

     CGAffineTransform firstTransform = FirstAssetTrack.preferredTransform; 
     if(firstTransform.a == 0 && firstTransform.b == 1.0 && firstTransform.c == -1.0 && firstTransform.d == 0) {FirstAssetOrientation_= UIImageOrientationRight; isFirstAssetPortrait_ = YES;} 
     if(firstTransform.a == 0 && firstTransform.b == -1.0 && firstTransform.c == 1.0 && firstTransform.d == 0) {FirstAssetOrientation_ = UIImageOrientationLeft; isFirstAssetPortrait_ = YES;} 
     if(firstTransform.a == 1.0 && firstTransform.b == 0 && firstTransform.c == 0 && firstTransform.d == 1.0) {FirstAssetOrientation_ = UIImageOrientationUp;} 
     if(firstTransform.a == -1.0 && firstTransform.b == 0 && firstTransform.c == 0 && firstTransform.d == -1.0) {FirstAssetOrientation_ = UIImageOrientationDown; 
     } 
     CGFloat FirstAssetScaleToFitRatio = VideoWidth/FirstAssetTrack.naturalSize.width; 

     if(isFirstAssetPortrait_) 
     { 
      FirstAssetScaleToFitRatio = VideoWidth/FirstAssetTrack.naturalSize.height; 
      CGAffineTransform FirstAssetScaleFactor = CGAffineTransformMakeScale(FirstAssetScaleToFitRatio,FirstAssetScaleToFitRatio); 
      [FirstlayerInstruction setTransform:CGAffineTransformConcat(FirstAssetTrack.preferredTransform, FirstAssetScaleFactor) atTime:kCMTimeZero]; 
     } 
     else 
     { 
      CGAffineTransform FirstAssetScaleFactor = CGAffineTransformMakeScale(FirstAssetScaleToFitRatio,FirstAssetScaleToFitRatio); 
      [FirstlayerInstruction setTransform:CGAffineTransformConcat(CGAffineTransformConcat(FirstAssetTrack.preferredTransform, FirstAssetScaleFactor),CGAffineTransformMakeTranslation(0, 160)) atTime:kCMTimeZero]; 
     } 
     [FirstlayerInstruction setOpacity:0.0 atTime:firstAsset.duration]; 

     MainInstruction.layerInstructions = [NSArray arrayWithObjects:FirstlayerInstruction,nil]; 

     AVMutableVideoComposition *MainCompositionInst = [AVMutableVideoComposition videoComposition]; 
     MainCompositionInst.instructions = [NSArray arrayWithObject:MainInstruction]; 
     MainCompositionInst.frameDuration = CMTimeMake(1, 30); 
     //  MainCompositionInst.renderSize = CGSizeMake(VideoWidth, 900); 
     MainCompositionInst.renderSize = CGSizeMake(VideoWidth, [UIScreen mainScreen].bounds.size.height); 

     NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES); 
     NSString *documentsDirectory = [paths objectAtIndex:0]; 
     NSString *myPathDocs = [documentsDirectory stringByAppendingPathComponent:@"CompressedVideo.mov"]; 

     NSLog(@"myPath Docs : %@",myPathDocs); 

     NSURL *url = [NSURL fileURLWithPath:myPathDocs]; 

     if ([[NSFileManager defaultManager] fileExistsAtPath:myPathDocs]) 
     { 
      NSError *error; 
      [[NSFileManager defaultManager] removeItemAtPath:myPathDocs error:&error]; 
     } 

     //Movie Quality 
     //================================================== 
     AVAssetExportSession *exporter = [[AVAssetExportSession alloc]  initWithAsset:mixComposition presetName:AVAssetExportPresetMediumQuality]; 
     //================================================== 

     exporter.outputURL=url; 

     //Movie Type 
     //================================================== 
     exporter.outputFileType = AVFileTypeQuickTimeMovie; 
     //================================================== 
     exporter.videoComposition = MainCompositionInst; 
     exporter.shouldOptimizeForNetworkUse = YES; 
     [exporter exportAsynchronouslyWithCompletionHandler:^ 
     { 
      dispatch_async(dispatch_get_main_queue(),^
          { 
           videoUrToUload = url; 
           [self exportDidFinish:exporter]; 
          }); 
     }]; 
    } 
} 

- (void)exportDidFinish:(AVAssetExportSession *)session 
{ 
    if(session.status == AVAssetExportSessionStatusCompleted) 
    { 
     //Store URL Somewhere using session.url 
    } 
} 
+0

這段代碼工作得很好。它也會以較低質量轉換視頻。 – Dipen 2015-04-17 14:09:13

1

我得到了同樣的崩潰問題,而是做了一些變化,這種方法爲我工作以後。只需更換此方法你上面的方法...

- (void)convertVideoToLowQuailtyWithInputURL:(NSURL*)inputURL 
            outputURL:(NSURL*)outputURL 
{ 
    //setup video writer 
    AVAsset *videoAsset = [[AVURLAsset alloc] initWithURL:inputURL options:nil]; 

    AVAssetTrack *videoTrack = [[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0]; 

    CGSize videoSize = videoTrack.naturalSize; 

    NSDictionary *videoWriterCompressionSettings = [NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithInt:1250000], AVVideoAverageBitRateKey, nil]; 

    NSDictionary *videoWriterSettings = [NSDictionary dictionaryWithObjectsAndKeys:AVVideoCodecH264, AVVideoCodecKey, videoWriterCompressionSettings, AVVideoCompressionPropertiesKey, [NSNumber numberWithFloat:videoSize.width], AVVideoWidthKey, [NSNumber numberWithFloat:videoSize.height], AVVideoHeightKey, nil]; 

    AVAssetWriterInput* videoWriterInput = [AVAssetWriterInput 
              assetWriterInputWithMediaType:AVMediaTypeVideo 
              outputSettings:videoWriterSettings]; 

    videoWriterInput.expectsMediaDataInRealTime = YES; 

    videoWriterInput.transform = videoTrack.preferredTransform; 

    AVAssetWriter *videoWriter = [[AVAssetWriter alloc] initWithURL:outputURL fileType:AVFileTypeQuickTimeMovie error:nil]; 

    [videoWriter addInput:videoWriterInput]; 

    //setup video reader 
    NSDictionary *videoReaderSettings = [NSDictionary dictionaryWithObject:[NSNumber numberWithInt:kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange] forKey:(id)kCVPixelBufferPixelFormatTypeKey]; 

    AVAssetReaderTrackOutput *videoReaderOutput = [[AVAssetReaderTrackOutput alloc] initWithTrack:videoTrack outputSettings:videoReaderSettings]; 

    AVAssetReader *videoReader = [[AVAssetReader alloc] initWithAsset:videoAsset error:nil]; 

    [videoReader addOutput:videoReaderOutput]; 

    //setup audio writer 
    AVAssetWriterInput* audioWriterInput = [AVAssetWriterInput 
              assetWriterInputWithMediaType:AVMediaTypeAudio 
              outputSettings:nil]; 

    audioWriterInput.expectsMediaDataInRealTime = NO; 

    [videoWriter addInput:audioWriterInput]; 

    //setup audio reader 
    AVAssetTrack* audioTrack = [[videoAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0]; 

    AVAssetReaderOutput *audioReaderOutput = [AVAssetReaderTrackOutput assetReaderTrackOutputWithTrack:audioTrack outputSettings:nil]; 

    AVAssetReader *audioReader = [AVAssetReader assetReaderWithAsset:videoAsset error:nil]; 

    [audioReader addOutput:audioReaderOutput]; 

    [videoWriter startWriting]; 

    //start writing from video reader 
    [videoReader startReading]; 

    [videoWriter startSessionAtSourceTime:kCMTimeZero]; 

    dispatch_queue_t processingQueue = dispatch_queue_create("processingQueue1", NULL); 

    [videoWriterInput requestMediaDataWhenReadyOnQueue:processingQueue usingBlock: 
    ^{ 

     while ([videoWriterInput isReadyForMoreMediaData]) { 

      CMSampleBufferRef sampleBuffer; 

      if ([videoReader status] == AVAssetReaderStatusReading && 
       (sampleBuffer = [videoReaderOutput copyNextSampleBuffer])) { 

       [videoWriterInput appendSampleBuffer:sampleBuffer]; 
       CFRelease(sampleBuffer); 
      } 

      else { 

       [videoWriterInput markAsFinished]; 

       if ([videoReader status] == AVAssetReaderStatusCompleted) { 
        if ([audioReader status] == AVAssetReaderStatusReading || [audioReader status] == AVAssetReaderStatusCompleted) { 

        } 
        else{ 
         //start writing from audio reader 
         [audioReader startReading]; 

         [videoWriter startSessionAtSourceTime:kCMTimeZero]; 

         dispatch_queue_t processingQueue = dispatch_queue_create("processingQueue2", NULL); 

         [audioWriterInput requestMediaDataWhenReadyOnQueue:processingQueue usingBlock:^{ 

          while (audioWriterInput.readyForMoreMediaData) { 

           CMSampleBufferRef sampleBuffer; 

           if ([audioReader status] == AVAssetReaderStatusReading && 
            (sampleBuffer = [audioReaderOutput copyNextSampleBuffer])) { 

            [audioWriterInput appendSampleBuffer:sampleBuffer]; 
            CFRelease(sampleBuffer); 
           } 
           else { 

            [audioWriterInput markAsFinished]; 

            if ([audioReader status] == AVAssetReaderStatusCompleted) { 

             [videoWriter finishWritingWithCompletionHandler:^(){ 
              //           [self sendMovieFileAtURL:outputURL]; 
              NSString *moviePath = [outputURL path]; 
              if (UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(moviePath)) { 
               UISaveVideoAtPathToSavedPhotosAlbum(moviePath, self, 
                        @selector(video:didFinishSavingWithError:contextInfo:), nil); 

              } 
             }]; 
             break; 
            } 
           } 
          } 

         } 
          ]; 
        } 
       } 

      } 
     } 
    } 
    ]; 
} 
+3

你的代碼正在工作,但它產生一個損壞的視頻意味着它無法運行任何媒體播放器,並在一段時間後出現內存警告,並終止應用程序。 – 2014-11-22 08:00:02

+0

出於某種原因,這段代碼在我第一次稱它時可以正常工作。如果我第二次調用它,它會在[audioReader startReading]中崩潰;說「閱讀已經開始後不能再打電話」 – Jan 2015-01-21 18:02:13

+0

仍然崩潰的問題在那裏。 – Dipen 2015-04-17 14:04:54