-- create
CREATE TABLE employee
(
emp_id INTEGER,
job_code TEXT,
cnt_check int
);
-- insert
INSERT INTO employee(emp_id,job_code) VALUES (0001, 'JC001');
INSERT INTO employee(emp_id,job_code) VALUES (0001, 'JC001');
INSERT INTO employee(emp_id,job_code) VALUES (0002, 'JC002');
INSERT INTO employee(emp_id,job_code) VALUES (0002, 'JC002');
INSERT INTO employee(emp_id,job_code) VALUES (0003, 'JC003');
INSERT INTO employee(emp_id,job_code) VALUES (0004, 'JC004');
INSERT INTO employee(emp_id,job_code) VALUES (0004, 'JC004');
INSERT INTO employee(emp_id,job_code) VALUES (0004, 'JC004');
Expected Output:
emp_id job_code cnt_check
--------------------------------
0001 JC001 1
0001 JC001 2
0002 JC002 1
0002 JC002 2
0003 JC003 1
0004 JC004 1
0004 JC004 2
0004 JC004 3
My try:
update employee t1
join
(
select emp_id ,job_code ,row_number() over(partition by emp_id ,job_code ) rnk
from employee
) t2
on t1.emp_id = t2.emp_id and t1.job_code = t2.job_code
set t1.cnt_check = t2.rnk;
But all records getting updated with value 1.
Demo link
Using the exact columns in your sample data, we can try:
SELECT emp_id, job_code,
ROW_NUMBER() OVER (PARTITION BY emp_id ORDER BY emp_id) AS cnt_check
FROM employee
ORDER BY 1, 3;
Note that I suggest not doing this update, as every time your data changes you might have to run the update again.
You may use generation which uses BEFORE INSERT trigger and additional table:
CREATE TABLE autoinc_helper (
emp_id INT,
cnt_check INT AUTO_INCREMENT,
PRIMARY KEY (emp_id, cnt_check)
) ENGINE = MyISAM;
CREATE TRIGGER generate_autoinc
BEFORE INSERT ON employee
FOR EACH ROW
BEGIN
DECLARE autoinc INT;
INSERT INTO autoinc_helper (emp_id) VALUES (NEW.emp_id);
SET autoinc = LAST_INSERT_ID();
SET NEW.cnt_check = autoinc;
DELETE FROM autoinc_helper WHERE emp_id = NEW.emp_id AND cnt_check < autoinc;
END
DEMO fiddle
Related
The goal is to insert a new record or update an existing if the id column exists. The ON DUPLICATE KEY UPDATE function seems to be a good fit but the problem I have is I want to make sure that the ID matches an ID in another table. This is for permissions. An example is:
INSERT INTO `testtable` (`id`, `user_id`, `order`, `active`)
VALUES
(1, 24, 3, 0),
(2, 24, 1, 1),
(NULL, 24, 2, 0) AS newupdate
ON DUPLICATE KEY UPDATE
order = newupdate.order,
active = newupdate.active
This works but I want to add a check where the testtable.user_id from above matches users.id from another table. I've been using this SELECT statement:
SELECT testtable.id, testtable.order, testtable.active
FROM testtable
INNER JOIN users on testtable.user_id = users.id
WHERE users.username = bob
So I can't figure out how to utilize the ON DUPLICATE KEY UPDATE with an INNER JOIN or perhaps I'm going about this all the wrong way. I search Stackoverflow and saw some suggestions to use a SELECT statement with the ON DUPLICATE KEY UPDATE but I don't understand how that passes the values. Anyway, is this possible? Thanks in advance.
UPDATE: I was able to get it working but it seems like there should be an easier/better way? Anyone have any other suggestions? This example below I'm inserting a row with id 24 which exists so the update runs and then I'm inserting a row with id NULL so it gets inserted as is.
INSERT INTO `testtable` (`id`, `user_id`, `order`, `active`)
SELECT `id`, `user_id`, `order`, `active`
FROM (SELECT
24 AS id,
(SELECT user_id FROM testtable INNER JOIN users on testtable.user_id = users.id WHERE users.username = 'bob' LIMIT 1) AS user_id,
6 AS order,
1 AS active
UNION ALL
SELECT
NULL AS id,
(SELECT user_id FROM testtable INNER JOIN users on testtable.user_id = users.id WHERE users.username = 'bob' LIMIT 1) AS user_id,
7 AS order,
1 AS active
)
AS newtable
ON DUPLICATE KEY UPDATE
order = newtable.order,
active = newtable.active
Here we have a foreign key to users and a trigger to block changing user_id.
create table users(
id int primary key,
user varchar(25));
insert into users values
(12,'Andrew'),
(24,'Bernard');
✓
✓
create table testtable (
id int ,
user_id int unique,
order_ int,
active int,
foreign key fk_user_id (user_id)
references users(id));
✓
INSERT INTO `testtable` (`id`, `user_id`, `order_`, `active`)
VALUES
(1, 24, 3, 0),
(2, 24, 1, 1),
(NULL, 24, 2, 0) AS newupdate
ON DUPLICATE KEY UPDATE
order_ = newupdate.order_,
active = newupdate.active
✓
/*delimiter $$ */
create trigger user_id_update
before update on testtable
for each row
begin
if (old.user_id <>new.user_id) then
signal SQLSTATE VALUE '45000'
SET MESSAGE_TEXT = 'user_id change forbidden';
end if ;
end ;
/* delimiter ; */
✓
✓
select * from testtable;
id | user_id | order_ | active
-: | ------: | -----: | -----:
1 | 24 | 2 | 0
update testtable
set user_id = 12 where user_id = 24;
user_id change forbidden
select * from testtable;
id | user_id | order_ | active
-: | ------: | -----: | -----:
1 | 24 | 2 | 0
db<>fiddle here
Hello to every poeple of stackoverflow community
i want to ask something to you
The type i don't know how can i compare it and to what
CREATE TABLE spawnhist (
`npctempladeld` INTEGER,
`count` INTEGER
);
INSERT INTO spawnhist
(`npctempladeld`, `count`)
VALUES
('100', '1'),
('200', '1'),
('300', '1');
CREATE TABLE npc (
`npcid` INTEGER,
`type` VARCHAR(9)
);
INSERT INTO npc
(`npcid`, `type`)
VALUES
('100', 'L2Monster'),
('200', 'L2NPC'),
('300', 'L2PET');
Update spawnhist SET `count` = `count` +1
WHERE npctempladeld IN (SELECT `npcid` FROM npc WHERE type = 'L2Monster')
SELECT * FROM spawnhist
npctempladeld | count
------------: | ----:
100 | 2
200 | 1
300 | 1
db<>fiddle here
Something like, ( what I understood at first glance)
My code is based on oracle but I believe it works the same in MySQL. Please try.
Update spawnlist sl
Set count = 2
Where exists (select 1
from npc nc
where nc.npc_id = sl.npc_templadeId
and nc.type = 'L2Monster')
Table Name: course_trainer_combination
Create Table Query:
create table `course_trainer_combination` (
`id` double ,
`course` varchar (150),
`trainer` varchar (150),
`distance` float
);
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('1','Course A','Trainer A','110.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('2','Course A','Trainer B','105.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('3','Course A','Trainer C','115.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('4','Course B','Trainer A','112.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('5','Course B ','Trainer B','108.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('6','Course B','Trainer C','109.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('7','Course C','Trainer A','124.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('8','Course C','Trainer B','128.00');
insert into `course_trainer_combination` (`id`, `course`, `trainer`, `distance`) values('9','Course C','Trainer C','121.00');
My table is like
Expected result:
2 | Course A | Trainer B | 105
6 | Course B | Trainer C | 109
7 | Course C | Trainer A | 124
You can use correlated subquery
select ct.*
from course_trainer_combination ct
where ct.distance = (select min(ct1.distance)
from course_trainer_combination ct1
where ct1.course = ct.course
);
If you are using newer version then you can use analytical function :
select ct.*
from (select ct.*,
rank() over (partition by ct.course order by ct.distance) as seq
from course_trainer_combination ct
) ct
where ct.seq = 1;
Another approach can be like following.
select id,
m.course,
trainer,
m.distance
from course_trainer_combination m
inner join (select course,
Min(distance) distance
from course_trainer_combination
group by course)t
on t.course = m.course
and t.distance = m.distance
Note: This will show two records if two trainer have same distance for 1 course.
One way to solve this is by:
keeping track of the trainers assigned to each course and
selecting a row with the below conditions:
the trainer was not already assigned to a course
no other row of the same course, with a trainer not yet selected, exists
Use a mysql user variable to keep track of the trainers already selected and use FIND_IN_SET() to check weather the trainer has already assigned to a course:
SET #selectedTrainers :='';
SELECT *, #selectedTrainers := CONCAT(#selectedTrainers, ',', a.trainer) AS selectedTrainers
FROM `course_trainer_combination` a
where NOT EXISTS (
SELECT 1
FROM `course_trainer_combination` b
WHERE b.course = a.course
AND a.distance > b.distance
AND FIND_IN_SET(b.trainer, #selectedTrainers) = 0
)
AND FIND_IN_SET(a.trainer, #selectedTrainers) = 0
This problem consists of a database system relating components to their suppliers with corresponding costs, which are used to build widget assemblies.
Here is the database and tables:
CREATE DATABASE IF NOT EXISTS DB;
USE DB;
CREATE TABLE components (
c_id INT AUTO_INCREMENT NOT NULL,
name_c VARCHAR(255),
PRIMARY KEY (c_id)
);
CREATE TABLE suppliers (
s_id INT AUTO_INCREMENT NOT NULL,
name_s VARCHAR(255),
PRIMARY KEY (s_id)
);
CREATE TABLE widgets (
w_id INT AUTO_INCREMENT NOT NULL,
name_w VARCHAR(255),
PRIMARY KEY (w_id)
);
CREATE TABLE assemblies (
w_id INT,
c_id INT,
FOREIGN KEY (w_id)
REFERENCES widgets (w_id),
FOREIGN KEY (c_id)
REFERENCES components (c_id)
);
CREATE TABLE prices (
p_id INT AUTO_INCREMENT NOT NULL,
s_id INT,
c_id INT,
cost INT,
PRIMARY KEY (p_id),
FOREIGN KEY (s_id)
REFERENCES suppliers (s_id),
FOREIGN KEY (c_id)
REFERENCES components (c_id)
);
insert into components(name_c)
values('C1');
insert into components(name_c)
values('C2');
insert into components(name_c)
values('C3');
insert into components(name_c)
values('C4');
insert into components(name_c)
values('C5');
insert into components(name_c)
values('C6');
insert into suppliers(name_s)
values('S1');
insert into suppliers(name_s)
values('S2');
insert into suppliers(name_s)
values('S3');
insert into suppliers(name_s)
values('S4');
insert into suppliers(name_s)
values('S5');
insert into widgets(name_w)
values('A1');
insert into widgets(name_w)
values('A2');
insert into widgets(name_w)
values('A3');
insert into widgets(name_w)
values('A4');
insert into assemblies(w_id, c_id)
values(1,1);
insert into assemblies(w_id, c_id)
values(2,2);
insert into assemblies(w_id, c_id)
values(2,3);
insert into assemblies(w_id, c_id)
values(3,4);
insert into assemblies(w_id, c_id)
values(3,5);
insert into assemblies(w_id, c_id)
values(4,6);
insert into assemblies(w_id, c_id)
values(4,3);
insert into prices(s_id, c_id, cost)
values(1,1,121);
insert into prices(s_id, c_id, cost)
values(1,2,135);
insert into prices(s_id, c_id, cost)
values(2,2,94);
insert into prices(s_id, c_id, cost)
values(2,3,155);
insert into prices(s_id, c_id, cost)
values(3,3,178);
insert into prices(s_id, c_id, cost)
values(3,4,199);
insert into prices(s_id, c_id, cost)
values(4,4,122);
insert into prices(s_id, c_id, cost)
values(4,5,155);
insert into prices(s_id, c_id, cost)
values(5,5,133);
insert into prices(s_id, c_id, cost)
values(5,6,184);
The resulting tables are as follows:
components
c_id | name_c
1 C1
2 C2
3 C3
4 C4
5 C5
6 C6
(Note: tables components, suppliers, and widgets all defined analogously such that widgets with 1-A1, 2-A2, etc. and suppliers with 1-S1, 2-S2, etc.)
prices - list of supplier-component cost combinations
p_id s_id c_id cost
1 1 1 121
2 1 2 135
3 2 2 94
4 2 3 155
5 3 3 178
6 3 4 199
7 4 4 122
8 4 5 155
9 5 5 133
10 5 6 184
assemblies - list of components used to build the widget assemblies
w_id | c_id
1 1
2 2
2 3
3 4
3 5
4 6
4 3
For the first part, with help I was able to determine the correct query to find the minimum-cost outputs with respect to assemblies as follows (objective was to determine the minimum-cost components to build each widget assembly):
select w.name_w, sum(price_min.min_cost) as min_cost
from widgets as w
left join assemblies as a on a.w_id = w.w_id
left join (select c_id, min(cost) as min_cost from prices group by c_id) as price_min on price_min.c_id = a.c_id
group by w.name_w
order by w.w_id
Output:
name_w min_cost
A1 121
A2 249
A3 255
A4 339
However, I'm stuck trying to figure out how to determine which suppliers (name_s) are not used in this cost optimization (i.e., did not provide a competitive component price to be used in the corresponding widget-assembly build incorporating those suppliers' components). Working out on paper, I expect the following result:
name_s
S3
But I can't seem to get this result using SQL (specifically MySQL)...
Thanks in advance!
Modified from the query for finding the minimum cost:
select distinct s.name_s
from suppliers s
left join
prices p
on s.s_id = p.s_id
where p.s_id not in (
select distinct s_id from prices p1 -- suppliers of low cost components
inner join
(select a.c_id, price_min.min_cost
from widgets as w
left join assemblies as a on a.w_id = w.w_id
left join (select c_id, min(cost) as min_cost from prices group by c_id) as price_min
on price_min.c_id = a.c_id ) t -- component price details before sum()
on p1.cost = t.min_cost and
p1.c_id = t.c_id )
I'm trying to join 4 tables and one of the tables doesn't have all the matching ID's BUT I still need to show the results of the join and even for the rows that didn't have a corresponding ID.
Here's an example of what I'm talking about:
Example tables:
DECLARE #Table1 TABLE (id INT PRIMARY KEY CLUSTERED, ts DateTime, tbl2_id INT, price DECIMAL(4,2), tbl3_id INT, tbl4_id INT)
INSERT INTO #Table1 VALUES(1, '2013-07-25 09:30:00', 10, 10.25, 1);
INSERT INTO #Table1 VALUES(2, '2013-07-25 10:25:00', 20, 25.25, 1);
INSERT INTO #Table1 VALUES(3, '2013-07-25 11:45:00', 30, 30.15, 2);
INSERT INTO #Table1 VALUES(4, '2013-07-25 13:31:00', 40, 80.40, 2);
DECLARE #Table2 TABLE (id INT PRIMARY KEY CLUSTERED, symbol VARCHAR(25), tbl1_id int)
INSERT INTO #Table2 VALUES(10, 'XYZ', 1);
INSERT INTO #Table2 VALUES(20, 'ABC', 2);
INSERT INTO #Table2 VALUES(30, 'RST', 3);
INSERT INTO #Table2 VALUES(40, 'EFG', 4);
DECLARE #Table3 TABLE (id INT PRIMARY KEY CLUSTERED, exch VARCHAR(25))
INSERT INTO #Table3 VALUES(1, 'A');
INSERT INTO #Table3 VALUES(2, 'B');
INSERT INTO #Table3 VALUES(3, 'C');
INSERT INTO #Table3 VALUES(4, 'D');
DECLARE #Table4 TABLE (id INT PRIMARY KEY CLUSTERED, int tbl1_id, cnt INT)
INSERT INTO #Table4 VALUES(1, 2, 19);
INSERT INTO #Table4 VALUES(2, 4, 2013);
Example Query:
SELECT tbl1.id, tbl1.ts, tbl2.symbol, IFNULL(tbl3.cnt,0) AS cnt
FROM TABLE1 tbl1
JOIN TABLE2 tbl2
ON tbl1.tbl2_id = tbl2.id
JOIN TABLE3 tbl3
ON tbl3.id = tbl1.tbl3_id
LEFT OUTER JOIN TABLE4 tbl4
ON tbl1.tbl4_id = tbl4.id
WHERE tbl1.ts BETWEEN '2013-07-25 09:30:00 AND '2013-07-25 16:00:00'
AND tbl1.price >= 15.00
LIMIT 1000;
So basically what I'm trying to do is if tbl4 doesn't have a tbl1_id I'd still want to see the result from table1 BUT show a 0 value for Cnt...when i run this query I'm getting a bunch of duplicate entries and the data isn't looking right.
You are only returning the one with matches from the left hand side of the join. Change the join with table 4 to right join and it will return all the records from the right regardless. I know that was an example query that doesn't work with fake data. But I have to work with what you give me. This probably wont work through copy and paste, but the theory is correct, you just gotta modify it. If you give more detailed information I will tailor it to that.
SELECT tbl1.id, tbl1.ts, tbl2.symbol, IFNULL(tbl3.cnt,0) AS cnt
FROM TABLE1 tbl1
JOIN TABLE2 tbl2
ON tbl1.tbl2_id = tbl2.id
JOIN TABLE3 tbl3
ON tbl3.id = tbl1.tbl3_id
RIGHT JOIN TABLE4 tbl4
ON tbl1.tbl4_id = tbl4.id
WHERE tbl1.ts BETWEEN '2013-07-25 09:30:00 AND '2013-07-25 16:00:00'
AND tbl1.price >= 15.00
LIMIT 1000;
First of all In your sample code you gave a mix of SQL Server and MySql syntax:
DECLARE TABLE, CLUSTERED are SQL Server features
IFNULL(), LIMIT are MySql's
Secondly cnt column is in Table4 not in Table3 so IFNULL(tbl3.cnt,0) is not going to fly.
Thirdly your last edit when you changed tbl1.id = tbl4.tbl1_id to tbl1.tbl4_id = tbl4.id doesn't add up with the rest of your sample data since you don't have values in your insert statements either for tbl3_id or for newly introduced tbl4_id. And tbl1_id column now seems to be unnecessary.
Therefore IMHO you need to revisit and correct your sample data. And posting desired output might also help you to get your answer.
Now here is a query that works and based on your data schema before your last edit. It will successfully work both in MySql and SQL Server except for the LIMIT part
SELECT t1.id, t1.ts, t2.symbol, COALESCE(t4.cnt, 0) cnt
FROM Table1 t1 JOIN Table2 t2
ON t1.id = t2.tbl1_id JOIN Table3 t3
ON t1.tbl3_id = t3.id LEFT JOIN Table4 t4
ON t1.id = t4.tbl1_id
WHERE t1.ts BETWEEN '2013-07-25 09:30:00' AND '2013-07-25 16:00:00'
AND t1.price >= 15.00
LIMIT 1000 -- LIMIT will work only in MySql
Output:
| ID | TS | SYMBOL | CNT |
----------------------------------------------------
| 2 | July, 25 2013 10:25:00+0000 | ABC | 19 |
| 3 | July, 25 2013 11:45:00+0000 | RST | 0 |
| 4 | July, 25 2013 13:31:00+0000 | EFG | 2013 |
Here is SQLFiddle (MySql) demo
Here is SQLFiddle (SQL Server) demo