ASP.NET MVC에서 컨트롤러 메서드를 오버로드 할 수 있는지 궁금합니다. 시도 할 때마다 아래 오류가 발생합니다. 두 가지 방법은 서로 다른 주장을 받아들입니다. 할 수없는 일입니까?
컨트롤러 유형 'MyController'에서 'MyMethod'조치에 대한 현재 요청은 다음 조치 메소드간에 모호합니다.
ASP.NET MVC에서 컨트롤러 메서드를 오버로드 할 수 있는지 궁금합니다. 시도 할 때마다 아래 오류가 발생합니다. 두 가지 방법은 서로 다른 주장을 받아들입니다. 할 수없는 일입니까?
컨트롤러 유형 'MyController'에서 'MyMethod'조치에 대한 현재 요청은 다음 조치 메소드간에 모호합니다.
답변:
코드가 오버로드되도록하려면 속성을 사용할 수 있습니다.
[ActionName("MyOverloadedName")]
그러나 동일한 http 메소드에 다른 액션 이름을 사용해야합니다 (다른 사람들이 말했듯이). 그래서 그것은 단지 의미 론적입니다. 오히려 코드 또는 속성에 이름이 있습니까?
Phil은 이와 관련된 기사를 가지고 있습니다 : http://haacked.com/archive/2008/08/29/how-a-method-becomes-an-action.aspx
return View();
. 예를 들면 다음과 같습니다 return View("MyOverloadedName");
..
예. 각 컨트롤러 메소드 의 HttpGet
/ HttpPost
(또는 동등한 AcceptVerbs
속성)을 고유 한 것으로 즉, HttpGet
또는 HttpPost
둘 다로 설정 하여이 작업을 수행 할 수있었습니다 . 그렇게하면 요청 유형에 따라 사용할 방법을 알 수 있습니다.
[HttpGet]
public ActionResult Show()
{
...
}
[HttpPost]
public ActionResult Show( string userName )
{
...
}
내가 가진 제안 중 하나는 이와 같은 경우 두 개의 공용 Action 메소드가 코드 중복을 피하기 위해 의존하는 개인 구현을 갖는 것입니다.
Show()
메소드의 서명이 다르기 때문에 이것이 제대로 작동하는지 확인하십시오 . Get 버전으로 정보를 보내야 할 경우 Get 및 Post 버전은 동일한 서명으로 끝나 ActionName
므로이 게시물에 언급 된 속성이나 다른 수정 사항 중 하나 가 필요합니다 .
ActionNameAttribute
. 실제로, 나는 그 경우가 거의 없다는 것을 알았습니다.
여기 당신이 할 수있는 또 다른 것이 있습니다 ... 당신은 매개 변수를 가질 수 있고 가질 수없는 방법을 원합니다.
이것을 시도해보십시오 ...
public ActionResult Show( string username = null )
{
...
}
이것은 나를 위해 일했습니다 ...이 방법에서는 실제로 들어오는 매개 변수가 있는지 테스트 할 수 있습니다.
string
Null을 허용 할 수 없습니다.)
string
될 수 없다는 것은 여전히 사실입니다 nullable
. 그러나 그것은 될 수 있습니다 null
! 어느 쪽이든 내가 진심으로 초기 의견을 게시했습니다.
아니요, 아니요 및 아니요. "LoadCustomer"가 오버로드 된 컨트롤러 코드를 아래에서 시도하십시오.
public class CustomerController : Controller
{
//
// GET: /Customer/
public ActionResult LoadCustomer()
{
return Content("LoadCustomer");
}
public ActionResult LoadCustomer(string str)
{
return Content("LoadCustomer with a string");
}
}
"LoadCustomer"조치를 호출하려고하면 아래 그림과 같이 오류가 발생합니다.
다형성은 C # 프로그래밍의 일부이며 HTTP는 프로토콜입니다. HTTP는 다형성을 이해하지 못합니다. HTTP는 개념이나 URL에서 작동하며 URL은 고유 한 이름 만 가질 수 있습니다. 따라서 HTTP는 다형성을 구현하지 않습니다.
같은 것을 고치려면 "ActionName"속성을 사용해야합니다.
public class CustomerController : Controller
{
//
// GET: /Customer/
public ActionResult LoadCustomer()
{
return Content("LoadCustomer");
}
[ActionName("LoadCustomerbyName")]
public ActionResult LoadCustomer(string str)
{
return Content("LoadCustomer with a string");
}
}
따라서 "Customer / LoadCustomer"URL을 호출하면 "LoadCustomer"조치가 호출되고 URL 구조 "Customer / LoadCustomerByName"을 사용하여 "LoadCustomer (string str)"가 호출됩니다.
이 코드 프로젝트 기사에서 가져온 위의 답변-> MVC 작업 오버로드
이 문제를 극복하기 위해 할 수 쓰기 ActionMethodSelectorAttribute
을 검사하는 MethodInfo
각 작업과 게시 된 양식 값과 비교를 한 후 (물론, 버튼 이름 제외) 형태의 값이 일치하지 않은 어떤 방법을 거부합니다.
예를 들면 다음과 같습니다 .- http://blog.abodit.com/2010/02/asp-net-mvc-ambiguous-match/
그러나 이것은 좋은 생각이 아닙니다.
내가 아는 한 다른 http 메소드를 사용할 때 동일한 메소드 만 가질 수 있습니다.
즉
[AcceptVerbs("GET")]
public ActionResult MyAction()
{
}
[AcceptVerbs("POST")]
public ActionResult MyAction(FormResult fm)
{
}
[HttpPost]
대신 속성을 사용하십시오 [AcceptVerbs("POST")]
.
MVC5 의 속성 라우팅 을 사용하여 이것을 달성했습니다 . 분명히 WebForms를 사용하여 수십 년 동안 웹을 개발 한 MVC를 처음 접했지만 다음과 같은 결과가 나에게 도움이되었습니다. 허용되는 답변과 달리 오버로드 된 모든 작업을 동일한 뷰 파일로 렌더링 할 수 있습니다.
먼저 App_Start / RouteConfig.cs에서 속성 라우팅을 활성화하십시오.
public class RouteConfig
{
public static void RegisterRoutes(RouteCollection routes)
{
routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
routes.MapMvcAttributeRoutes();
routes.MapRoute(
name: "Default",
url: "{controller}/{action}/{id}",
defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
);
}
}
선택적으로 기본 경로 접두어로 컨트롤러 클래스를 장식하십시오.
[RoutePrefix("Returns")]
public class ReturnsController : BaseController
{
//.......
그런 다음 공통 경로 및 매개 변수로 서로 과부하되는 컨트롤러 조치를 장식하십시오. 유형 제한 매개 변수를 사용하면 유형이 다른 ID와 동일한 URI 형식을 사용할 수 있습니다.
[HttpGet]
// Returns
public ActionResult Index()
{
//.....
}
[HttpGet]
[Route("View")]
// Returns/View
public ActionResult View()
{
// I wouldn't really do this but it proves the concept.
int id = 7026;
return View(id);
}
[HttpGet]
[Route("View/{id:int}")]
// Returns/View/7003
public ActionResult View(int id)
{
//.....
}
[HttpGet]
[Route("View/{id:Guid}")]
// Returns/View/99300046-0ba4-47db-81bf-ba6e3ac3cf01
public ActionResult View(Guid id)
{
//.....
}
이것이 도움이되고 누군가를 잘못된 길로 인도하지 않기를 바랍니다. :-)
당신은 하나를 사용할 수 있습니다 ActionResult
모두 처리 Post
및 Get
:
public ActionResult Example() {
if (Request.HttpMethod.ToUpperInvariant() == "GET") {
// GET
}
else if (Request.HttpMethod.ToUpperInvariant() == "POST") {
// Post
}
}
메소드 Get
와 Post
일치하는 서명이있는 경우 유용합니다 .
방금이 질문을 보았는데 지금은 나이가 들었지만 여전히 관련이 있습니다. 아이러니하게도,이 글에서 올바른 의견 하나는 MVC에서 자백 한 초보자가 글을 쓸 때 게시 한 것입니다. ASP.NET 문서조차도 완전히 정확하지는 않습니다. 큰 프로젝트가 있고 작업 방법을 성공적으로 오버로드했습니다.
간단한 {controller} / {action} / {id} 기본 라우트 패턴을 넘어 라우팅을 이해하면 컨트롤러 패턴을 고유 한 패턴을 사용하여 매핑 할 수 있습니다. 여기 누군가가 다형성에 대해 이야기하고 "HTTP는 다형성을 이해하지 못한다"고 말했지만 라우팅은 HTTP와 관련이 없습니다. 간단히 말해서 문자열 패턴 일치 메커니즘입니다.
이 작업을 수행하는 가장 좋은 방법은 다음과 같은 라우팅 속성을 사용하는 것입니다.
[RoutePrefix("cars/{country:length(3)}")]
public class CarHireController
{
[Route("{location}/{page:int=1}", Name = "CarHireLocation")]
public ActionResult Index(string country, string location, int page)
{
return Index(country, location, null, page);
}
[Route("{location}/{subLocation}/{page:int=1}", Name = "CarHireSubLocation")]
public ActionResult Index(string country, string location, string subLocation, int page)
{
//The main work goes here
}
}
이러한 작업은 같은 URL을 처리됩니다 /cars/usa/new-york
및 /cars/usa/texas/dallas
각각 제 1 및 제 2 인덱스 작업에 매핑합니다.
이 예제 컨트롤러를 살펴보면 위에서 언급 한 기본 경로 패턴을 넘어서는 것이 분명합니다. URL 구조가 코드 명명 규칙과 정확히 일치하면 기본값이 잘 작동하지만 항상 그런 것은 아닙니다. 코드는 도메인을 설명해야하지만 URL은 내용이 SEO 요구 사항과 같은 다른 기준을 기반으로해야하기 때문에 종종 더 나아가 야합니다.
기본 라우팅 패턴의 장점은 고유 한 경로를 자동으로 생성한다는 것입니다. URL은 고유 한 컨트롤러 유형 및 멤버와 일치하므로 컴파일러에서 적용합니다. 자신 만의 경로 패턴을 굴리면 고유성을 보장하고 작동하도록 신중하게 고려해야합니다.
중요 참고 사항 한 가지 단점은 UrlHelper.Action을 사용할 때와 같이 작업 이름을 기반으로하는 경우 오버로드 된 작업에 대한 URL을 생성하기 위해 라우팅을 사용하면 작동하지 않는다는 것입니다. 그러나 이름이 지정된 경로 (예 : UrlHelper.RouteUrl)를 사용하는 경우 작동합니다. 그리고 잘 알려진 소스에 따르면, 명명 된 경로를 사용하는 것은 어떻게 든 갈 수있는 방법입니다 ( http://haacked.com/archive/2010/11/21/named-routes-to-the-rescue.aspx/ ).
행운을 빕니다!
[ActionName ( "NewActionName")]을 사용하여 다른 이름으로 동일한 메소드를 사용할 수 있습니다.
public class HomeController : Controller
{
public ActionResult GetEmpName()
{
return Content("This is the test Message");
}
[ActionName("GetEmpWithCode")]
public ActionResult GetEmpName(string EmpCode)
{
return Content("This is the test Messagewith Overloaded");
}
}
과부하가 필요했습니다.
public ActionResult Index(string i);
public ActionResult Index(int groupId, int itemId);
내가 이것을 끝내는 데 충분한 논쟁이 거의 없었다.
public ActionResult Index(string i, int? groupId, int? itemId)
{
if (!string.IsNullOrWhitespace(i))
{
// parse i for the id
}
else if (groupId.HasValue && itemId.HasValue)
{
// use groupId and itemId for the id
}
}
특히 많은 주장이있는 경우 완벽한 솔루션이 아니지만 그것은 나에게 잘 작동합니다.
내 응용 프로그램에서도 동일한 문제에 직면했습니다. Modifiyig에 메소드 정보가 없으면 조치 헤드에 [ActionName ( "SomeMeaningfulName")]을 제공했습니다. 문제가 해결되었습니다
[ActionName("_EmployeeDetailsByModel")]
public PartialViewResult _EmployeeDetails(Employee model)
{
// Some Operation
return PartialView(model);
}
}
[ActionName("_EmployeeDetailsByModelWithPagination")]
public PartialViewResult _EmployeeDetails(Employee model,int Page,int PageSize)
{
// Some Operation
return PartialView(model);
}
기본 방법을 가상으로 작성
public virtual ActionResult Index()
재정의 된 메서드를 재정의로 만듭니다.
public override ActionResult Index()
편집 : 이것은 override 메소드가 OP의 의도가 아닌 파생 클래스에있는 경우에만 적용됩니다.
이 답변이 다른 스레드에 게시 된 것을 좋아합니다
이것은 주로 다른 컨트롤러에서 상속하고 기본 컨트롤러에서 액션을 재정의하려는 경우에 사용됩니다.
각 컨트롤러 방법에는 하나의 공개 서명 만 허용됩니다. 과부하를 시도하면 컴파일되지만 런타임 오류가 발생합니다.
오버로드 된 메소드 (작동 할)를 구별하거나 라우팅을 변경 하기 위해 다른 동사 ( [HttpGet]
및 [HttpPost]
속성 과 같은)를 사용하지 않으려는 경우 남은 방법은 다른 이름을 가진 다른 메소드를 제공하거나 기존 메소드 내부에 디스패치하십시오. 내가 한 방법은 다음과 같습니다.
한때 이전 버전과의 호환성을 유지해야하는 상황에 처했습니다. 원래 방법에는 두 개의 매개 변수가 필요했지만 새 매개 변수에는 하나만있었습니다. MVC가 더 이상 진입 점을 찾지 못해 예상 한 방식으로 오버로드하지 못했습니다.
이를 해결하기 위해 다음을 수행했습니다.
"단지"2 개의 문자열 매개 변수를 포함하는 새로운 공용 메소드를 작성했습니다. 그 사람은 발송자 역할을했습니다.
public ActionResult DoSomething(string param1, string param2)
{
if (string.IsNullOrEmpty(param2))
{
return DoSomething(ProductName: param1);
}
else
{
int oldId = int.Parse(param1);
return DoSomething(OldParam: param1, OldId: oldId);
}
}
private ActionResult DoSomething(string OldParam, int OldId)
{
// some code here
return Json(result);
}
private ActionResult DoSomething(string ProductName)
{
// some code here
return Json(result);
}
물론 이것은 해킹이므로 나중에 리팩터링해야합니다. 그러나 당분간은 나를 위해 일했습니다.
다음과 같은 디스패처를 작성할 수도 있습니다.
public ActionResult DoSomething(string action, string param1, string param2)
{
switch (action)
{
case "update":
return UpdateAction(param1, param2);
case "remove":
return DeleteAction(param1);
}
}
UpdateAction에는 2 개의 매개 변수가 필요하지만 DeleteAction에는 하나만 필요하다는 것을 알 수 있습니다.
지연 돼서 죄송합니다. 나는 같은 문제가 있었고 좋은 답변이있는 링크를 찾았습니다. 새로운 사람들을 도울 수 있습니다.
BinaryIntellect 웹 사이트 및 저자에 대한 모든 크레딧
기본적으로 네 가지 상황이 있습니다 : 다른 동사 사용 , 라우팅 사용 , [NoAction] 속성으로 과부하 표시 및 [ActionName]으로 활동 속성 이름 변경
그래서 그것은 당신의 요구와 상황에 달려 있습니다.
그러나 링크를 따르십시오.
링크 : http://www.binaryintellect.net/articles/8f9d9a8f-7abf-4df6-be8a-9895882ab562.aspx
다른 모델의 여러 조치에 POST하는 여러보기에 대해 하나의 GET 조치를 사용하려는 경우, 각 POST 조치마다 GET 조치를 추가하여 첫 번째 GET으로 경로 재지 정하여 새로 고침시 404를 방지하십시오.
장거리지만 일반적인 시나리오.