How to Invoke CFC Methods with CFML
In this tutorial, we'll learn the basics of creating and working with CFCs in CFML, their importance, and code examples so you can get started with your own custom functions quickly.
What Are CFCs?
CFCs (ColdFusion Components) are a great demonstration of the true power behind CFML and a way to separate your application logic from your interface. They're especially useful for creating reusable bits of code that you can utilize and invoke throughout your application.
CFCs use the .cfc
file extension to help differentiate from a regular .cfm
script file.
Tags for Creating CFCs
We'll explore three different tags associated while creating our CFC:
cfcomponent
- Defines the component and includes specific attributes for introspection.cffunction
- Defines a component method, or function, that you can invoke from other CFCs or CFML files.cfargument
- Defines a custom argument, or parameter, for each function.
CFC Code Example
Let's create our first CFC and walk through its structure and each of its features. Create a file called Data.cfc
and copy/paste the following code:
<cfcomponent displayName="myComp" output="false">
<cffunction name="myFunc" returnType="query" output="false">
<cfargument name="id" type="numeric" required="true">
<cfset var qryData = "">
<cfquery name="qryData" datasource="myDatasource">
SELECT *
FROM myTable
WHERE id = <cfqueryparam value="#arguments.id#" cfsqltype="cf_sql_bigint">;
</cfquery>
<cfreturn qryData>
</cffunction>
</cfcomponent>
Every CFC must start with an opening and closing cfcomponent
tag. This defines the CFC and creates an association with its included methods. The displayName
attribute is used for introspection. The output
attribute is an option boolean parameter. In this case, a value of "false" prevents unnecessary page output while processing the function.
Within each component, you can create as many functions as you'd like using the cffunction
tag. In this example, we've created a single function called "myFunc" where we'll query a hypothetical database table by returning all columns and their associated values that match a given "id". The returnType
specifies the resulting data type expected to be returned from the function call, in this case, our query results. And the optional output
parameter with a value of "false" instructions the function to refrain from returning text, including whitespace, in the result.
Within the function, you can create as many arguments as you'd like using the cfargument
tag. In this case, a numeric ID variable is required to continue processing the remainder of the function call. If the ID variable is not passed in, an error will be thrown.
Next, we have a local variable qryData
prefixed with the var
keyword, indicating that this variable can only be used inside the function where it's declared. This is important and best practice to prevent accidental variable rewriting within other CFC methods.
Finally, we use the cfquery
tag to execute our SQL call and return the resultset from the query.
Invoking CFC Methods
To invoke our CFC methods, retrieve the return data, and output the resultset to the screen, we'll explore a few different options.
The cfinvoke
tag
The first method is with the cfinvoke
tag. Here, we'll invoke a function call to our newly created component and method, and assign a return variable we can use to dump the data:
<cfinvoke component="myComp" method="myFunc" returnVariable="qryData">
<cfdump var="#qryData#">
The cfobject
tag
The second method is with the cfobject
tag. Similar to the first, we invoke our newly created component method, but instead, we create an instance of the component that we can reuse without redefining it each time. This can significantly improve processing time where you're making multiple component calls in a single page request:
<cfobject name="objComp" component="myComp">
<cfset qryData = objComp.myFunc(id = 1)>
<cfdump var="#qryData#">
<cfset qryData = objComp.myFunc(id = 2)>
<cfdump var="#qryData#">
The CreateObject()
function
The third method is very similar to the previous method, except we use the CreateObject()
function to create an instance of our component and associated methods instead of a tag.
<cfset objComp = CreateObject("component", "myComp")>
<cfset qryData = objComp.myFunc(id = 1)>
<cfdump var="#qryData#">
<cfset qryData = objComp.myFunc(id = 2)>
<cfdump var="#qryData#">
The CreateObject()
method is my preferred way of defining objects in CFML due to the flexibility it allows. Here, you're passing in two arguments to the function call. The first argument is the type of object we're creating an instance for, in this case, a component. The second argument is the name of the component, "myComp".
The one thing I'd like to note about passing in a component name is the location of the CFC file. Passing in "myComp" assumes that the component is in the same folder as the script executing the code. However, if the component file were located in a sub-directory, you would use dot notation to separate each directory name to its location:
<cfset objComp = CreateObject("component", "path.to.myComp")>
Conclusion
That's it! Components in real-world applications can be a lot more complex than this, but this basic structure should get you going in the right direction.
Written by: Josh Rowe
Created: July 16, 2023