2017-06-15 5 views
2

スレッド1:EXC_BAD_INSTRACTIONにエラーが発生しました。 私はSwiftアプリを作っていて、自分のアプリでDjangoサーバーに画像をアップロードするシステムを作りたいと思っています。画像を選択せず​​にサーバーに画像を送る「送信」ボタンを置くと、このエラーが発生します。 error さて、PhotoControllerは(それはシステムのためである)スレッド1:EXC_BAD_INSTRACTION

import Foundation 
import MobileCoreServices 
import UIKit 
class PhotoController:UIViewController,UINavigationControllerDelegate,UIImagePickerControllerDelegate{ 

    @IBOutlet weak var myActivityIndicator: UIActivityIndicatorView! 
    @IBOutlet weak var label: UILabel! 
    @IBOutlet weak var myImageView: UIImageView! 
    private var imagePicker:UIImagePickerController! 

    @IBAction func uploadButtonTapped(_ sender: Any) { 
     myImageUploadRequest() 
    } 
    override func viewDidLoad() { 
     super.viewDidLoad() 
     label.adjustsFontSizeToFitWidth = true 
     label.minimumScaleFactor = 0.5 
     label.text = "Tap the PhotoSelect or Camera to upload a picture" 
    } 

    @IBAction func PhotoSelect(_ sender: Any) { 
     let myPickerController = UIImagePickerController() 
     myPickerController.delegate = self; 
     myPickerController.sourceType = UIImagePickerControllerSourceType.photoLibrary 

     self.present(myPickerController, animated: true, completion: nil) 

    } 

    @IBAction func Camera(_ sender: Any) { 
     let sourceType:UIImagePickerControllerSourceType = UIImagePickerControllerSourceType.camera 
     // カメラが利用可能かチェック 
     if UIImagePickerController.isSourceTypeAvailable(UIImagePickerControllerSourceType.camera){ 
      // インスタンスの作成 
      let cameraPicker = UIImagePickerController() 
      cameraPicker.sourceType = sourceType 
      cameraPicker.delegate = self 
      self.present(cameraPicker, animated: true, completion: nil) 

     } 
     else{ 
      label.text = "error" 
     } 
    } 
    // 撮影が完了時した時に呼ばれる 
    func imagePickerController(_ imagePicker: UIImagePickerController, didFinishPickingMediaWithInfo info: [String : Any]) { 

     if let pickedImage = info[UIImagePickerControllerOriginalImage] as? UIImage { 
      myImageView.contentMode = .scaleAspectFit 
      myImageView.image = pickedImage 

     } 

     //閉じる処理 
     imagePicker.dismiss(animated: true, completion: nil) 
     label.text = "Tap the Send to save a picture" 

    } 

    // 撮影がキャンセルされた時に呼ばれる 
    func imagePickerControllerDidCancel(_ picker: UIImagePickerController) { 
     picker.dismiss(animated: true, completion: nil) 
     label.text = "Canceled" 
    } 

    override func didReceiveMemoryWarning() { 
     super.didReceiveMemoryWarning() 
     // Dispose of any resources that can be recreated. 
    } 

    func myImageUploadRequest() 
    { 

     let myUrl = NSURL(string: "http://localhost:8000/admin/accounts/post/"); 
     let request = NSMutableURLRequest(url:myUrl! as URL); 
     request.httpMethod = "POST"; 
     //ユーザーごとに割り振りたい 
     let param = [ 
      "firstName" : "Sergey", 
      "lastName" : "Kargopolov", 
      "userId" : "9" 
     ] 

     let boundary = generateBoundaryString() 

     request.setValue("multipart/form-data; boundary=\(boundary)", forHTTPHeaderField: "Content-Type") 


     let imageData = UIImageJPEGRepresentation(myImageView.image!, 1) 

     if(imageData==nil) { return; } 

     request.httpBody = createBodyWithParameters(parameters: param, filePathKey: "file", imageDataKey: imageData! as NSData, boundary: boundary) as Data 


     myActivityIndicator.startAnimating(); 

     let task = URLSession.shared.dataTask(with: request as URLRequest) { 
      data, response, error in 

      if error != nil { 
       print("error=\(error)") 
       return 
      } 

      // You can print out response object 
      print("******* response = \(response)") 

      // Print out reponse body 
      let responseString = NSString(data: data!, encoding: String.Encoding.utf8.rawValue) 
      print("****** response data = \(responseString!)") 

      do { 
       let json = try JSONSerialization.jsonObject(with: data!, options: []) as? NSDictionary 

       DispatchQueue.main.async { 
        self.myActivityIndicator.stopAnimating() 
        self.myImageView.image = nil; 
       } 

      }catch 
      { 
       print(error) 
      } 

     } 

     task.resume() 
    } 
    func createBodyWithParameters(parameters: [String: String]?, filePathKey: String?, imageDataKey: NSData, boundary: String) -> NSData { 
     let body = NSMutableData(); 

     if parameters != nil { 
      for (key, value) in parameters! { 
       body.appendString(string: "--\(boundary)\r\n") 
       body.appendString(string: "Content-Disposition: form-data; name=\"\(key)\"\r\n\r\n") 
       body.appendString(string: "\(value)\r\n") 
      } 
     } 

     let filename = "user-profile.jpg" 
     let mimetype = "image/jpg" 

     body.appendString(string: "--\(boundary)\r\n") 
     body.appendString(string: "Content-Disposition: form-data; name=\"\(filePathKey!)\"; filename=\"\(filename)\"\r\n") 
     body.appendString(string: "Content-Type: \(mimetype)\r\n\r\n") 
     body.append(imageDataKey as Data) 
     body.appendString(string: "\r\n") 
     body.appendString(string: "--\(boundary)--\r\n") 

     return body 
    } 

    func generateBoundaryString() -> String { 
     return "Boundary-\(NSUUID().uuidString)" 
    } 

} 
extension NSMutableData { 

    func appendString(string: String) { 
     let data = string.data(using: String.Encoding.utf8, allowLossyConversion: true) 
     append(data!) 
    } 
} 

トレースバックは、このエラーが起こる理由は、私がこの部分

に予期しない何かのためにif文を書いたので、私は理解できない

fatal error: unexpectedly found nil while unwrapping an Optional value 
(lldb) 

です

let task = URLSession.shared.dataTask(with: request as URLRequest) { 
      data, response, error in 

      if error != nil { 
       print("error=\(error)") 
       return 
      } 
      // You can print out response object 
      print("******* response = \(response)") 
      // Print out reponse body 
      let responseString = NSString(data: data!, encoding: String.Encoding.utf8.rawValue) 
      print("****** response data = \(responseString!)") 
      do { 
       let json = try JSONSerialization.jsonObject(with: data!, options: []) as? NSDictionary 
       DispatchQueue.main.async { 
        self.myActivityIndicator.stopAnimating() 
        self.myImageView.image = nil; 
       } 

      }catch 
      { 
       print(error) 
      } 

     } 

どうすれば解決できますか?私の目標は、自分のアプリケーションでDjangoサーバーに画像をアップロードするシステムを作ることです。 (ちなみに、画像を選択しなかった場合、「ERROR」のような警告を表示したい)

+0

ユーザが少なくとも1つの画像を選択しない限り、ボタンを無効にする必要があります。 –

+0

@IBOutletと同じようにイメージビューを宣言しようとしていますvar myImageView:UIImageView? – KKRocks

+0

また、iboutlet接続を確認してください。それが無ければ、このエラーがスローされます。 – KKRocks

答えて

1

OKここので、あなたのアプリのクラッシュのためにはnilチェックを入れることができます。

  1. myImageViewは実際に画像がありません。

上記のコメントで示唆したようmyImageView @IBOutletが正しく接続されていないあなたはここにチェックがあります。

if(imageData==nil) { return; } 

をしかし、それは遅すぎるあなたが見ることができるようにちょうど1行です:)

myImageView.imageのアンラップを行い、それが何かである場合に限り、続行します。それ以外の場合は、機能から復帰します。そのようになります

guard let myImage = myImageView.image else { 
    return 
} 

今、あなたはあなたのイメージが有効な値を持っており、安全に継続することができることを知っています。

だから、あなたの全体の機能は次のようになります。あなたを助け

func myImageUploadRequest() 
{ 
    let myUrl = NSURL(string: "http://localhost:8000/admin/accounts/post/"); 
    let request = NSMutableURLRequest(url:myUrl! as URL); 
    request.httpMethod = "POST"; 
    //ユーザーごとに割り振りたい 
    let param = [ 
     "firstName" : "Sergey", 
     "lastName" : "Kargopolov", 
     "userId" : "9" 
    ] 

    let boundary = generateBoundaryString() 

    request.setValue("multipart/form-data; boundary=\(boundary)", forHTTPHeaderField: "Content-Type") 

    guard let myImage = myImageView.image else { 
     return 
    } 

    let imageData = UIImageJPEGRepresentation(myImage, 1) 

    if(imageData==nil) { return; } 

    request.httpBody = createBodyWithParameters(parameters: param, filePathKey: "file", imageDataKey: imageData! as NSData, boundary: boundary) as Data 


    myActivityIndicator.startAnimating(); 

    let task = URLSession.shared.dataTask(with: request as URLRequest) { 
     data, response, error in 

     if error != nil { 
      print("error=\(error)") 
      return 
     } 

     // You can print out response object 
     print("******* response = \(response)") 

     // Print out reponse body 
     let responseString = NSString(data: data!, encoding: String.Encoding.utf8.rawValue) 
     print("****** response data = \(responseString!)") 

     do { 
      let json = try JSONSerialization.jsonObject(with: data!, options: []) as? NSDictionary 

      DispatchQueue.main.async { 
       self.myActivityIndicator.stopAnimating() 
       self.myImageView.image = nil; 
      } 

     }catch 
     { 
      print(error) 
     } 

    } 

    task.resume() 
} 

希望。

2

エラーは、myImageViewがnilであり、アンラップしようとしたために発生します。それ。 myImageView.imageは、いくつかのことが原因で発生することができ皆無であることを示していること

let imageData = UIImageJPEGRepresentation(myImageView.image!, 1) 

: あなたはmyImageView.image

if myImageView.image!= nil { 
let imageData = UIImageJPEGRepresentation(myImageView.image!, 1) 

     if(imageData==nil) { return; } 
} 
関連する問題