草庐IT

c# - 错误处理 我应该抛出异常吗?还是源头处理?

coder 2024-05-24 原文

我有这种格式

asp.net MVC View -> 服务层 -> 存储库。

因此 View 调用服务层,其中包含业务/验证逻辑,后者又调用存储库。

现在我的服务层方法通常有一个 bool 返回类型,这样我就可以在数据库查询成功时返回 true。或者如果它失败了。然后向用户显示一条通用消息。

我当然会用 elmah 记录错误。但是我不确定我应该如何做到这一点。

像现在一样,我的存储库有用于更新、创建、删除的 void 返回类型。

也就是说,如果更新失败,我是否应该在我的存储库中有一个 try/catch 来抛出错误,然后我的服务层会捕获它并发出 elmah 信号并返回 false?

或者我应该让这些存储库方法返回一个“bool”,尝试/捕获存储库中的错误,然后将“true”或“false”返回给服务层,然后服务层返回“true”或“false”观点?

异常处理仍然让我困惑如何处理错误以及何时抛出错误以及何时捕获错误。

最佳答案

我一直使用的经验法则是:

  • 在低级别,当操作由于异常情况而无法完成时抛出。
  • 在中间层,捕获多个异常类型并重新包装为一个异常类型。
  • 在最后负责的时刻处理异常。
  • 文档!

这是一个多层 ASP.NET MVC 应用程序(UI、 Controller 、逻辑、安全、存储库)的伪代码示例:

  1. 用户点击提交按钮。
  2. 执行 Controller 操作并调用逻辑(业务)层。
  3. 逻辑方法使用当前用户凭据调用安全性
    • 用户无效
      • 安全层抛出SecurityException
      • 逻辑层捕获,并用更通用的错误消息包装在 LogicException 中
      • Controller 捕获 LogicException,重定向到错误页面。
    • 用户有效且安全返回
  4. 逻辑层调用存储库以完成操作
    • 存储库失败
      • 存储库抛出 RepositoryException
      • 逻辑层捕获,并用更通用的错误消息包装在 LogicException 中
      • Controller 捕获 LogicException,重定向到错误页面。
    • 存储成功
  5. 逻辑层返回
  6. Controller 重定向到成功 View 。

注意,逻辑层只抛出一个异常类型——LogicException。任何冒泡的低级异常都被捕获,包装在一个新的 LogicException 实例中,然后被抛出。这给了我们很多优势。

首先,堆栈跟踪是可访问的。其次,调用者只需要处理一个异常类型而不是多个异常。第三,可以对技术异常消息进行处理以显示给用户,同时仍保留原始异常消息。最后,只有负责处理用户输入的代码才能真正了解用户的意图,并确定操作失败时的适当响应。 Repository 不知道 UI 是否应该显示错误页面或请求用户使用不同的值重试。 Controller 知道这一点。


顺便说一句,没有人说你不能这样做:

try
{
  var result = DoSomethingOhMyWhatIsTheReturnType();
}
catch(LogicException e)
{
  if(e.InnerException is SqlException)
  {
    // handle sql exceptions
  }else if(e.InnerException is InvalidCastException)
  {
    // handle cast exceptions
  }
  // blah blah blah
}

关于c# - 错误处理 我应该抛出异常吗?还是源头处理?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1352359/

有关c# - 错误处理 我应该抛出异常吗?还是源头处理?的更多相关文章

  1. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  2. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  3. ruby - 检查 "command"的输出应该包含 NilClass 的意外崩溃 - 2

    为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar

  4. ruby-on-rails - 迷你测试错误 : "NameError: uninitialized constant" - 2

    我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test

  5. ruby-on-rails - Rails - 乐观锁定总是触发 StaleObjectError 异常 - 2

    我正在学习Rails,并阅读了关于乐观锁的内容。我已将类型为integer的lock_version列添加到我的articles表中。但现在每当我第一次尝试更新记录时,我都会收到StaleObjectError异常。这是我的迁移:classAddLockVersionToArticle当我尝试通过Rails控制台更新文章时:article=Article.first=>#我这样做:article.title="newtitle"article.save我明白了:(0.3ms)begintransaction(0.3ms)UPDATE"articles"SET"title"='dwdwd

  6. ruby-on-rails - 如何在 Rails View 上显示错误消息? - 2

    我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c

  7. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

    我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

  8. ruby-on-rails - 错误 : Error installing pg: ERROR: Failed to build gem native extension - 2

    我克隆了一个rails仓库,我现在正尝试捆绑安装背景:OSXElCapitanruby2.2.3p173(2015-08-18修订版51636)[x86_64-darwin15]rails-v在您的Gemfile中列出的或native可用的任何gem源中找不到gem'pg(>=0)ruby​​'。运行bundleinstall以安装缺少的gem。bundleinstallFetchinggemmetadatafromhttps://rubygems.org/............Fetchingversionmetadatafromhttps://rubygems.org/...Fe

  9. ruby - #之间? Cooper 的 *Beginning Ruby* 中的错误或异常 - 2

    在Cooper的书BeginningRuby中,第166页有一个我无法重现的示例。classSongincludeComparableattr_accessor:lengthdef(other)@lengthother.lengthenddefinitialize(song_name,length)@song_name=song_name@length=lengthendenda=Song.new('Rockaroundtheclock',143)b=Song.new('BohemianRhapsody',544)c=Song.new('MinuteWaltz',60)a.betwee

  10. ruby-on-rails - 每次我尝试部署时,我都会得到 - (gcloud.preview.app.deploy) 错误响应 : [4] DEADLINE_EXCEEDED - 2

    我是Google云的新手,我正在尝试对其进行首次部署。我的第一个部署是RubyonRails项目。我基本上是在关注thisguideinthegoogleclouddocumentation.唯一的区别是我使用的是我自己的项目,而不是他们提供的“helloworld”项目。这是我的app.yaml文件runtime:customvm:trueentrypoint:bundleexecrackup-p8080-Eproductionconfig.ruresources:cpu:0.5memory_gb:1.3disk_size_gb:10当我转到我的项目目录并运行gcloudprevie

随机推荐