我的MVC5项目解决方案中有一个WebAPI控制器。 WebAPI有一个方法,它将特定文件夹中的所有文件作为Json列表返回:
[{"name":"file1.zip", "path":"c:\\"}, {...}]
从我的HomeController我想调用此方法,将Json响应转换为List<QDocument>
并将此列表返回到Razor视图。此列表可能为空:[]
如果文件夹中没有文件。
这是APIController:
public class DocumentsController : ApiController
{
#region Methods
/// <summary>
/// Get all files in the repository as Json.
/// </summary>
/// <returns>Json representation of QDocumentRecord.</returns>
public HttpResponseMessage GetAllRecords()
{
// All code to find the files are here and is working perfectly...
return new HttpResponseMessage()
{
Content = new StringContent(JsonConvert.SerializeObject(listOfFiles), Encoding.UTF8, "application/json")
};
}
}
这是我的HomeController:
public class HomeController : Controller
{
public Index()
{
// I want to call APi GetAllFiles and put the result to variable:
var files = JsonConvert.DeserializeObject<List<QDocumentRecord>>(API return Json);
}
}
最后,这是您需要的模型:
public class QDocumentRecord
{
public string id {get; set;}
public string path {get; set;}
.....
}
那我怎么能打这个电话呢?
答案 0 :(得分:17)
从我的HomeController我想调用此方法并将Json响应转换为List
不,不。当代码触手可及时,您真的不想添加HTTP调用和(反)序列化的开销。它甚至在同一个集会中!
无论如何,你的ApiController反对(my preferred) convention。让它返回一个具体的类型:
public IEnumerable<QDocumentRecord> GetAllRecords()
{
listOfFiles = ...
return listOfFiles;
}
如果您不想要,并且您确定需要返回HttpResponseMessage
,那么仍然绝对没有need to bother with calling JsonConvert.SerializeObject()
yourself:
return Request.CreateResponse<List<QDocumentRecord>>(HttpStatusCode.OK, listOfFiles);
然后,您不再需要控制器中的业务逻辑,因此您将其提取到一个为您工作的类中:
public class FileListGetter
{
public IEnumerable<QDocumentRecord> GetAllRecords()
{
listOfFiles = ...
return listOfFiles;
}
}
无论哪种方式,您都可以直接从MVC控制器调用此类或ApiController:
public class HomeController : Controller
{
public ActionResult Index()
{
var listOfFiles = new DocumentsController().GetAllRecords();
// OR
var listOfFiles = new FileListGetter().GetAllRecords();
return View(listOfFiles);
}
}
但是,如果你真的,真的必须做一个HTTP请求,你可以使用HttpWebRequest
,WebClient
,HttpClient
或RestSharp
,因为所有这些教程都存在
答案 1 :(得分:16)
这里很晚,但想分享下面的代码。 如果我们将WebApi作为一个完全不同的项目放在同一个解决方案中,那么我们可以从MVC控制器中调用它,如下所示
public class ProductsController : Controller
{
// GET: Products
public async Task<ActionResult> Index()
{
string apiUrl = "http://localhost:58764/api/values";
using (HttpClient client=new HttpClient())
{
client.BaseAddress = new Uri(apiUrl);
client.DefaultRequestHeaders.Accept.Clear();
client.DefaultRequestHeaders.Accept.Add(new System.Net.Http.Headers.MediaTypeWithQualityHeaderValue("application/json"));
HttpResponseMessage response = await client.GetAsync(apiUrl);
if (response.IsSuccessStatusCode)
{
var data = await response.Content.ReadAsStringAsync();
var table = Newtonsoft.Json.JsonConvert.DeserializeObject<System.Data.DataTable>(data);
}
}
return View();
}
}
答案 2 :(得分:5)
为什么不简单地将ApiController调用中的代码 - DocumentsController移动到可以从HomeController和DocumentController调用的类。将其拉出到您从两个控制器调用的类中。这个问题在你的问题中:
//找到文件的所有代码都在这里并且工作正常...
从同一网站上的另一个控制器调用API控制器是没有意义的。
这将简化代码,当你再回到它时,你将有一个共同的类来查找文件并在那里做那个逻辑......
答案 3 :(得分:1)
嗯,你可以通过很多方式做到这一点......其中一个就是创建一个HttpRequest。我会建议你不要从你自己的MVC调用你自己的webapi(这个想法是多余的......)但是,here's a end to end tutorial。
答案 4 :(得分:0)
控制器:
public JsonResult GetProductsData()
{
using (var client = new HttpClient())
{
client.BaseAddress = new Uri("http://localhost:5136/api/");
//HTTP GET
var responseTask = client.GetAsync("product");
responseTask.Wait();
var result = responseTask.Result;
if (result.IsSuccessStatusCode)
{
var readTask = result.Content.ReadAsAsync<IList<product>>();
readTask.Wait();
var alldata = readTask.Result;
var rsproduct = from x in alldata
select new[]
{
Convert.ToString(x.pid),
Convert.ToString(x.pname),
Convert.ToString(x.pprice),
};
return Json(new
{
aaData = rsproduct
},
JsonRequestBehavior.AllowGet);
}
else //web api sent error response
{
//log response status here..
var pro = Enumerable.Empty<product>();
return Json(new
{
aaData = pro
},
JsonRequestBehavior.AllowGet);
}
}
}
public JsonResult InupProduct(string id,string pname, string pprice)
{
try
{
product obj = new product
{
pid = Convert.ToInt32(id),
pname = pname,
pprice = Convert.ToDecimal(pprice)
};
using (var client = new HttpClient())
{
client.BaseAddress = new Uri("http://localhost:5136/api/product");
if(id=="0")
{
//insert........
//HTTP POST
var postTask = client.PostAsJsonAsync<product>("product", obj);
postTask.Wait();
var result = postTask.Result;
if (result.IsSuccessStatusCode)
{
return Json(1, JsonRequestBehavior.AllowGet);
}
else
{
return Json(0, JsonRequestBehavior.AllowGet);
}
}
else
{
//update........
//HTTP POST
var postTask = client.PutAsJsonAsync<product>("product", obj);
postTask.Wait();
var result = postTask.Result;
if (result.IsSuccessStatusCode)
{
return Json(1, JsonRequestBehavior.AllowGet);
}
else
{
return Json(0, JsonRequestBehavior.AllowGet);
}
}
}
/*context.InUPProduct(Convert.ToInt32(id),pname,Convert.ToDecimal(pprice));
return Json(1, JsonRequestBehavior.AllowGet);*/
}
catch (Exception ex)
{
return Json(0, JsonRequestBehavior.AllowGet);
}
}
public JsonResult deleteRecord(int ID)
{
try
{
using (var client = new HttpClient())
{
client.BaseAddress = new Uri("http://localhost:5136/api/product");
//HTTP DELETE
var deleteTask = client.DeleteAsync("product/" + ID);
deleteTask.Wait();
var result = deleteTask.Result;
if (result.IsSuccessStatusCode)
{
return Json(1, JsonRequestBehavior.AllowGet);
}
else
{
return Json(0, JsonRequestBehavior.AllowGet);
}
}
/* var data = context.products.Where(x => x.pid == ID).FirstOrDefault();
context.products.Remove(data);
context.SaveChanges();
return Json(1, JsonRequestBehavior.AllowGet);*/
}
catch (Exception ex)
{
return Json(0, JsonRequestBehavior.AllowGet);
}
}