如何在ColdFusion/Lucee组件中动态编写隐式getter和setter?

eda*_*dam 5 coldfusion cfc metaprogramming getter-setter lucee

我希望能够在CFML/LUCEE组件中动态编写一组getter和setter(没有硬编码的cfproperty标签).

<!--- MyComp.cfc --->
<cfcomponent displayname="MyComp" hint="MyComp" accessors="true">
    <cffunction name="init">
       <cfargument name="dynamicprops" type="array">
       <cfloop array="#dynamicprops#" index="item">
          <!--- 
           Now what? I cannot do a cfsavecontent and write props here.
           It demands cfproperty just after the cfcomponent begins. I 
           tried to do with closures but they are not acually setters 
           and getters. Does anyone know how to better do it? 
          ---> 
      </cfloop>
    </cffunction>
</cfcomponent>

<!--- example call --->
<cfset mc = CreateObject("component","MyComp").init( [{"name"="a","default"=1}] ) />
Run Code Online (Sandbox Code Playgroud)

然后我希望能够调用mc.setA(100)mc.getA().但是没有发生.

所以我的谦虚问题是如何在组件上动态编写setter和getter?

PS:请记住我已尝试过关闭方式:

 variables[item.name] = item.default;
 variables["set"&item.name] = function(_val){ variables[item.name] =_val; }
 variables["get"&item.name] = function(){ return variables[item.name; }
Run Code Online (Sandbox Code Playgroud)

无法工作.我该怎么做?谢谢 :)

CfS*_*ity 7

你可以用onMissingMethod()它.

component name="myComponent" hint="myComponent.cfc"{

    function init( struct dynamicProperties={} ){
        dynamicProperties.each( function( name, default ){
            variables[ name ] = default;
        } );
        return this;
    }

    function onMissingMethod( name, args ){
        if( name.Left( 3 ) IS "get" )
            return get( name );
        if( ( name.Left( 3 ) IS "set" ) AND ( args.Len() IS 1 ) )
            return set( name, args[ 1 ] );
        cfthrow( type="NonExistentMethod", message="The method '#name#' doesn't exist" );
    }

    public any function get( required string accessorName ){
        var propertyName = parsePropertyName( accessorName );
        if( !variables.KeyExists( propertyName ) )
            cfthrow( type="NonExistentProperty", message="The property '#propertyName#' doesn't exist" );
        return variables[ propertyName ];
    }

    public void function set( required string accessorName, required any value ){
        var propertyName = parsePropertyName( accessorName );
        if( !variables.KeyExists( propertyName ) )
            cfthrow( type="NonExistentProperty", message="The property '#propertyName#' doesn't exist" );
        variables[ propertyName ] = value;
    }

    private string function parsePropertyName( accessorName ){
        return accessorName.RemoveChars( 1, 3 );
    }

}
Run Code Online (Sandbox Code Playgroud)

传递它的属性名称/默认值的结构,它将"监听"匹配的getter/setter.任何不会导致异常.

<cfscript>
myDynamicProperties = { A: 0, B: 0 }; // this is a struct of names and default values
mc = new myComponent( myDynamicProperties );
mc.setA( 100 );
WriteDump( mc.getA() ); // 100
WriteDump( mc.getB() ); // 0
WriteDump( mc.getC() ); // exception
</cfscript>
Run Code Online (Sandbox Code Playgroud)

更新1:将属性名称数组替换为名称/默认值struct作为init参数,以允许设置默认值.

更新2:如果要传递包含名称/默认值对的结构数组,例如

dynamicProperties = [ { name: "A", default: 1 }, { name: "B", default: 2 } ];

那么init()方法将是:

function init( array dynamicProperties=[] ){
    dynamicProperties.each( function( item ){
        variables[ item.name ] = item.default;
    } );
    return this;
}
Run Code Online (Sandbox Code Playgroud)

更新3:如果您必须使用标签并<cfloop>设置动态属性,那么这就是您在init方法中所需要的:

<cfloop array="#dynamicProperties#" item="item">
  <cfset variables[ item.name ] = item.default>
</cfloop>
Run Code Online (Sandbox Code Playgroud)

onMissingMethod 如果您尝试将动态方法作为以下属性调用,则不会触发:

method = mc[ "set#property#" ];
method( value );
Run Code Online (Sandbox Code Playgroud)

相反,只需将组件中的set()get()方法设为公共并直接调用它们:

mc.set( property, value );
mc.get( property );
Run Code Online (Sandbox Code Playgroud)


Jam*_*ler 3

考虑使用访问器

<cfcomponent displayname="MyComp" hint="MyComp" accessors="true">
Run Code Online (Sandbox Code Playgroud)

来源:https ://helpx.adobe.com/coldfusion/cfml-reference/coldfusion-tags/tags-c/cfcomponent.html