MySQL应用之CROSS JOIN用法简介教程

@[toc]
本博客翻译自两篇博客的:

1. cross join简介

MySQL cross join是mysql中的一种连接方式,区别于内连接和外连接,对于cross join连接来说,其实使用的就是笛卡尔连接。在MySQL中,当CROSS JOIN不使用WHERE子句时,CROSS JOIN产生了一个结果集,该结果集是两个关联表的行的乘积。通常,如果每个表分别具有n和m行,则结果集将具有n*m行

引用https://www.w3resource.com/mysql/advance-query-in-mysql/mysql-cross-join.php的图片,如图演示了cross join的过程,这个过程其实就是笛卡尔连接查询

在这里插入图片描述

2. cross join用法

cross join用法:

SELECT * FROM t1
CROSS JOIN t2;

注意:cross join的时候是不需要on或者using关键字的,这个是区别于inner join和join的

如果WHERE在条件表中添加一个子句t1并t2具有关系,则CROSS JOIN该INNER JOIN子句的工作方式类似于以下查询中所示:

SELECT * FROM t1
CROSS JOIN t2
WHERE t1.id = t2.id;

ok,再列举一下cross join表作为衍生表的例子

SELECT * 
FROM table111
LEFT JOIN(table112 CROSS JOIN table113)
ON table111.id=table113.id;

ok,介绍了cross join的简单用法,现在拿http://www.mysqltutorial.org/mysql-cross-join/的例子来介绍:

首先,创建一个新数据库salesdb:

CREATE DATABASE IF NOT EXISTS salesdb;

其次,将当前数据切换到新数据库testdb:

USE testdb;

在salesdb数据库中创建新表:

  • 该表 products包含产品主数据,其中包括产品ID,产品名称和销售价格。
  • 该表stores包含出售产品的商店。
  • 该表sales包含按数量和日期在特定商店中出售的产品。

CREATE TABLE products (
    id INT PRIMARY KEY AUTO_INCREMENT,
    product_name VARCHAR(100),
    price DECIMAL(13,2 )
);
 
CREATE TABLE stores (
    id INT PRIMARY KEY AUTO_INCREMENT,
    store_name VARCHAR(100)
);
 
CREATE TABLE sales (
    product_id INT,
    store_id INT,
    quantity DECIMAL(13 , 2 ) NOT NULL,
    sales_date DATE NOT NULL,
    PRIMARY KEY (product_id , store_id),
    FOREIGN KEY (product_id)
        REFERENCES products (id)
        ON DELETE CASCADE ON UPDATE CASCADE,
    FOREIGN KEY (store_id)
        REFERENCES stores (id)
        ON DELETE CASCADE ON UPDATE CASCADE
);

将数据插入三个表中。假设我们有三个产品iPhone,iPad并且Macbook Pro其在两个商店出售North和South。

INSERT INTO products(product_name, price)
VALUES('iPhone', 699),
      ('iPad',599),
      ('Macbook Pro',1299);
 
INSERT INTO stores(store_name)
VALUES('North'),
      ('South');
 
INSERT INTO sales(store_id,product_id,quantity,sales_date)
VALUES(1,1,20,'2017-01-02'),
      (1,2,15,'2017-01-05'),
      (1,3,25,'2017-01-05'),
      (2,1,30,'2017-01-02'),
      (2,2,35,'2017-01-05');

ok,业务场景:现在要统计每个商店每种商品总共营业额是多少钱?

很显然,用SUM(quantity * price),再group by一下就可以,这个sql很好写

SELECT 
  sto.`store_name`,
  pro.`product_name`,
  SUM(quantity * price) AS revenue
FROM
  sales sal 
  INNER JOIN stores sto 
    ON sto.`id` = sal.`store_id`
  INNER JOIN products pro
    ON sal.`product_id` = pro.`id`
  GROUP BY sto.`store_name`,pro.`product_name`;
在这里插入图片描述

ok,看了一下,发现没卖出的商品是没统计出来的,所以不太符合业务需求,业务是要统计所有的商店商品,所以可以用cross join笛卡尔连接,得出所有的商店商品组合数据

笛卡尔查询组合数据sql:

SELECT 
  a.`store_name`,
  b.product_name
from stores cross join products
在这里插入图片描述

前面统计sql已经有了,所以将组合数据SQL和统计数据的SQL进行关联:

SELECT 
  a.`store_name`,
  b.product_name,
  IFNULL(c.revenue, 0) AS revenue 
FROM
  stores a 
  CROSS JOIN products b 
  LEFT JOIN 
    (SELECT 
      sto.`id` AS store_id,
      pro.`id` AS product_id,
      sto.`store_name`,
      pro.`product_name`,
      SUM(quantity * price) AS revenue 
    FROM
      sales sal 
      INNER JOIN stores sto 
        ON sto.`id` = sal.`store_id` 
      INNER JOIN products pro 
        ON sal.`product_id` = pro.`id` 
    GROUP BY sto.`store_name`,
      pro.`product_name`) c 
    ON a.id = c.store_id 
    AND b.id = c.product_id 
ORDER BY a.store_name ;

在这里插入图片描述

请注意,IFNULL如果收入为NULL (在商店没有销售的情况下),查询使用该函数返回0。

通过CROSS JOIN这种方式使用该子句,您可以回答广泛的问题,例如,按销售员,月份查找销售收入,即使该销售员在特定月份没有销售。

ok,本博客是翻译两篇英文博客的:

ok,本博客内容翻译自两篇英文博客,不过本博客进行一定修整,将两篇博客内容进行理解整合成这篇中文博客,原因是这两篇博客的例子还是不错的,举出了cross join的常用使用场景,当然除了两篇博客提出的用法,cross join因为其笛卡尔连接的特性,还可以用于批量写数据,对应批量的写法,可以参考我之前的MySQL博客,本博客性质属于翻译的,所以转载请注明出处

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容

  • MySQL技术内幕:SQL编程 姜承尧 第1章 SQL编程 >> B是由MySQL创始人之一Monty分支的一个版...
    沉默剑士阅读 2,404评论 0 3
  • 翻译完成有一种莫名的幸福感--- JOIN 语法 Mysql支持select 语句和多表delete、update...
    LemonCandy阅读 787评论 0 0
  • 什么是数据库? 数据库是存储数据的集合的单独的应用程序。每个数据库具有一个或多个不同的API,用于创建,访问,管理...
    chen_000阅读 4,028评论 0 19
  • 昨儿下了一整天雨,没有去跳绳,心里总觉得有些惴惴然,以至于午睡时做梦,梦见说是晚饭后雨停了,对我跳绳没有丝毫影响,...
    dhmlr阅读 220评论 0 1
  • 28天快结束了,饮食习惯还是没有什么改善,每天打卡偶尔得姐点评说有进步的那天,是老人没有在家里,所以炒菜时可以多样...
    annie_bbe1阅读 244评论 0 0