我将从我的问题开始:
我有一个webmethod,我通过AJAX / JSON调用。 我们称之为“GlobalMethod”,它用于管理“容器”对象,该对象具有从相同“基类”派生的项列表。 这是我的情况的代码示例:
[WebService(Namespace = "http://tempuri.org/")]
[WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
[System.ComponentModel.ToolboxItem(false)]
[System.Web.Script.Services.ScriptService]
public class MyService : System.Web.Services.WebService
{
[WebMethod]
public string GlobalMethod(Container data)
{
string result = "";
foreach (var item in data.Items)
{
result += item.BaseProperty;
if (item is FirstDerivedClass)
result += ((FirstDerivedClass)item).FirstDerivedProperty;
else if (item is SecondDerivedClass)
result += ((SecondDerivedClass)item).SecondDerivedProperty;
}
return result;
}
}
public class Container
{
public List<BaseClass> Items;
}
public abstract class BaseClass
{
public string BaseProperty;
}
public class FirstDerivedClass : BaseClass
{
public string FirstDerivedProperty;
}
public class SecondDerivedClass : BaseClass
{
public string SecondDerivedProperty;
}
这种方法根本无法解决。我将无法使用默认的JavascriptSerializer调用此方法,因为序列化程序无法解析容器将具有的对象类型:它们可以是FirstDerivedClass或SecondDerivedClass类型。
所以,浏览网页寻找我的问题的解决方案,我遇到Json.NET,其方法JsonConvert.DeserializeObject能够检索使用JsonConvert.SerializeObject序列化的原始对象类型,因为它添加一个名为“$ type”的属性。
我现在的问题是:如何使用此序列化程序而不是ASMX使用的默认序列化程序来制作web方法? 如果方法签名仍然是
[WebMethod]
public string GlobalMethod(Container data)
然后我会得到一个完全空的 Container对象,因为框架正在进行反序列化工作,并且不知道要实例化哪些项目,我无法告诉框架它应该使用Json.NET来完成工作,这将能够完全反序列化数据。
如果我尝试以这种方式修改方法
[WebMethod]
public string GlobalMethod(string data)
{
string result = "";
Container container = JsonConvert.DeserializeObject<Container>(data,
new JsonSerializerSettings
{
TypeNameHandling = TypeNameHandling.Objects
});
foreach (var item in container.Items) {
...
然后我会得到另一个服务器错误500因为“没有为u0027system.string u0027类型定义无参数构造函数”
好的,希望我的问题很清楚......我一整天都花在这上面,我似乎找不到解决办法。重新审视我的方法的体系结构以避免使用派生类是不是一个选择(要注意实际的代码要复杂得多,我只是简化它以达到目的!)。
顺便说一下,我将添加调用webmethod的ajax方法以这种方式完成:
$.ajax({
type: "POST",
url: wsUrl + method,
data: JSON.stringify(data),
contentType: "application/json; charset=utf-8",
dataType: "json",
processData: false,
success: function(msg)
{
try{
success(JSON.parse(msg.d));
} finally {
}
},
error: error
});
感谢任何与我分享知识的人!
我想分享一下我是如何解决问题的。正如我已经说过的那样,我用这种方式修改了我的方法:
[WebMethod]
public string GlobalMethod(string data)
{
string result = "";
Container container = JsonConvert.DeserializeObject<Container>(data,
new JsonSerializerSettings
{
TypeNameHandling = TypeNameHandling.Objects
});
foreach (var item in container.Items) {
最初,我开始收到错误500 “没有为u0027system.string u0027类型定义的无参数构造函数”
但最终我发现了原因:从客户端到服务器的json字符串被.NET框架反序列化为包含的对象,该对象不是字符串,而是实际的JSON对象!
所以诀窍是以下,在我的ajax方法中:
$.ajax({
type: "POST",
url: wsUrl + method,
**data: JSON.stringify(JSON.stringify(data)),**
contentType: "application/json; charset=utf-8",
dataType: "json",
processData: false,
success: function(msg)
{
try{
success(JSON.parse(msg.d));
} finally {
}
},
error: error
});
这意味着json对象被封装在另一个字符串中,该字符串将由我的服务器端GlobalMethod(字符串)通过Json.NET手动反序列化。
显然我不会在ajax例程中包含这个双“stringify”,但我会注意将输入中的JSON.stringified数据传递给ajax例程本身!
答案 0 :(得分:3)
这只是一个黑客, 这只是一个想法,听起来并不优雅但应该有效 我只是好奇
以这种方式更改容器:
public class Container
{
public List<MyUnionBaseClass> Items;
}
并定义(伪代码):
public class MyUnionBaseClass{
public void MyUnionBaseClass(BaseClass b){
this.setValue(b);
};
final public BaseClass getValue(){
if(first!=null) return (BaseClass) first;
if(second!=null)return (BaseClass) second;
return null;
}
final public setValue(BaseClass b){
if(b instanceOf firstClass){
first = (FirstClass) b;
}
if(b instanceOf secondClass){
second = (SecondClass) b;
}
}
private FirstDerivedClass first=null;
private SecondDerivedClass second=null;
}
P.s:这是非常好的,应该加以改进。 这有意义吗?
答案 1 :(得分:2)
更改网络方法以接受object
[WebMethod]
public string GlobalMethod(object data)
这可以解决您遇到的问题。