1

我有一个多对多的关系设置,需要控制顺序。

我有一个具有以下属性的实体。通过这种关系检索时,如何控制组件的顺序?

property name="Components" fieldtype="many-to-many" cfc="CmsComponent" 
      type="array" 
      singularname="Component" 
      linktable="CMSPageComponents" 
      fkcolumn="page_id" 
      inversejoincolumn="component_id";

我在链接表(CMSPageComponents)中设置了一个名为dispOrder. 但是当我在上面的属性上设置orderby="dispOrder"ororderby="CMSPageComponents.dispOrder"属性时,它似乎忽略了它。

关于如何控制多对多关系顺序的任何建议?

4

1 回答 1

3

当关系是多对多时,CF 会将orderBy语句应用于对象表而不是链接表。因此,理论上您可以将dispOrder列从CMSPageComponents链接表移动到CmsComponent表以​​使其工作。

但在实践中,我希望排序特定于多对多关系(即特定页面),在这种情况下,您可以遵循 Peter 的建议并创建一个单独的实体来链接其他两个实体并让您定义一个 order 属性.

所以你会有3个实体:

  1. CMS页面
  2. CMS组件
  3. CmsPage 组件

CmsPageComponent可能看起来像这样:

<cfcomponent displayname="CmsPageComponent" persistent="true" table="cmsPageComponents">
    <!--- Add a primary key for the link Entity --->
    <cfproperty name="ID" fieldType="id" generator="native">
    <cfproperty name="dispOrder">
    <cfproperty name="page" fieldType="many-to-one" cfc="CmsPage" fkColumn="pageID">
    <cfproperty name="component" fieldType="many-to-one" cfc="CmsComponent" fkColumn="componentID">
    <!--- init() etc --->
</cfcomponent>

CmsPage然后可以与链接实体建立一对多的关系,允许使用 dispOrder 列进行排序:

<cfcomponent displayname="CmsPage" persistent="true" table="cmsPages">
    <cfproperty name="ID" fieldType="id" generator="native">
    <cfproperty name="pageComponents" singularName="pageComponent" fieldType="one-to-many" cfc="PageComponent" fkColumn="pageID" orderBy="dispOrder">
    <!--- init() etc --->
</cfcomponent>

更新 下面显示了如何添加和显示页面组件。不是唯一或必然是最好的方法,只是为了给你一个想法:

<cfscript>
transaction{
    //load the page
    page    =   EntityLoadByPK( "CmsPage",1 );
    //load the components we want to add
    component1  =   EntityLoadByPK( "CmsComponent",1 );
    component2  =   EntityLoadByPK( "CmsComponent",2 );
    //create link objects
    pageComponent1  =   EntityNew( "CmsPageComponent" );
    pageComponent2  =   EntityNew( "CmsPageComponent" );
    // link them to the pages and components in the order we want
    pageComponent1.setComponent( component1 );
    pageComponent1.setPage( page );
    pageComponent1.setDispOrder( 2 );
    EntitySave( pageComponent1 );
    pageComponent2.setComponent( component2 );
    pageComponent2.setPage( page );
    pageComponent2.setDispOrder( 1 );
    EntitySave( pageComponent2 );
}
//Reload from the database so the order is applied
EntityReload( page );
</cfscript>
<!DOCTYPE html>
<html>
<head>
    <title>Test</title>
</head>
<body>
    <cfoutput>
        <h2>Page #page.getID()#</h2>
        <ol>
            <cfloop array="#page.getPageComponents()#" index="pageComponent">
                <cfset component    =   pageComponent.getComponent()>
                <li>Component ID #component.getID()#, Display Order = #pageComponent.getDispOrder()#)</li>
            </cfloop>
        </ol>
    </cfoutput>
</body>
</html>

注意:这假设flushAtRequestEndApplication.cfc 中的 ORM 设置为真

于 2013-11-01T12:49:30.573 回答