Форум программистов, компьютерный форум, киберфорум
PostgreSQL
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.80/15: Рейтинг темы: голосов - 15, средняя оценка - 4.80
 Аватар для Ника))
43 / 43 / 20
Регистрация: 08.12.2010
Сообщений: 342

PostgreSQL склад

28.10.2016, 14:45. Показов 3064. Ответов 2
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Учитывая рекомендации, перестроил базу данных (Вот небольшая часть, необходимая для интересующего меня вопроса):
SQL
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
/* Сам тип расхода */
CREATE TYPE  ComprositiontRashod AS (
    Name CHARACTER VARYING,
    COUNT NUMERIC
);
 
/* Таблица самого склада */
CREATE TABLE "Sklad" (
    "Name" CHARACTER VARYING NOT NULL,
    "Uint" CHARACTER VARYING NOT NULL,
    "Count" NUMERIC,
    "Changed" BOOLEAN, /* эти поля для 1с */
        "OneCID" CHARACTER VARYING, /* эти поля для 1с */
    CONSTRAINT Sklad_pk PRIMARY KEY ("Name")
) WITH (
  OIDS=FALSE
);
 
/* таблица для расхода */
CREATE TABLE "Rashod" (
    "ID" serial NOT NULL,
    "Date" TIMESTAMP WITHOUT TIME zone NOT NULL,
    "NameSklad" CHARACTER VARYING NOT NULL,
    "Composition" ComprositiontRashod[] NOT NULL,
    "Operator" CHARACTER VARYING NOT NULL,
    "Total" NUMERIC NOT NULL,
    "Changed" BOOLEAN,
        "OneCID" CHARACTER VARYING,
    CONSTRAINT Rashod_pk PRIMARY KEY ("ID")
) WITH (
  OIDS=FALSE
);
 
/* Для калькуляции */
CREATE TABLE "Calculation" (
    "Name" CHARACTER VARYING NOT NULL,
    "Gor_Obrabotka" NUMERIC ,
    "Hol_Obrabotka" NUMERIC ,
    "Exit_procent" NUMERIC,
    CONSTRAINT Calculation_pk PRIMARY KEY ("Name")
) WITH (
  OIDS=FALSE
);
 
/* Активные продукты на точке */
CREATE TABLE "ActivSklad" (
    "Name" CHARACTER VARYING NOT NULL,
    "Count" NUMERIC NOT NULL,
    CONSTRAINT ActivSklad_pk PRIMARY KEY ("Name")
) WITH (
  OIDS=FALSE
);

Собственно сейчас задумано следующее, на складе, есть какие то продукты, необходимо сформировать запись расхода. В отдельном интерфейсе выбирается количество и товары. Далее внутри самой программы (ПО на GOLANG), считываются цены, из отдельного микросервиса, по всем продуктам и сумма заносятся в поле "Total". После чего с помощью таблицы калькуляции вычисляется сколько продуктов после обработки должно получиться на точке.

После того как я вытащил цену, мне необходимо, произвести своего рода транзакцию, т.е. вычесть определенное количество продуктов со указанного склада ( Имя склада есть в функции обработчике в программе ).
Вопрос: Как будет правильно это осуществить?
1 вариант: В программе создать собственную эмуляцию транзакции, которая в случае чего вернет все на место. (Но по моему это будет просто дико нагружать сервер)
2 вариант: Сделать триггер с изменяемым названием склада (Пока даже не представляю, как это сделать)
3 вариант: Хранимая процедура, так же с изменяемым названием склада. (Коммент от второго)

2-3 варианты не рекомендую в интернете, так как оно вечно будет что то пересчитывать и постоянно через prepare.
Если приведете примеры для динамического имени склада для моего примера, буду благодарен.

Спасибо за внимание =)


Добавлено через 21 минуту
Можно сделать более лаконично?
А то к примеру в update через RETURNING у меня не работало =(
SQL
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
CREATE OR REPLACE FUNCTION rashod_trigger_func() RETURNS TRIGGER AS $TRIGGER$
DECLARE
    rec ComprositiontRashod;
    counts NUMERIC;
    allcount NUMERIC;
    SQL VARCHAR;
    sqlUpdate VARCHAR;
    sqlSelect VARCHAR;
    nameSklad VARCHAR;
 
BEGIN 
    nameSklad :=  (SELECT  "Rashod"."NameSklad" FROM "Rashod" WHERE NEW."ID" = "Rashod"."ID" );
    SQL :='select "Count" from ' || nameSklad || ' s Where s."Name" = $1';  
    sqlUpdate := 'UPDATE '|| nameSklad ||' SET "Count"= ($1 - $2)  WHERE '|| nameSklad ||'."Name" = $3 AND '|| nameSklad ||'."Count" >= $2';
    sqlSelect := 'select "Count" from '|| nameSklad ||' s Where s."Name" = $1';
 
    FOR rec.Count, rec.Name IN SELECT u.Count, u.Name FROM  unnest(NEW."Composition") u  GROUP BY u.Count, u.Name
    LOOP
      EXECUTE SQL INTO counts USING rec.Name;
      EXECUTE sqlUpdate USING counts, rec.Count, rec.Name;
      EXECUTE sqlSelect INTO allcount USING rec.Name;
      IF allcount = counts THEN
        RAISE EXCEPTION 'Error';
      END IF;
    END LOOP;
    RETURN NEW;
END;
$TRIGGER$
LANGUAGE  plpgsql;
 
CREATE TRIGGER rashod_trigger
AFTER INSERT OR UPDATE ON "Rashod" FOR EACH ROW 
EXECUTE PROCEDURE rashod_trigger_func();
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
28.10.2016, 14:45
Ответы с готовыми решениями:

PostgreSQL 9.4.10 1с
Крутится 15 БД 1с Винты баракуды в 10 Пользователей немного порядка 20 Базы занимают порядка 300 гб. Возможно ли разнести БД на...

PostgreSQL
Здраствуйте. Работаю на одной из фирм , програмистом. Тока начинают. Програмирую на Линуксе. Так вот дали задание сотворить програмку с...

мониторинг postgresql
привет. Есть проект. БД postgres. на linux, windows 7 и ниже работает норм ставлю Проект на windows 8, как то долго работают запросы к...

2
1264 / 978 / 384
Регистрация: 02.09.2012
Сообщений: 3,021
29.10.2016, 01:51
Цитата Сообщение от Ника)) Посмотреть сообщение
После того как я вытащил цену, мне необходимо, произвести своего рода транзакцию, т.е. вычесть определенное количество продуктов со указанного склада ( Имя склада есть в функции обработчике в программе ).
Вопрос: Как будет правильно это осуществить?
1 вариант: В программе создать собственную эмуляцию транзакции, которая в случае чего вернет все на место. (Но по моему это будет просто дико нагружать сервер)
Правильнее в любом случае транзакция (по сравнению с другими вариантами). Другое дело как вы ее реализуете.
Я вот читаю код (триггера) и не понимаю, что нельзя одним UPDATE'ом обойтись.
"UPDATE .... SET ... WHERE Name=(SELECT ....) AND .... RETURNING ..."
Вообще каждое выражение postgresql выполняет как бы в транзакции (хоть вы ее явно и не определяете). Конечно, это еще может зависеть от клиента, через которого вы передаете sql-команды.
1
 Аватар для Ника))
43 / 43 / 20
Регистрация: 08.12.2010
Сообщений: 342
29.10.2016, 09:15  [ТС]
"UPDATE .... SET ... WHERE Name=(SELECT ....) AND .... RETURNING ..."
В том то и дело, что returning не обрабатывал результат, реализовано было так:
SQL
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
CREATE OR REPLACE FUNCTION rashod_trigger_func() RETURNS TRIGGER AS $TRIGGER$
DECLARE
    rec ComprositiontRashod;
    counts NUMERIC;
    allcount NUMERIC;
    SQL VARCHAR;
    sqlUpdate VARCHAR;
    sqlSelect VARCHAR;
    nameSklad VARCHAR;
 
BEGIN 
    nameSklad :=  (SELECT  "Rashod"."NameSklad" FROM "Rashod" WHERE NEW."ID" = "Rashod"."ID" );
    SQL :='select "Count" from ' || nameSklad || ' s Where s."Name" = $1';  
    sqlUpdate := 'UPDATE '|| nameSklad ||' SET "Count"= ($1 - $2)  WHERE '|| nameSklad ||'."Name" = $3 AND '|| nameSklad ||'."Count" >= $2 RETURNING '|| nameSklad ||'."Count"';
    sqlSelect := 'select "Count" from '|| nameSklad ||' s Where s."Name" = $1';
 
    FOR rec.COUNT, rec.Name IN SELECT u.COUNT, u.Name FROM  unnest(NEW."Composition") u  GROUP BY u.COUNT, u.Name
    LOOP
      EXECUTE SQL INTO counts USING rec.Name;
      EXECUTE sqlUpdate  INTO allcount USING counts, rec.COUNT, rec.Name;
     /* EXECUTE sqlSelect INTO allcount USING rec.Name;*/
      IF allcount = counts THEN
        RAISE EXCEPTION 'Error';
      END IF;
    END LOOP;
    RETURN NEW;
END;
$TRIGGER$
LANGUAGE  plpgsql;
 
CREATE TRIGGER rashod_trigger
AFTER INSERT OR UPDATE ON "Rashod" FOR EACH ROW 
EXECUTE PROCEDURE rashod_trigger_func();
Может быть не много не точно, так как на машине которая сейчас под рукой нет данной DB. Но я думаю суть понятна.
Вот в таком виде функция не запихивала в allcount необходимое значение, хотя если просто в консоли pgadmin использовать данный запрос, то все хорошо. Собственно поэтому была реализована дополнительная функция.
А что касается внутреннего select, что то не подумал спасибо.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
29.10.2016, 09:15
Помогаю со студенческими работами здесь

PostgreSQL + MCBC
Доброго времени суток! Только начал разбираться с базами данных и столкнулся со следующей проблемой. При установке PostgreSQL на...

Обновление Postgresql 9.1 - 9.6
Добрый день! Прошу вашей помощи по обновлению версии Postgresql c версии 9.1 до версии 9.6 при помощи pg_upgrade Вдохновившись...

PostgreSQL функции
CREATE OR REPLACE FUNCTION fizzbuzz() RETURNS TEXT AS $$ DECLARE result TEXT; BEGIN FOR i IN 1..100 LOOP IF i % 3 = 0 THEN...

Pgpool и postgresql
Добрый день, должна использовать pgpool, но есть масса вопросов. ПОмогите пожалуйста. 1. Устанавила pgpool по yum install, и теперь когда...

GUID postgresql 9.3
Здравствуйте. Пытаюсь снести постгресс при помощи msiexec, он требует GUID (Уникальный номер продукта). Где его можно найти?


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
3
Ответ Создать тему
Новые блоги и статьи
Семь CDC на одном интерфейсе: 5 U[S]ARTов, 1 CAN и 1 SSI
Eddy_Em 18.02.2026
Постепенно допиливаю свою "многоинтерфейсную плату". Выглядит вот так: https:/ / www. cyberforum. ru/ blog_attachment. php?attachmentid=11617&stc=1&d=1771445347 Основана на STM32F303RBT6. На борту пять. . .
Символьное дифференцирование
igorrr37 13.02.2026
/ * Программа принимает математическое выражение в виде строки и выдаёт его производную в виде строки и вычисляет значение производной при заданном х Логарифм записывается как: (x-2)log(x^2+2) -. . .
Камера Toupcam IUA500KMA
Eddy_Em 12.02.2026
Т. к. у всяких "хикроботов" слишком уж мелкий пиксель, для подсмотра в ESPriF они вообще плохо годятся: уже 14 величину можно рассмотреть еле-еле лишь на экспозициях под 3 секунды (а то и больше),. . .
И ясному Солнцу
zbw 12.02.2026
И ясному Солнцу, и светлой Луне. В мире покоя нет и люди не могут жить в тишине. А жить им немного лет.
«Знание-Сила»
zbw 12.02.2026
«Знание-Сила» «Время-Деньги» «Деньги -Пуля»
SDL3 для Web (WebAssembly): Подключение Box2D v3, физика и отрисовка коллайдеров
8Observer8 12.02.2026
Содержание блога Box2D - это библиотека для 2D физики для анимаций и игр. С её помощью можно определять были ли коллизии между конкретными объектами и вызывать обработчики событий столкновения. . . .
SDL3 для Web (WebAssembly): Загрузка PNG с прозрачным фоном с помощью SDL_LoadPNG (без SDL3_image)
8Observer8 11.02.2026
Содержание блога Библиотека SDL3 содержит встроенные инструменты для базовой работы с изображениями - без использования библиотеки SDL3_image. Пошагово создадим проект для загрузки изображения. . .
SDL3 для Web (WebAssembly): Загрузка PNG с прозрачным фоном с помощью SDL3_image
8Observer8 10.02.2026
Содержание блога Библиотека SDL3_image содержит инструменты для расширенной работы с изображениями. Пошагово создадим проект для загрузки изображения формата PNG с альфа-каналом (с прозрачным. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru