繁体   English   中英

控制器可以捕获模型抛出的异常吗?

[英]Can a controller catch an exception thrown from a model?

嗯,这在技术上是可行的,但这会打破MVC架构吗?

我不确定在控制器和型号之间是否建议使用这种类型的通信。 我将使用一个简单的示例和两种方法来描述它:

选项1(模型抛出异常,控制器捕获它):

class Controller {
  private $model;

  public function save($data) {
     try {
         $this->model->save($data); 
     } catch (Exception $e) {  
         // handle exception
     }
  }
}

class Model {
  public function save($data) {
     // Call to internal function to save data in BD
     if (! $this->_save($data)) throw new Exception('Error saving data');
  }
}

选项2(控制器完全处理异常):

class Controller {
  private $model;

  public function save($data) {
     try {
         if (! $this->model->save($data)) throw new Exception('Error saving data'); 
     } catch (Exception $e) { 
         // handle exception
     }
  }
}

class Model {
  public function save($data) {
     // Call to internal function to save data in BD
     if (! $this->_save($data)) return false;
  }
}

**

一些回复后编辑:

**

这些是根据您的建议解决它的其他方法。 我希望不要让事情变得太复杂。

选项3(模型完全处理异常,正如Ray所说.KingCrunch还建议在模型中做得更好)

class Controller {
  private $model;

  public function save($data) {

     if (! $this->model->save($data)) {
         // possible action: redirect to the form with an error message
     }

  }
}

class Model {
  public function save($data) {
     try {
         if (! $this->_save($data)) throw new Exception('Error saving data'); 
     } catch (Exception $e) { 
         // handle exception
         return false;
     }
     return true; 
  }
}

选项4(控制器获取模型抛出的自定义子异常,如shiplu.mokadd.im所说。)

class Controller {
  private $model;

  public function save($data) {
     try {
         $this->model->save($data); 
     } catch (Exception $e) {  
         if ($e instanceof ValidationException) {
            // handle validation error
         }
         elseif ($e instanceof DBStorageException) {
            // handle DB error
         }
     }
  }
}

class Model {
  public function save($data) {
     if (! $this->_validate($data)) {
        throw new ValidationException ('Validation error');
     }
     if (! $this->_save($data)) {
        throw new DBStorageException ('Storage error');
     }
  }
}

模型可以抛出异常,控制器或视图应该捕获它 否则你永远不知道一切都在那里正常工作。 所以使用第一个选项。 但请确保您正在抛出对控制器和View有意义的正确抽象异常

为了说明上面的粗线,请参阅模型中使用的这两个throw语句。

 throw new Exception('SQL Error: '.$mysqli->error()); // dont use it
 throw new DuplicateFieldException('Duplicate username'); // use this

第二个示例未显示内部错误。 相反,它隐藏了它。 控制者永远不应该知道内部发生了什么。

在您的代码中,您将单个模型绑定到单个控制器。 控制器不代表单个模型。 它使用模型。 它可以使用任意数量的模型。 因此,不要将单个模型与具有变量的private $model的控制器捆绑在一起。

绝对是第一选择。 一些单词:

  • 这是控制器的工作......好吧,控制。 这意味着,应该注意,至少会出现一个有用的错误消息。 应用程序的其他部分可以在它们能够处理特殊情况之前执行。 这包括模型本身:如果它能够处理它,它应该这样做。

  • save()表示“保存”。 不要滥用状态信息的返回值。 当该方法无法save()它是一个例外,当一个方法不必给你一些东西时,它就不应该给你一些东西。

我更喜欢选项3。

模型应该捕获异常,尝试解决它,如果没有将其渗透到控制器,但只有在控制器可以解决和恢复的情况下。 在这种情况下,(某种类型的DB保存失败)在模型中捕获它返回false应该是保存错误的足够分辨率并且提供足够的值以便控制器在保存时知道出错。

控制器不需要担心模型如何实现保存的实现细节。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM