.NET 4.0 Web API 返回 JSON 而不转义字符

gar*_*968 2 asp.net json json.net asp.net-web-api2

我已经阅读了无数的问题和答案,但还没有看到任何解决问题的方法。

我正在使用 newtonsoft JSON 库,只是想从表中返回一些 JSON。

所以在我的控制器中,我只有(服务器和数据库详细信息为隐私而空白):

    public IHttpActionResult GetHelloWorld()
    {
        string JSONResult;

        SqlConnection conn = new SqlConnection("Server=<servername>;Database=<dbname>;user=<username>;password=<password>");
        DataTable table = new DataTable();

        conn.Open();

        SqlCommand cmd = new SqlCommand("select name, latitude, longitude from [<schema>].[<tablename>]", conn);

        using (SqlDataAdapter adapter = new SqlDataAdapter(cmd))
        {
            adapter.Fill(table);
        }

        //System.Web.Script.Serialization.JavaScriptSerializer serializer = new System.Web.Script.Serialization.JavaScriptSerializer();

        //List<Dictionary<string, object>> rows = new List<Dictionary<string, object>>();

        //Dictionary<string, object> row;

        //foreach (DataRow dr in table.Rows)
        //{
        //    row = new Dictionary<string, object>();
        //    foreach (DataColumn col in table.Columns)
        //    {
        //        row.Add(col.ColumnName, dr[col]);
        //    }
        //    rows.Add(row);
        //}
        //return Ok(serializer.Serialize(rows),);
        JSONResult = JsonConvert.SerializeObject(table);
        return Ok(JSONResult);
    }
Run Code Online (Sandbox Code Playgroud)

现在它按预期返回我的结果集,但我无法在没有转义双引号的情况下返回它,因此数据如下所示:

"[{\"name\":\"Aaron's Hill\",\"latitude\":51.18276,\"longitude\":-0.63503},{\"name\":\"Abbas Combe\",\"latitude\":51.00113,\"longitude\":-2.42178},
Run Code Online (Sandbox Code Playgroud)

现在,我读过的许多答案都只是简单地说,因为您正在 IDE 中查看结果。在我的情况下,这不是真的,这是浏览器窗口中的输出(尝试了 IE 和 chrome),甚至当你说给我看 JSON 输出时,提琴手也会抱怨它说位置 3 处的字符无效。

你会看到我已经注释掉了代码,这是我在这里找到的另一种返回 JSON 的方法,但它仍然转义了引号。不知道还有什么可以尝试的?

Yel*_*yev 6

你不应该自己序列化对象!
您应该返回一个将由 ASP.NET 序列化的类对象。现在,ASP.NET 序列化一个序列化对象——这就是为什么你会得到转义引号。

它应该是这样的:

public IHttpActionResult GetHelloWorld()
{
    SqlConnection conn = new SqlConnection("Server=<servername>;Database=<dbname>;user=<username>;password=<password>");
    DataTable table = new DataTable();

    conn.Open();

    SqlCommand cmd = new SqlCommand("select name, latitude, longitude from [<schema>].[<tablename>]", conn);

    using (SqlDataAdapter adapter = new SqlDataAdapter(cmd))
    {
        adapter.Fill(table);
    }

    return Ok(table); // DO NOT serialize
}
Run Code Online (Sandbox Code Playgroud)

然而,返回一个DataTable对象听起来是错误的——它不会被正确地序列化。
您需要像这样返回代表您的实体的自定义对象(仅作为示例,可能不起作用):

public IHttpActionResult GetHelloWorld()
{
    List<YourCustomClass> result = new List<YourCustomClass>();

    SqlConnection conn = new SqlConnection("Server=<servername>;Database=<dbname>;user=<username>;password=<password>");
    conn.Open();

    SqlCommand cmd = new SqlCommand("select name, latitude, longitude from [<schema>].[<tablename>]", conn);
    var reader = cmd.ExecuteReader();

    while (reader.Read())
    {
        result.Add(new YourCustomClass {
            Name = (string)reader["name"],
            Latitude = (decimal)reader["latitude"],
            Longitude = (decimal)reader["longitude"]
        });
    }

    return Ok(result);
}
Run Code Online (Sandbox Code Playgroud)

展望未来,我可能建议不要在 ASP.NET 类中进行 SQL 查询。SQL 连接和查询属于数据访问层 (DAL) - 它应该存储在单独的逻辑层中 - 在一个单独的项目中,或者至少在一个单独的类中。您可以使用 Repository 模式(它是一种模式吗?):

// Somewhere in a class library
public class Address
{
    public string Name { get; set; }
    public decimal Latitude { get; set; }
    public decimal Longitude { get; set; }
}

public class AddressesRepository : IAddressesRepository
{
    public Address[] ReadAll()
    {
        List<YourCustomClass> result = new List<YourCustomClass>();

        var connection = SqlConnectionFactory.CreateAndOpenConnection();

        var reader = new SqlCommand("select name, latitude, longitude from [<schema>].[<tablename>]", connection).ExecuteReader();

        while (reader.Read())
        {
            result.Add(new Address {
                Name = (string)reader["name"],
                Latitude = (decimal)reader["latitude"],
                Longitude = (decimal)reader["longitude"]
            });
        }

        return result;
    }
}

// Somewhere in ASP.NET WebAPI
private IAddressesRepository _addressesRepository = new AddressesRepository();

public IHttpActionResult GetHelloWorld()
{
    var data = _addressesRepository.ReadAll();
    return Ok(data);
}
Run Code Online (Sandbox Code Playgroud)

我建议阅读一些关于分层架构和 ASP.NET WebAPI 本身的书籍,以获得所有基本 ASP.NET 问题的答案。