2015-05-27 2 views
2

Я следующую структуру базы данных:Как выбрать из трех таблиц в одном запросе

db_config_list id | id_config | type

DB_CONFIG id | id_list | name | value

db_config_lang id | id_list | id_lang | name | value

В одном запросе MYSQL Я хотел бы, чтобы выбрать вещи из db_config и db_config_lang где db_config.id_list = db_config_list.id и db_config_lang.id_list = db_config_list.id

Я попытался:

$sql = 'SELECT * FROM `'._DB_PREFIX_.$this->db_config_list.'` cl 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config.'` c 
     ON (cl.id = c.id_list) 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config_lang.'` cll 
     ON (cl.id = cll.id_list) 
     WHERE cl.id_config = '.$this->c_id.' 
     AND cl.type = "'.$this->c_type.'" 
     AND cll.id_lang = "'.$this->default_lang.'"'; 

Но это не работает надлежащим. Это на самом деле работает, когда я делаю это в двух разных запросов:

$sql = 'SELECT * FROM `'._DB_PREFIX_.$this->db_config_list.'` cl 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config.'` c 
     ON (cl.id = c.id_list) 
     WHERE cl.id_config = '.$this->c_id.' 
     AND cl.type = "'.$this->c_type.'"'; 

$sql = 'SELECT * FROM `'._DB_PREFIX_.$this->db_config_list.'` cl 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config_lang.'` cll 
     ON (cl.id = cll.id_list) 
     WHERE cll.id_lang = "'.$this->default_lang.'"'; 

Но я хотел бы сделать это в одном запросе. Это возможно ?

// EDIT

Я думаю, что я пытался добиться чего-то невозможного. Поэтому я выбрал другое решение

$sql = 'SELECT name,value FROM `'._DB_PREFIX_.$this->db_config_list.'` cl 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config.'` c 
     ON (cl.id = c.id_list) 
     WHERE cl.id_config = '.$this->c_id.' 
     AND cl.type = "'.$this->c_type.'"'; 

$sql2 = 'SELECT name,value FROM `'._DB_PREFIX_.$this->db_config_list.'` cl 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config_lang.'` cll 
     ON (cl.id = cll.id_list) 
     WHERE cll.id_lang = "'.$this->default_lang.'"'; 

$query1 = Db::getInstance()->executeS($sql); 
$query2 = Db::getInstance()->executeS($sql2); 

$query = array_merge($query1, $query2); 

// EDIT 2

Вот код для sqlfiddle.com

Создать код:

CREATE TABLE IF NOT EXISTS `db_config_list` (
    `id` int(10) unsigned NOT NULL AUTO_INCREMENT, 
    `id_config` int(10) unsigned NOT NULL, 
    `type` varchar(10) NOT NULL, 
    PRIMARY KEY (`id`) 
) DEFAULT CHARSET=UTF8; 

CREATE TABLE IF NOT EXISTS `db_config` (
    `id_list` int(10) unsigned NOT NULL, 
    `id` int(10) unsigned NOT NULL AUTO_INCREMENT, 
    `name` varchar(255) NOT NULL, 
    `value` text NOT NULL, 
    PRIMARY KEY (`id`) 
) DEFAULT CHARSET=UTF8; 


CREATE TABLE IF NOT EXISTS `db_config_lang` (
    `id_list` int(10) unsigned NOT NULL, 
    `id_lang` int(10) unsigned NOT NULL, 
    `id` int(10) unsigned NOT NULL AUTO_INCREMENT, 
    `name` varchar(255) NOT NULL, 
    `value` text NOT NULL, 
    PRIMARY KEY (`id`) 
) DEFAULT CHARSET=UTF8; 

ALTER TABLE `db_config` ADD INDEX(`id_list`); 

ALTER TABLE `db_config_lang` ADD INDEX(`id_list`); 

ALTER TABLE `db_config` ADD FOREIGN KEY (`id_list`) REFERENCES `db_config_list`(`id`) ON DELETE CASCADE ON UPDATE NO ACTION; 

ALTER TABLE `db_config_lang` ADD FOREIGN KEY (`id_list`) REFERENCES `db_config_list`(`id`) ON DELETE CASCADE ON UPDATE NO ACTION; 

INSERT INTO `db_config_list` (`id`, `id_config`, `type`) VALUES 
(1, 1, 'global'); 

INSERT INTO `db_config` (`id_list`, `id`, `name`, `value`) VALUES 
(1, 1, 'font_family', ''), 
(1, 2, 'first_main_color', '#19BCE7'), 
(1, 3, 'use_background_image', '1'), 
(1, 4, 'background_color', '#F60'), 
(1, 5, 'animated_tabs_carousel', '1'), 
(1, 6, 'boxed_layout', '1'), 
(1, 7, 'loading_animation', '1'), 
(1, 8, 'smooth_scroll', '1'), 
(1, 9, 'responsiveness', '1'), 
(1, 10, 'sticky_header', '1'); 

INSERT INTO `db_config_lang` (`id_list`, `id_lang`, `id`, `name`, `value`) VALUES 
(1, 1, 1, 'center_column_content', 'boo foo'), 
(1, 1, 2, 'bottom_column_content', 'boo foo'), 
(1, 2, 3, 'center_column_content', 'boo foo'), 
(1, 2, 4, 'bottom_column_content', 'boo foo'), 
(1, 3, 5, 'center_column_content', 'boo foo'), 
(1, 3, 6, 'bottom_column_content', 'boo foo'), 
(1, 4, 7, 'center_column_content', 'boo foo'), 
(1, 4, 8, 'bottom_column_content', 'boo foo'); 

Выберите код:

SELECT name,value FROM `db_config_list` cl 
LEFT JOIN `db_config` c 
ON (cl.id = c.id_list) 
WHERE cl.id_config = 1 
AND cl.type = "global"; 

SELECT name,value FROM `db_config_list` cl 
LEFT JOIN `db_config_lang` cll 
ON (cl.id = cll.id_list) 
WHERE cll.id_lang = 1 
AND cl.id_config = 1 
AND cl.type = "global"; 

ответ

3

Я думаю, что более вероятно, что выбор - это атрибут *, который будет выбирать только одно значение для каждого имени столбца, независимо от того, сколько таблиц соединено. Если вы можете разместить SQL Fiddle для этой настройки, я могу помочь вам изменить запрос. По существу, я считаю, что вы хотите сделать что-то вроде:

$sql = 'SELECT *, c.name AS config_name, cll.name as config_lang_name 
     FROM `'._DB_PREFIX_.$this->db_config_list.'` cl 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config.'` c 
     ON (cl.id = c.id_list) 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config_lang.'` cll 
     ON (cl.id = cll.id_list) 
     WHERE cl.id_config = '.$this->c_id.' 
     AND cl.type = "'.$this->c_type.'" 
     AND cll.id_lang = "'.$this->default_lang.'"'; 

EDIT: Doing два отдельных запросов, а затем объединение наборов результатов будет гораздо медленнее, чем регулировать запрос, чтобы вернуть нужный набор результатов. Этот запрос ближе к тому, что вы хотели? (Выше был только один пример, теперь, когда вы показали свою конечную цель, я считаю, что это должно работать.)

$sql = 'SELECT cl.name AS config_list_name, cl.value AS config_list_value, 
     cll.name AS config_lang_name, cll.value AS config_lang_value 
     FROM `'._DB_PREFIX_.$this->db_config_list.'` cl 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config.'` c 
     ON (cl.id = c.id_list) 
     LEFT JOIN `'._DB_PREFIX_.$this->db_config_lang.'` cll 
     ON (cl.id = cll.id_list) 
     WHERE cl.id_config = '.$this->c_id.' 
     AND cl.type = "'.$this->c_type.'" 
     AND cll.id_lang = "'.$this->default_lang.'"'; 

EDIT 2: Похоже, вы используете рамки для этого запроса.Я бы рекомендовал использовать некоторую привязку параметров вместо того, чтобы вставлять значения непосредственно в запрос.

+0

Это может быть возможно. Я проверю вашу идею. Thx для ответа, –

+0

@WalterWhite, не забудьте принять ответ Микеля, если он решает вашу проблему. – SpencerD

+0

@Mikel Bitson Вы открыли мне глаза. Проблема решена, я обновил сообщение. Спасибо. –

Смежные вопросы