API JS / C# 接口标准 - Camel 与 Pascal 大小写
API standard for JS / C# interface - Camel vs. Pascal case
我们有一个系统,服务器是用C#写的,实现了REST接口,客户端是用JS写的,大部分数据都是JSON传递的。
在某种程度上,驼峰式和 PascalCase 世界之间存在冲突。
参数主要来自服务器内的名称,并且全部采用 PascalCase。
JS 前端使用驼峰命名法编写,并且希望服务器能够接受这样的参数,因为 JSON 来自 JS 世界。
什么是可接受的解决方案,为什么?
不知道是不是完整的答案,因为不清楚你到底想达到什么目的。
显然 C# 是 Pascal 大小写,JSON 是 Camel 大小写。
所以对于我们的 web ASP.Net Web API 应用程序,我们已经为 Json.NET 实现了 DataConverter:
public class DataConverter : JsonConverter
{
#region Overriding
public override bool CanRead
{
get { return true; }
}
public override bool CanWrite
{
get { return true; }
}
public override bool CanConvert(Type objectType)
{
return true;
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
return this.ReadValue(reader);
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
this.WriteValue(writer, value, serializer);
}
#endregion
#region Assistants
private object ReadValue(JsonReader reader)
{
while (reader.TokenType == JsonToken.Comment)
{
if (reader.Read() == false)
throw new Exception("Unexpected end.");
}
switch (reader.TokenType)
{
case JsonToken.StartObject:
return this.ReadObject(reader);
case JsonToken.StartArray:
return this.ReadList(reader);
default:
if (this.CheckPrimitive(reader.TokenType) == true)
return reader.Value;
throw new Exception(string.Format(CultureInfo.InvariantCulture, "Unexpected token when converting ExpandoObject: {0}", reader.TokenType));
}
}
private object ReadList(JsonReader reader)
{
List<object> collection = new List<object>();
while (reader.Read() == true)
{
switch (reader.TokenType)
{
case JsonToken.Comment:
break;
case JsonToken.EndArray:
return collection;
default:
object value = this.ReadValue(reader);
collection.Add(value);
break;
}
}
throw new Exception("Unexpected end.");
}
private object ReadObject(JsonReader reader)
{
IDictionary<string, object> expando = new ExpandoObject();
while (reader.Read() == true)
{
switch (reader.TokenType)
{
case JsonToken.PropertyName:
string property = reader.Value.ToString().ToCase(Casing.Pascal);
if (reader.Read() == false)
throw new Exception("Unexpected end.");
object value = this.ReadValue(reader);
expando[property] = value;
break;
case JsonToken.Comment:
break;
case JsonToken.EndObject:
return expando;
}
}
throw new Exception("Unexpected end.");
}
private void WriteValue(JsonWriter writer, object value, JsonSerializer serializer)
{
if (this.CheckPrimitive(value) == true)
{
writer.WriteValue(value);
return;
}
if (value is Guid)
{
this.WriteValue(writer, (Guid)value, serializer);
return;
}
if (value is MyType)
{
this.WriteValue(writer, (MyType)value, serializer);
return;
}
if (value is IDynamicMetaObjectProvider && value is IDictionary<string, object>)
{
this.WriteObject(writer, (IDictionary<string, object>)value, serializer);
return;
}
if (value is IEnumerable)
{
IEnumerable enumerable = value as IEnumerable;
this.WriteArray(writer, enumerable, serializer);
return;
}
this.WriteObject(writer, value, serializer);
}
private void WriteValue(JsonWriter writer, Guid guid, JsonSerializer serializer)
{
writer.WriteValue(guid.ToString());
}
private void WriteValue(JsonWriter writer, MyType myType, JsonSerializer serializer)
{
writer.WriteValue(myType.ToString());
}
private void WriteArray(JsonWriter writer, IEnumerable enumerable, JsonSerializer serializer)
{
writer.WriteStartArray();
foreach (object value in enumerable)
{
this.WriteValue(writer, value, serializer);
}
writer.WriteEndArray();
}
private void WriteObject(JsonWriter writer, object value, JsonSerializer serializer)
{
writer.WriteStartObject();
foreach (PropertyInfo properties in value.GetType().GetProperties())
{
ParameterInfo[] parameters = properties.GetGetMethod().GetParameters();
if (parameters.Length == 0)
{
writer.WritePropertyName(properties.Name.ToCase(Casing.Camel));
this.WriteValue(writer, properties.GetValue(value), serializer);
}
}
writer.WriteEndObject();
}
private void WriteObject(JsonWriter writer, IDictionary<string, object> value, JsonSerializer serializer)
{
writer.WriteStartObject();
foreach (KeyValuePair<string, object> properties in value)
{
writer.WritePropertyName(properties.Key.ToCase(Casing.Camel)); // Implement own casing...
this.WriteValue(writer, properties.Value, serializer);
}
writer.WriteEndObject();
}
private bool CheckPrimitive(JsonToken token)
{
switch (token)
{
case JsonToken.Integer:
case JsonToken.Float:
case JsonToken.String:
case JsonToken.Boolean:
case JsonToken.Null:
case JsonToken.Undefined:
case JsonToken.Date:
case JsonToken.Bytes:
return true;
}
return false;
}
private bool CheckPrimitive(object value)
{
if (value == null)
return true;
if (value is bool)
return true;
if (value is byte)
return true;
if (value is sbyte)
return true;
if (value is short)
return true;
if (value is ushort)
return true;
if (value is int)
return true;
if (value is uint)
return true;
if (value is long)
return true;
if (value is ulong)
return true;
if (value is float)
return true;
if (value is double)
return true;
if (value is decimal)
return true;
if (value is char)
return true;
if (value is string)
return true;
if (value is DateTime)
return true;
if (value is Enum)
return true;
return false;
}
#endregion
}
它在JSON转换为C#对象时将"someProperty"转换为"SomeProperty",相反方向"SomeProperty"转换为"someProperty"。它允许我们在 API 控制器中使用动态关键字(在 ASP.Net 核心项目的情况下,输入参数需要属性 [FromBody])。
对于 Web API 方法的参数使用 "camelCase",我认为 .NET
中方法参数的建议命名约定
对于属性使用 JSON serialization attributes
public class Person
{
[JsonProperty("id")]
public int Id { get; set; }
[JsonProperty("name")]
public string Name { get; set; }
}
序列化对象如下所示:
{ "id": 10, "name": "My Name" }
通过这种方法,您的 C# 代码将保留 PascalCase,客户端 (Javascript) 将使用驼峰命名法。
我们有一个系统,服务器是用C#写的,实现了REST接口,客户端是用JS写的,大部分数据都是JSON传递的。
在某种程度上,驼峰式和 PascalCase 世界之间存在冲突。
参数主要来自服务器内的名称,并且全部采用 PascalCase。
JS 前端使用驼峰命名法编写,并且希望服务器能够接受这样的参数,因为 JSON 来自 JS 世界。
什么是可接受的解决方案,为什么?
不知道是不是完整的答案,因为不清楚你到底想达到什么目的。
显然 C# 是 Pascal 大小写,JSON 是 Camel 大小写。
所以对于我们的 web ASP.Net Web API 应用程序,我们已经为 Json.NET 实现了 DataConverter:
public class DataConverter : JsonConverter
{
#region Overriding
public override bool CanRead
{
get { return true; }
}
public override bool CanWrite
{
get { return true; }
}
public override bool CanConvert(Type objectType)
{
return true;
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
return this.ReadValue(reader);
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
this.WriteValue(writer, value, serializer);
}
#endregion
#region Assistants
private object ReadValue(JsonReader reader)
{
while (reader.TokenType == JsonToken.Comment)
{
if (reader.Read() == false)
throw new Exception("Unexpected end.");
}
switch (reader.TokenType)
{
case JsonToken.StartObject:
return this.ReadObject(reader);
case JsonToken.StartArray:
return this.ReadList(reader);
default:
if (this.CheckPrimitive(reader.TokenType) == true)
return reader.Value;
throw new Exception(string.Format(CultureInfo.InvariantCulture, "Unexpected token when converting ExpandoObject: {0}", reader.TokenType));
}
}
private object ReadList(JsonReader reader)
{
List<object> collection = new List<object>();
while (reader.Read() == true)
{
switch (reader.TokenType)
{
case JsonToken.Comment:
break;
case JsonToken.EndArray:
return collection;
default:
object value = this.ReadValue(reader);
collection.Add(value);
break;
}
}
throw new Exception("Unexpected end.");
}
private object ReadObject(JsonReader reader)
{
IDictionary<string, object> expando = new ExpandoObject();
while (reader.Read() == true)
{
switch (reader.TokenType)
{
case JsonToken.PropertyName:
string property = reader.Value.ToString().ToCase(Casing.Pascal);
if (reader.Read() == false)
throw new Exception("Unexpected end.");
object value = this.ReadValue(reader);
expando[property] = value;
break;
case JsonToken.Comment:
break;
case JsonToken.EndObject:
return expando;
}
}
throw new Exception("Unexpected end.");
}
private void WriteValue(JsonWriter writer, object value, JsonSerializer serializer)
{
if (this.CheckPrimitive(value) == true)
{
writer.WriteValue(value);
return;
}
if (value is Guid)
{
this.WriteValue(writer, (Guid)value, serializer);
return;
}
if (value is MyType)
{
this.WriteValue(writer, (MyType)value, serializer);
return;
}
if (value is IDynamicMetaObjectProvider && value is IDictionary<string, object>)
{
this.WriteObject(writer, (IDictionary<string, object>)value, serializer);
return;
}
if (value is IEnumerable)
{
IEnumerable enumerable = value as IEnumerable;
this.WriteArray(writer, enumerable, serializer);
return;
}
this.WriteObject(writer, value, serializer);
}
private void WriteValue(JsonWriter writer, Guid guid, JsonSerializer serializer)
{
writer.WriteValue(guid.ToString());
}
private void WriteValue(JsonWriter writer, MyType myType, JsonSerializer serializer)
{
writer.WriteValue(myType.ToString());
}
private void WriteArray(JsonWriter writer, IEnumerable enumerable, JsonSerializer serializer)
{
writer.WriteStartArray();
foreach (object value in enumerable)
{
this.WriteValue(writer, value, serializer);
}
writer.WriteEndArray();
}
private void WriteObject(JsonWriter writer, object value, JsonSerializer serializer)
{
writer.WriteStartObject();
foreach (PropertyInfo properties in value.GetType().GetProperties())
{
ParameterInfo[] parameters = properties.GetGetMethod().GetParameters();
if (parameters.Length == 0)
{
writer.WritePropertyName(properties.Name.ToCase(Casing.Camel));
this.WriteValue(writer, properties.GetValue(value), serializer);
}
}
writer.WriteEndObject();
}
private void WriteObject(JsonWriter writer, IDictionary<string, object> value, JsonSerializer serializer)
{
writer.WriteStartObject();
foreach (KeyValuePair<string, object> properties in value)
{
writer.WritePropertyName(properties.Key.ToCase(Casing.Camel)); // Implement own casing...
this.WriteValue(writer, properties.Value, serializer);
}
writer.WriteEndObject();
}
private bool CheckPrimitive(JsonToken token)
{
switch (token)
{
case JsonToken.Integer:
case JsonToken.Float:
case JsonToken.String:
case JsonToken.Boolean:
case JsonToken.Null:
case JsonToken.Undefined:
case JsonToken.Date:
case JsonToken.Bytes:
return true;
}
return false;
}
private bool CheckPrimitive(object value)
{
if (value == null)
return true;
if (value is bool)
return true;
if (value is byte)
return true;
if (value is sbyte)
return true;
if (value is short)
return true;
if (value is ushort)
return true;
if (value is int)
return true;
if (value is uint)
return true;
if (value is long)
return true;
if (value is ulong)
return true;
if (value is float)
return true;
if (value is double)
return true;
if (value is decimal)
return true;
if (value is char)
return true;
if (value is string)
return true;
if (value is DateTime)
return true;
if (value is Enum)
return true;
return false;
}
#endregion
}
它在JSON转换为C#对象时将"someProperty"转换为"SomeProperty",相反方向"SomeProperty"转换为"someProperty"。它允许我们在 API 控制器中使用动态关键字(在 ASP.Net 核心项目的情况下,输入参数需要属性 [FromBody])。
对于 Web API 方法的参数使用 "camelCase",我认为 .NET
中方法参数的建议命名约定对于属性使用 JSON serialization attributes
public class Person
{
[JsonProperty("id")]
public int Id { get; set; }
[JsonProperty("name")]
public string Name { get; set; }
}
序列化对象如下所示:
{ "id": 10, "name": "My Name" }
通过这种方法,您的 C# 代码将保留 PascalCase,客户端 (Javascript) 将使用驼峰命名法。