在我的代码中,我有一个页面,其中包含来自3个不同表格的信息。为了显示这些信息,我做了3个SQL select调用,并将它们统一在一个列表中,作为Model传递给我的视图。我可以用一个SQL调用吗?数据彼此之间没有联系。
我的代码:
public ActionResult Index()
{
StorePageData PageData = new StorePageData();
return View(PageData);
}
public class StorePageData
{
public List<Table1Data> Table1 { get; set; }
public List<Table2Data> Table2 { get; set; }
public List<Table3Data> Table3 { get; set; }
public StorePageData()
{
Table1 = //loading from Database1
Table2 = //loading from Database2
Table3 = //loading from Database3
}
}
public class Table1Data
{
public int Id { get; set; }
public double Info1 { get; set; }
public string Info2 { get; set; }
}
public class Table2Data
{
public int Id { get; set; }
public List<int> Info1 { get; set; }
public List<int> Info2 { get; set; }
}
public class Table3Data
{
public int Id { get; set; }
public List<string> Info1 { get; set; }
public List<string> Info2 { get; set; }
}
如果有办法在一个SQL请求中加载所有3个表,它将显着改善此页面的加载时间。
谢谢。
答案 0 :(得分:12)
您可以使用DataReader
在单个请求中获取多个结果集。您可以在有或没有实体框架的情况下使用它。
如果您使用的是Entity Framework,则可以将DbDataReader
传递给ObjectContext.Translate
方法,以将多个结果集转换为请求的对象类型。用于创建数据读取器的命令可以是存储过程,也可以只使用包含查询的命令来形成多个结果集。
实施例
List<Table1> list1;
List<Table2> list2;
using (var cn = new SqlConnection(@"Connection String"))
{
cn.Open();
using (var cmd = cn.CreateCommand())
{
cmd.CommandText = "SELECT * FROM Table1; SELECT * FROM Table2";
var reader = cmd.ExecuteReader();
using (var db = new YourDbContext())
{
var context = ((IObjectContextAdapter)db).ObjectContext;
list1 = context.Translate<Table1>(reader).ToList();
reader.NextResult();
list2 = context.Translate<Table2>(reader).ToList();
}
}
}
如果您使用SqlDataAdapter
,则只需传递包含查询的命令,然后使用Fill
填写数据集即可。数据适配器本身将在场景后面使用DataReader
。
实施例
var connectionString = @"Connection String";
var commandText = "SELECT * FROM Table1; SELECT * FROM Table2;";
var ds = new DataSet();
using (var da = new SqlDataAdapter(commandText, connectionString))
{
da.Fill(ds);
}
然后,您可以将结果整理为List<Table1>
和List<Table2>
。
答案 1 :(得分:2)
您可以使用presence_diff
合并多个查询。
做这样的事情:
UNION ALL
修改强>
如果您想知道单条记录的来源,可以执行此操作:
SELECT * FROM Table1
UNION ALL
SELECT * FROM Table2
这将添加另一列,可用于将数组拆分为3个列表。
答案 2 :(得分:2)
答案 3 :(得分:2)
不要使用UNION。 DataAdapter是选择的武器。
var commandText = "SELECT * FROM Table1; SELECT * FROM Table2;";
var ds = new DataSet();
using (var da = new SqlDataAdapter(commandText, "your cn"))
{
da.Fill(ds);
}
使用:
ds.Tables["Table1"]...
ds.Tables["Table2"]...
答案 4 :(得分:1)
你可以做这样的事情
SELECT Info1, Info2
FROM Table1Data
UNION ALL
SELECT Id, Info2
FROM Table2Data;
然后你可以解析结果。 取决于案例,但您也可以考虑使用交易。
答案 5 :(得分:1)
可能这对你有帮助。
在单个过程中选择三个表作为单独的结果集,然后在C#端获取结果集并将其转换为JSON格式,然后从JSON中将每个表数据放入列表中。
DataSet dataSet = new DataSet("dataSet");
dataSet.Namespace = "NetFrameWork";
DataTable table = new DataTable();
DataColumn idColumn = new DataColumn("id", typeof(int));
idColumn.AutoIncrement = true;
DataColumn itemColumn = new DataColumn("item");
table.Columns.Add(idColumn);
table.Columns.Add(itemColumn);
dataSet.Tables.Add(table);
for (int i = 0; i < 2; i++)
{
DataRow newRow = table.NewRow();
newRow["item"] = "item " + i;
table.Rows.Add(newRow);
}
dataSet.AcceptChanges();
string json = JsonConvert.SerializeObject(dataSet, Formatting.Indented);
Console.WriteLine(json);
//{
// "Table1": [
// {
// "id": 0,
// "item": "item 0"
// },
// {
// "id": 1,
// "item": "item 1"
// }
// ]
//"Table2": [
// {
// "id": 0,
// "item": "item 0",
// "rate": 200.00
// },
// {
// "id": 1,
// "item": "item 1",
// "rate": 225.00
//}
// ]
// "Table3": [
// {
// "id": 0,
// "item": "item 0",
// "rate": 200.00,
// "UOM" : "KG"
// },
// {
// "id": 1,
// "item": "item 1",
// "rate": 225.00,
// "UOM" : "LTR"
// }
// ]
//}
答案 6 :(得分:1)
您可以使用以下通用代码段
/// <method>
/// Select Query
/// </method>
public DataTable executeSelectQuery(String _query, SqlParameter[] sqlParameter)
{
SqlCommand myCommand = new SqlCommand();
DataTable dataTable = new DataTable();
dataTable = null;
DataSet ds = new DataSet();
try
{
myCommand.Connection = openConnection();
myCommand.CommandText = _query;//or proc
myCommand.Parameters.AddRange(sqlParameter);
myAdapter.SelectCommand = myCommand;
myAdapter.Fill(ds);
dataTable = ds.Tables[0];//as per your requirement
}
catch (SqlException e)
{//any logger
Console.Write("Error - Connection.executeSelectQuery - Query: " + _query + " \nException: " + e.StackTrace.ToString());
return null;
}
return dataTable;
}