【问题标题】:Overlapped Non Primary Key for Composite Foreign Key Doctrine Symfony 2复合外键学说 Symfony 2 的重叠非主键
【发布时间】:2013-12-09 19:36:19
【问题描述】:

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

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

我试图修改学说对象,但我仍然无法将所有实体都保存到 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”。
另一个是指具有两列“country_id”和“province_id”的表“geography_province”,列“province_id”是可选的并且可能为NULL(签名为“MATCH SIMPLE”FOREIGN KEY),因为A City 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”的值,但根据上面的错误,它不会。


任何帮助我都会非常感激。
非常感谢你。
最好的问候。

【问题讨论】:

  • 什么不起作用?你有错误信息吗?
  • 为什么不在每个表中使用整数作为 id?我认为在所有表中用作主键和外键会更容易。
  • 请在您的问题中添加错误信息并将其从 cmets 中删除,这样会更容易阅读。
  • country_id text NOT NULL, 这个外键应该是整数,而不是字符串。
  • @n.1 经过额外的实验我仍然无法解决我的问题,非常感谢您的建议以更完整地填写我的问题信息

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


【解决方案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

我的问题现在解决了。

感谢上帝。

【讨论】:

    【解决方案2】:

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

    所以你必须打电话

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

    在使用新实体作为外键之前。

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

    【讨论】:

      猜你喜欢
      • 2011-09-17
      • 1970-01-01
      • 1970-01-01
      • 2016-07-04
      • 2018-04-07
      • 1970-01-01
      • 1970-01-01
      • 2013-06-29
      • 1970-01-01
      相关资源
      最近更新 更多