复合外键学说Symfony的重叠非主键2

时间:2013-12-09 19:36:19

标签: php postgresql symfony doctrine-orm foreign-keys

我正在尝试使用Doctrine Symfony 2构建四个实体。

其中一个实体有“重叠或相互交叉的非主键复合外键” - 为我糟糕的英语道歉。

我试图修改doctrine对象,但我仍然无法将所有实体持久化到PostgreSQL。


在我构建的四个实体的学说对象下面:

    <?php

    namespace EntityBundle\Entity;

    use Doctrine\ORM\Mapping as ORM;

    /**
     * @ORM\Entity(repositoryClass="EntityBundle\EntityRepository\GeographyContinentRepository")
     * @ORM\Table(
     *     name="geography_continent",
     *     uniqueConstraints={
     *         @ORM\UniqueConstraint(name="geography_continent_u1", columns={"continent_name"})
     *     }
     * )
     */
    class GeographyContinent
    {       
        /**
         * @ORM\Column(name="id", type="integer", nullable=false)
         * @ORM\Id
         * @ORM\GeneratedValue(strategy="AUTO")
        */
        protected $id;            

        /**
         *
         * @ORM\Column(name="continent_name", type="string", nullable=false)
         */    
        protected $continentName;            

        /**
         *
         * @ORM\Column(name="description", type="string", nullable=true)
         */    
        protected $description;        

        /**
         * Get id
         *
         * @return integer 
         */
        public function getId()
        {
            return $this->id;
        }

        /**
         * Set continentName
         *
         * @param string $continentName
         * @return GeographyContinent
         */
        public function setContinentName($continentName)
        {
            $this->continentName = $continentName;

            return $this;
        }

        /**
         * Get continentName
         *
         * @return string 
         */
        public function getContinentName()
        {
            return $this->continentName;
        }

        /**
         * Set description
         *
         * @param string $description
         * @return GeographyContinent
         */
        public function setDescription($description)
        {
            $this->description = $description;

            return $this;
        }

        /**
         * Get description
         *
         * @return string 
         */
        public function getDescription()
        {
            return $this->description;
        }
    }

    ?>


    <?php

    namespace EntityBundle\Entity;

    use Doctrine\ORM\Mapping as ORM;

    /**
     * @ORM\Entity(repositoryClass="EntityBundle\EntityRepository\GeographyCountryRepository")
     * @ORM\Table(
     *     name="geography_country",
     *     uniqueConstraints={
     *         @ORM\UniqueConstraint(name="geography_country_u1", columns={"country_name"}),
     *         @ORM\UniqueConstraint(name="geography_country_u2", columns={"telephone_code"}),
     *         @ORM\UniqueConstraint(name="geography_country_u3", columns={"currency_name"}),
     *         @ORM\UniqueConstraint(name="geography_country_u4", columns={"currency_symbol"})
     *     }
     * )
     */
    class GeographyCountry 
    {
        /**
         * @ORM\Column(name="id", type="integer", nullable=false)
         * @ORM\Id
         * @ORM\GeneratedValue(strategy="AUTO")
        */
        protected $id;            

        /**
         *
         * @ORM\Column(name="country_name", type="string", nullable=false)
         */    
        protected $countryName;            

        /**
         *
         * @ORM\Column(name="telephone_code", type="string", nullable=true)
         */    
        protected $telephoneCode;            

        /**
         *
         * @ORM\Column(name="currency_name", type="string", nullable=true)
         */    
        protected $currencyName;            

        /**
         *
         * @ORM\Column(name="currency_symbol", type="string", nullable=true)
         */    
        protected $currencySymbol;                

        /**
         *
         * @ORM\Column(name="continent_id", type="integer", nullable=false)
         */    
        protected $continentId;            

        /**
         *
         * @ORM\ManyToOne(targetEntity="GeographyContinent", cascade={"persist", "remove"})
         * @ORM\JoinColumn(name="continent_id", referencedColumnName="id")
         */    
        protected $fkContinent;            

        /**
         *
         * @ORM\Column(name="description", type="string", nullable=true)
         */    
        protected $description;

        /**
         * Get id
         *
         * @return integer 
         */
        public function getId()
        {
            return $this->id;
        }

        /**
         * Set countryName
         *
         * @param string $countryName
         * @return GeographyCountry
         */
        public function setCountryName($countryName)
        {
            $this->countryName = $countryName;

            return $this;
        }

        /**
         * Get countryName
         *
         * @return string 
         */
        public function getCountryName()
        {
            return $this->countryName;
        }

        /**
         * Set telephoneCode
         *
         * @param string $telephoneCode
         * @return GeographyCountry
         */
        public function setTelephoneCode($telephoneCode)
        {
            $this->telephoneCode = $telephoneCode;

            return $this;
        }

        /**
         * Get telephoneCode
         *
         * @return string 
         */
        public function getTelephoneCode()
        {
            return $this->telephoneCode;
        }

        /**
         * Set currencyName
         *
         * @param string $currencyName
         * @return GeographyCountry
         */
        public function setCurrencyName($currencyName)
        {
            $this->currencyName = $currencyName;

            return $this;
        }

        /**
         * Get currencyName
         *
         * @return string 
         */
        public function getCurrencyName()
        {
            return $this->currencyName;
        }

        /**
         * Set currencySymbol
         *
         * @param string $currencySymbol
         * @return GeographyCountry
         */
        public function setCurrencySymbol($currencySymbol)
        {
            $this->currencySymbol = $currencySymbol;

            return $this;
        }

        /**
         * Get currencySymbol
         *
         * @return string 
         */
        public function getCurrencySymbol()
        {
            return $this->currencySymbol;
        }

        /**
         * Set continentId
         *
         * @param integer $continentId
         * @return GeographyCountry
         */
        public function setContinentId($continentId)
        {
            $this->continentId = $continentId;

            return $this;
        }

        /**
         * Get continentId
         *
         * @return integer 
         */
        public function getContinentId()
        {
            return $this->continentId;
        }

        /**
         * Set description
         *
         * @param string $description
         * @return GeographyCountry
         */
        public function setDescription($description)
        {
            $this->description = $description;

            return $this;
        }

        /**
         * Get description
         *
         * @return string 
         */
        public function getDescription()
        {
            return $this->description;
        }

        /**
         * Set fkContinent
         *
         * @param \EntityBundle\Entity\GeographyContinent $fkContinent
         * @return GeographyCountry
         */
        public function setFkContinent(\EntityBundle\Entity\GeographyContinent $fkContinent = null)
        {
            $this->fkContinent = $fkContinent;

            return $this;
        }

        /**
         * Get fkContinent
         *
         * @return \EntityBundle\Entity\GeographyContinent 
         */
        public function getFkContinent()
        {
            return $this->fkContinent;
        }
    }       

    ?>


    <?php

    namespace EntityBundle\Entity;

    use Doctrine\ORM\Mapping as ORM;

    /**
     * @ORM\Entity(repositoryClass="EntityBundle\EntityRepository\GeographyProvinceRepository")
     * @ORM\Table(
     *     name="geography_province", 
     *     uniqueConstraints={
     *         @ORM\UniqueConstraint(name="geography_province_u1", columns={"country_id", "id"}),
     *         @ORM\UniqueConstraint(name="geography_province_u2", columns={"country_id", "province_name"})
     *     }
     * )
     */
    class GeographyProvince 
    {
        /**
        * @ORM\Column(name="id", type="integer", nullable=false)
        * @ORM\Id
        * @ORM\GeneratedValue(strategy="AUTO")
        */
        protected $id;            

        /**
         *
         * @ORM\Column(name="province_name", type="string", nullable=false)
         */    
        protected $provinceName;        

        /**
        * @ORM\Column(name="country_id", type="integer", nullable=false)
        */
        protected $countryId;        

        /**
         *
         * @ORM\ManyToOne(targetEntity="GeographyCountry", cascade={"persist", "remove"})
         * @ORM\JoinColumn(name="country_id", referencedColumnName="id")
         */    
        protected $fkCountry;            

        /**
         *
         * @ORM\Column(name="description", type="string", nullable=true)
         */    
        protected $description;    

        /**
         * Get id
         *
         * @return integer 
         */
        public function getId()
        {
            return $this->id;
        }

        /**
         * Set provinceName
         *
         * @param string $provinceName
         * @return GeographyProvince
         */
        public function setProvinceName($provinceName)
        {
            $this->provinceName = $provinceName;

            return $this;
        }

        /**
         * Get provinceName
         *
         * @return string 
         */
        public function getProvinceName()
        {
            return $this->provinceName;
        }

        /**
         * Set countryId
         *
         * @param integer $countryId
         * @return GeographyProvince
         */
        public function setCountryId($countryId)
        {
            $this->countryId = $countryId;

            return $this;
        }

        /**
         * Get countryId
         *
         * @return integer 
         */
        public function getCountryId()
        {
            return $this->countryId;
        }

        /**
         * Set description
         *
         * @param string $description
         * @return GeographyProvince
         */
        public function setDescription($description)
        {
            $this->description = $description;

            return $this;
        }

        /**
         * Get description
         *
         * @return string 
         */
        public function getDescription()
        {
            return $this->description;
        }

        /**
         * Set fkCountry
         *
         * @param \EntityBundle\Entity\GeographyCountry $fkCountry
         * @return GeographyProvince
         */
        public function setFkCountry(\EntityBundle\Entity\GeographyCountry $fkCountry = null)
        {
            $this->fkCountry = $fkCountry;

            return $this;
        }

        /**
         * Get fkCountry
         *
         * @return \EntityBundle\Entity\GeographyCountry 
         */
        public function getFkCountry()
        {
            return $this->fkCountry;
        }
    }

    ?>


    <?php

    namespace EntityBundle\Entity;

    use Doctrine\ORM\Mapping as ORM;

    /**
     * @ORM\Entity(repositoryClass="EntityBundle\EntityRepository\GeographyCityRepository")
     * @ORM\Table(
     *     name="geography_city",
     *     uniqueConstraints={
     *         @ORM\UniqueConstraint(name="geography_city_u1", columns={"province_id", "is_municipality", "city_name"})
     *     }
     * )
     */
    class GeographyCity 
    {
        /**
        * @ORM\Column(name="id", type="integer", nullable=false)
        * @ORM\Id
        * @ORM\GeneratedValue(strategy="AUTO")
        */
        protected $id;            

        /**
         *
         * @ORM\Column(name="city_name", type="string", nullable=false)
         */    
        protected $cityName;        

        /**
         *
         * @ORM\Column(name="is_municipality", type="boolean", nullable=true)
         */    
        protected $isMunicipality;            

        /**
         *
         * @ORM\Column(name="province_id", type="integer", nullable=true)
         */    
        protected $provinceId;         

        /**
         *
         * @ORM\Column(name="country_id", type="integer", nullable=false)
         */    
        protected $countryId;            

        /**
         *
         * @ORM\ManyToOne(targetEntity="GeographyCountry", cascade={"persist", "remove"})
         * @ORM\JoinColumn(name="country_id", referencedColumnName="id", nullable=false)
         */    
        protected $fkCountry;            

        /**
         *
         * @ORM\ManyToOne(targetEntity="GeographyProvince", cascade={"persist", "remove"})
         * @ORM\JoinColumns
         * (
         *   {
         *     @ORM\JoinColumn(name="country_id", referencedColumnName="country_id", nullable=false),
         *     @ORM\JoinColumn(name="province_id", referencedColumnName="id", nullable=true)
         *   }
         * )
         */    
        protected $fkProvince;            

        /**
         *
         * @ORM\Column(name="description", type="string", nullable=true)
         */    
        protected $description;        

        /**
         * Get id
         *
         * @return integer 
         */
        public function getId()
        {
            return $this->id;
        }

        /**
         * Set cityName
         *
         * @param string $cityName
         * @return GeographyCity
         */
        public function setCityName($cityName)
        {
            $this->cityName = $cityName;

            return $this;
        }

        /**
         * Get cityName
         *
         * @return string 
         */
        public function getCityName()
        {
            return $this->cityName;
        }

        /**
         * Set isMunicipality
         *
         * @param boolean $isMunicipality
         * @return GeographyCity
         */
        public function setIsMunicipality($isMunicipality)
        {
            $this->isMunicipality = $isMunicipality;

            return $this;
        }

        /**
         * Get isMunicipality
         *
         * @return boolean 
         */
        public function getIsMunicipality()
        {
            return $this->isMunicipality;
        }

        /**
         * Set provinceId
         *
         * @param integer $provinceId
         * @return GeographyCity
         */
        public function setProvinceId($provinceId)
        {
            $this->provinceId = $provinceId;

            return $this;
        }

        /**
         * Get provinceId
         *
         * @return integer 
         */
        public function getProvinceId()
        {
            return $this->provinceId;
        }

        /**
         * Set countryId
         *
         * @param integer $countryId
         * @return GeographyCity
         */
        public function setCountryId($countryId)
        {
            $this->countryId = $countryId;

            return $this;
        }

        /**
         * Get countryId
         *
         * @return integer 
         */
        public function getCountryId()
        {
            return $this->countryId;
        }

        /**
         * Set description
         *
         * @param string $description
         * @return GeographyCity
         */
        public function setDescription($description)
        {
            $this->description = $description;

            return $this;
        }

        /**
         * Get description
         *
         * @return string 
         */
        public function getDescription()
        {
            return $this->description;
        }

        /**
         * Set fkCountry
         *
         * @param \EntityBundle\Entity\GeographyCountry $fkCountry
         * @return GeographyCity
         */
        public function setFkCountry(\EntityBundle\Entity\GeographyCountry $fkCountry)
        {
            $this->fkCountry = $fkCountry;

            return $this;
        }

        /**
         * Get fkCountry
         *
         * @return \EntityBundle\Entity\GeographyCountry 
         */
        public function getFkCountry()
        {
            return $this->fkCountry;
        }

        /**
         * Set fkProvince
         *
         * @param \EntityBundle\Entity\GeographyProvince $fkProvince
         * @return GeographyCity
         */
        public function setFkProvince(\EntityBundle\Entity\GeographyProvince $fkProvince)
        {
            $this->fkProvince = $fkProvince;

            return $this;
        }

        /**
         * Get fkProvince
         *
         * @return \EntityBundle\Entity\GeographyProvince 
         */
        public function getFkProvince()
        {
            return $this->fkProvince;
        }
    }

    ?>




上面的四个实体能够正确生成SQL DDL PostgreSQL。
在生成的sql下面(经过少量修改):

    CREATE TABLE geography_continent 
    (
        id INT NOT NULL, 
        continent_name VARCHAR(255) NOT NULL, 
        description VARCHAR(255) DEFAULT NULL, 
        PRIMARY KEY(id)
    );

    CREATE TABLE geography_country 
    (
        id INT NOT NULL, 
        continent_id INT NOT NULL, 
        country_name VARCHAR(255) NOT NULL, 
        telephone_code VARCHAR(255) DEFAULT NULL, 
        currency_name VARCHAR(255) DEFAULT NULL, 
        currency_symbol VARCHAR(255) DEFAULT NULL, 
        description VARCHAR(255) DEFAULT NULL, 
        PRIMARY KEY(id)
    );

    CREATE TABLE geography_province 
    (
        id INT NOT NULL, 
        country_id INT NOT NULL, 
        province_name VARCHAR(255) NOT NULL, 
        description VARCHAR(255) DEFAULT NULL, 
        PRIMARY KEY(id)
    );

    CREATE TABLE geography_city 
    (
        id INT NOT NULL, 
        country_id INT NOT NULL, 
        province_id INT DEFAULT NULL, 
        city_name VARCHAR(255) NOT NULL, 
        is_municipality BOOLEAN DEFAULT NULL, 
        description VARCHAR(255) DEFAULT NULL, 
        PRIMARY KEY(id)
    );

    CREATE SEQUENCE geography_continent_id_seq INCREMENT BY 1 MINVALUE 1 START 1;
    CREATE SEQUENCE geography_country_id_seq INCREMENT BY 1 MINVALUE 1 START 1;
    CREATE SEQUENCE geography_province_id_seq INCREMENT BY 1 MINVALUE 1 START 1;
    CREATE SEQUENCE geography_city_id_seq INCREMENT BY 1 MINVALUE 1 START 1;        
    CREATE UNIQUE INDEX geography_continent_u1 ON geography_continent (continent_name);        
    CREATE INDEX IDX_6D7254DD921F4C77 ON geography_country (continent_id);
    CREATE UNIQUE INDEX geography_country_u1 ON geography_country (country_name);
    CREATE UNIQUE INDEX geography_country_u2 ON geography_country (telephone_code);
    CREATE UNIQUE INDEX geography_country_u3 ON geography_country (currency_name);
    CREATE UNIQUE INDEX geography_country_u4 ON geography_country (currency_symbol);        
    CREATE INDEX IDX_1657BF92F92F3E70 ON geography_province (country_id);
    CREATE UNIQUE INDEX geography_province_u1 ON geography_province (country_id, id);
    CREATE UNIQUE INDEX geography_province_u2 ON geography_province (country_id, province_name);        
    CREATE INDEX IDX_3F82CFCAF92F3E70 ON geography_city (country_id);
    CREATE INDEX IDX_3F82CFCAF92F3E70E946114A ON geography_city (country_id, province_id);
    CREATE UNIQUE INDEX geography_city_u1 ON geography_city (province_id, is_municipality, city_name);

    ALTER TABLE geography_country ADD CONSTRAINT FK_6D7254DD921F4C77 FOREIGN KEY (continent_id) REFERENCES geography_continent (id) MATCH FULL ON UPDATE CASCADE ON DELETE RESTRICT NOT DEFERRABLE INITIALLY IMMEDIATE;
    ALTER TABLE geography_province ADD CONSTRAINT FK_1657BF92F92F3E70 FOREIGN KEY (country_id) REFERENCES geography_country (id) MATCH FULL ON UPDATE CASCADE ON DELETE RESTRICT NOT DEFERRABLE INITIALLY IMMEDIATE;
    ALTER TABLE geography_city ADD CONSTRAINT FK_3F82CFCAF92F3E70 FOREIGN KEY (country_id) REFERENCES geography_country (id) MATCH FULL ON UPDATE CASCADE ON DELETE RESTRICT NOT DEFERRABLE INITIALLY IMMEDIATE;

    -- MATCH SIMPLE FOREIGN KEY
    ALTER TABLE geography_city ADD CONSTRAINT FK_3F82CFCAF92F3E70E946114A FOREIGN KEY (country_id, province_id) REFERENCES geography_province (country_id, id) MATCH SIMPLE ON UPDATE CASCADE ON DELETE RESTRICT NOT DEFERRABLE INITIALLY IMMEDIATE;

表“geography_city”有两个外键。
一个引用带有“country_id”列的表“geography_country”。
另一个引用表“geography_province”,其中包含两列“country_id”和“province_id”,列“province_id”是可选的,并且可以为NULL(签名为“MATCH SIMPLE”FOREIGN KEY),因为A Country of A Country可能没有省。



在控制器的主要PHP代码下面保留上面的四个实体:

    <?php

    $geographyContinentName = "A Continent Name";
    $geographyContinent = new GeographyContinent();
    $geographyContinent->setContinentName($geographyContinentName);

    $geographyCountryName = "A Country Name";
    $geographyCountry = new GeographyCountry();
    $geographyCountry->setCountryName($geographyCountryName);
    $geographyCountry->setFkContinent($geographyContinent);

    $geographyProvinceName = "A Province Name";
    $geographyProvince = new GeographyProvince();
    $geographyProvince->setProvinceName($geographyProvinceName);
    $geographyProvince->setFkCountry($geographyCountry);

    $geographyCityName = "A City Name";
    $geographyCity = new GeographyCity();
    $geographyCity->setCityName($geographyCityName);
    $geographyCity->setFkCountry($geographyCountry);
    $geographyCity->setFkProvince($geographyProvince);

    $entityManager = $this->getDoctrine()->getManager();
    $entityManager->persist($geographyContinent);
    $entityManager->persist($geographyCountry);
    $entityManager->persist($geographyProvince);
    $entityManager->persist($geographyCity);
    $entityManager->flush();

    ?>




低于运行上述控制器后产生的错误:

    [2013-12-15 06:41:38] request.INFO: 
        Matched route "entity_geography_create"
        (
           parameters: 
               "_controller": "Entity\GeographyBundle\Controller\CreateController::indexAction", 
               "_route": "entity_geography_create"
        )

    [2013-12-15 06:41:38] app.ERROR: 
        Doctrine\DBAL\DBALException: 
            An exception occurred while executing 
            'INSERT INTO geography_city (id, city_name, is_municipality, province_id, country_id, description) VALUES (?, ?, ?, ?, ?, ?)'
            with params [1, "A City Name", null, 1, null, null]
            SQLSTATE[23502]: Not null violation
            ERROR:  null value in column "country_id" violates not-null constraint 
            (uncaught exception) at D:\server\htdocs\application\vendor\doctrine\dbal\lib\Doctrine\DBAL\DBALException.php line 47

我希望在插入“geography_city”表时会自动提供列“country_id”的值,但基于上面的错误,它不会。


任何帮助,我真的很感激。
非常感谢你。
最诚挚的问候。

2 个答案:

答案 0 :(得分:1)

@ n.1非常感谢您的帮助。

我的完整交易代码运行良好。

我执行命令:

    php app/console doctrine:cache:clear-metadata
    php app/console doctrine:cache:clear-query
    php app/console doctrine:cache:clear-result
    php app/console cache:clear --no-warmup

我的问题现在解决了。

感谢上帝。

答案 1 :(得分:0)

我刚刚遇到同样的问题,并在每次调用$entityManager->flush();后致电$entityManager->persist($your_new_entity);解决了问题。它将确保持久化实体将具有id,因此该id将可用于将其用作外键的其他实体。

所以你必须致电

$entityManager->persist($your_new_entity);
$entityManager->flush();

在将新实体用作外键之前。

通过定义级联操作似乎是可行的,有关详细信息,请参阅documentation of Doctrine2 about Persisting entities