从远程CFC函数返回的JSON出错

时间:2011-09-27 16:22:09

标签: jquery json coldfusion cfc

我有一个返回结构的远程CFC。它使用cfajaxproxy调用。我希望返回的JSON按顺序排列,即首先进入JSON对象的结构。但是,返回的JSON是混合顺序。

这是远程功能。

<cfcomponent displayname="validation" hint="">
    <cffunction name="validateForm" displayname="validateForm" hint="" access="remote" verifyClient="yes" returntype="struct">

        <cfargument name="formVals" type="struct" required="yes">

        <cfset errors = StructNew()>

        <cfif formVals.project neq "project">
              <cfset errors["project"] = "Invalid project name." />
        </cfif>

        <cfif Len(formVals.description) eq 0>
             <cfset errors["description"] = "Please enter a description." />
        </cfif>

        <cfif StructIsEmpty(errors)>
            <cfset errors["message"]["type"] = "success">
            <cfset errors["message"]["text"] = "Client and server-side validation passed successfully.">
            <cfset errors["areErrors"] = false>
        <cfelse>
            <cfset errors["message"]["type"] = "validation">
            <cfset errors["message"]["text"] = "Please fix the errors, and resubmit.">
            <cfset errors["areErrors"] = true>
        </cfif>

        <cfreturn errors />

    </cffunction>
</cfcomponent>

这是我在表单页面顶部设置的cfajaxproxy。

<cfajaxproxy cfc="validation" jsclassname="validation">

这是在我的表单的onSubmit处理程序中对远程函数的调用。

var v = new validation();
v.setHTTPMethod("POST");
var errors = v.validateForm(o);

这是发布请求中发送给函数的数据(上面的o变量)。

{"formVals":{"project":"","description":""}}

这是从函数返回的JSON响应。

{"message":{"text":"Please fix the errors, and resubmit.","type":"validation"},"description":"Please enter a description.","project":"Invalid project name.","areErrors":true}

我希望响应的顺序与创建的结构的顺序相同。

{"project":"Invalid project name.","description":"Please enter a description.","message":{"text":"Please fix the errors, and resubmit.","type":"validation"},"areErrors":true}

这样当我迭代响应时,我可以将焦点设置为第一个表单字段,其中包含错误。

var focusSet = false;

$.each(errors, function(key, val){
    //alert(key + ': ' + val);
    if(key != 'message' && key != 'areErrors') {
        var fi = $('#' + key).parents('.formItem').filter(':first');
        fi.addClass("inError");
        fi.find('.err').filter(':first').html(val);
        if(!focusSet) {
            $('#' + key).focus();
            focusSet = true;
        }
    }
});

现在,这会将焦点放在表单的第二个字段中,而不是项目字段中。

4 个答案:

答案 0 :(得分:7)

ColdFusion结构的键永远不会以任何特定顺序存储。但是,我found one post显示了如何创建一个java LinkedHashMap(它是CF Struct下面的java)来按特定顺序存储和检索键。

<cfset pets = CreateObject("java", "java.util.LinkedHashMap").init()>
<cfscript>
pets["Cat Name"] = "Leo";
pets["Dog Name"] = "Meatball";
pets["Fish Name"] = "Lil Fish";
pets["Bird Name"] = "PePe";
pets["Snake Name"] = "Sizzle";
</cfscript>
<cfloop collection="#pets#" item="key" >
    <cfoutput>
    #key#: #pets[key]#<br/>
    </cfoutput>
</cfloop>

编辑: Dan的解决方案(数组而不是struct)可能会更容易。

答案 1 :(得分:5)

除非手动构建字符串以返回该数据,否则不能(轻松)控制返回的JSON结构数据的顺序。如果您必须依赖订单,那么您需要在数组而不是结构中返回错误。您甚至可以返回一组错误结构,CF将保持正确的数组顺序。

我会像这样返回你的数据:

<cfcomponent displayname="validation" hint="">
    <cffunction name="validateForm" displayname="validateForm" hint="" access="remote" verifyClient="yes" returntype="struct">

        <cfargument name="formVals" type="struct" required="yes">

        <cfset var retVal = StructNew() />
        <cfset var tempError = StructNew() />
        <cfset retVal.errors = ArrayNew(1) />

        <cfif formVals.project neq "project">
            <cfset tempError["key"] = "project" />
            <cfset tempError["message"] = "Invalid project name." />
            <cfset ArrayAppend(retVal.errors, Duplicate(tempError)) />
        </cfif>

        <cfif Len(formVals.description) eq 0>
            <cfset tempError["key"] = "description" />
            <cfset tempError["message"] = "Please enter a description." />
            <cfset ArrayAppend(retVal.errors, Duplicate(tempError)) />
        </cfif>

        <cfif ArrayIsEmpty(retVal.Errors)>
            <cfset retVal["message"]["type"] = "success" />
            <cfset retVal["message"]["text"] = "Client and server-side validation passed successfully.">
            <cfset retVal["areErrors"] = false>
        <cfelse>
            <cfset retVal["message"]["type"] = "validation">
            <cfset retVal["message"]["text"] = "Please fix the errors, and resubmit.">
            <cfset retVal["areErrors"] = true>
        </cfif>

        <cfreturn retVal />

    </cffunction>
</cfcomponent>

这会为您提供一个单独的错误数组,而不是在您的错误的同时处理您的基本messageareErrors键。将它们分解为一个单独的实体,您将在客户端更容易地循环它们。

答案 2 :(得分:3)

struct不是用CFML排序的(它只是一个类似Hashmap的集合)。

如果您想要有序的结构,请使用

struct function orderedStructNew()
{
    return createObject("java","java.util.LinkedHashMap").init();
}

答案 3 :(得分:2)

如果你想要保留一些东西以便使用数组而不是结构,最简单的方法。