powered by simpleCommunicator - 2.0.50     © 2025 Programmizd 02
Форумы / ADO.NET, LINQ, Entity Framework, NHibernate, DAL, ORM [игнор отключен] [закрыт для гостей] / Двойная связь
13 сообщений из 13, страница 1 из 1
Двойная связь
    #38607374
__Pavel__
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Доброго времени суток!

Пишу диплом по проектированию АИС...
Для создания базы данных использую EF Code-first.

В базе есть две сущности: Пользователь и Задание
Пользователь может создать сколько угодно Заданий, а в задании может быть сколько угодно исполнителей. Для этого должно быть 2 разные связи между таблицами, чтобы можно было определять кто создал задание и кто должен его исполнять.

Проблема возникла, когда я попытался их связать, вот код классов:
Код: c#
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.
53.
54.
55.
56.
57.
58.
59.
60.
61.
62.
63.
64.
65.
66.
67.
68.
    [Table("Users")]
    public class User
    {
        public User()
        {
            Users = new List<User> { };
        }

        [Key]
        public int UserID { get; set; }
        [Required, MaxLength(15)]
        public string Login { get; set; }
        [Required, MaxLength(32)]
        public string Password { get; set; }
        [Required, MaxLength(1000)]
        public string Name { get; set; }
        [MaxLength(10)]
        public string sex { get; set; }
        public DateTime DateOfBirth { get; set; }

        //связи
        public int GroupID { get; set; }
        public virtual Group Group { get; set; }

        public int TypeUserID { get; set; }
        public virtual TypeUser TypeUser { get; set; }

        public int PostUserID { get; set; }
        public virtual PostUser PostUser { get; set; }

        public int StatusUserID { get; set; }
        public virtual StatusUser StatusUser { get; set; }

        public virtual List<User> Users { get; set; }
        public virtual List<Document> Document { get; set; }
        public virtual List<Problem> Problem { get; set; }
    }

    [Table("Problems")]
    public class Problem
    {
        public Problem()
        {
            Users = new List<User> { };
            Documents = new List<Document> { };
            Files = new List<File> { };
        }

        [Key]
        public int ProblemID { get; set; }
        [Required, MaxLength(150)]
        public string Name { get; set; }
        [MaxLength(2500)]
        public string Description { get; set; }
        public DateTime DateStart { get; set; }
        public DateTime DateFinish { get; set; }

        //связи
        public int TypeProblemID { get; set; }
        public virtual TypeProblem TypeProblem { get; set; }

        public int StatusProblemID { get; set; }
        public virtual StatusProblem StatusProblem{ get; set; }

        public virtual List<User> Users { get; set; }
        public virtual List<Document> Documents { get; set; }
        public virtual List<File> Files { get; set; }
    }



Естественно при этом EF создал 1 связь "Многие ко многим", а мне нужно 2 разные связи! Подскажите как это сделать?

Заранее спасибо!
...
Рейтинг: 0 / 0
Двойная связь
    #38607626
__Pavel__
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Чтобы было понятнее, прилагаю файл с концептуалдьной схемой базы данных.

Повторюсь, проблема со связями между сущностями Пользователь и Задание.
...
Рейтинг: 0 / 0
Двойная связь
    #38607651
Lelouch
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
__Pavel__,

Введите по 2 свойсва:

например такие.
public virtual List<User> CreateUsers { get; set; }
public virtual List<User> ExecuteUsers { get; set; }

И, соответсвенно, 2 поля для задач в User.
Правда не знаю, как их замапить между собой с помощью атрибутов. Через Fluent Mapping можно
...
Рейтинг: 0 / 0
Двойная связь
    #38607661
Lelouch
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
__Pavel__,

И кстати, как я понял, автор то у задачи 1. Так что вам нужна 1 связь "один ко многим" для автора задачи и одна "многие ко многим" для исполнителей?
...
Рейтинг: 0 / 0
Двойная связь
    #38607667
__Pavel__
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Lelouch,

да, получается в схеме ошибка
...
Рейтинг: 0 / 0
Двойная связь
    #38608431
__Pavel__
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Lelouch,

Да, вы правы, можно через Fluent API. Я указал явно все сложные связи, но возникла проблема в другом...

Вот код указания связей:
Код: c#
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.
protected override void OnModelCreating(DbModelBuilder modelBuilder)
        {
            modelBuilder.Entity<User>()
                .HasMany<Problem>(t => t.CreatesProblems)
                .WithRequired(t => t.User)
                .HasForeignKey(t => t.UserID)
                .WillCascadeOnDelete(true); //1 to many

            modelBuilder.Entity<User>()
                .HasMany<Document>(t => t.Documents)
                .WithRequired(t => t.User)
                .HasForeignKey(t => t.UserID)
                .WillCascadeOnDelete(true); //1 to many

            modelBuilder.Entity<User>()
                .HasMany<User>(t => t.Users)
                .WithOptional(t => t.Boss)
                .HasForeignKey(t => t.BossID)
                .WillCascadeOnDelete(false); //0-1 to many

            modelBuilder.Entity<User>()
                .HasMany(t => t.Problems)
                .WithMany(t => t.Users); //many to many

            modelBuilder.Entity<Problem>()
                .HasMany(t => t.Documents)
                .WithMany(t => t.Problems); //many to many

            modelBuilder.Entity<Contractor>()
                .HasMany<Document>(t => t.Documents)
                .WithRequired(t => t.Contractor1)
                .HasForeignKey(t => t.Contractor1ID)
                .WillCascadeOnDelete(false); //1 to many

            modelBuilder.Entity<Contractor>()
                .HasMany<Document>(t => t.Documents)
                .WithRequired(t => t.Contractor2)
                .HasForeignKey(t => t.Contractor2ID)
                .WillCascadeOnDelete(false); //1 to many
        }



Вот на всякий случай коды сущностей:
Код: c#
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.
53.
54.
55.
56.
57.
58.
59.
60.
61.
62.
63.
64.
65.
66.
67.
68.
69.
70.
71.
72.
73.
74.
75.
76.
77.
78.
79.
80.
81.
82.
83.
84.
85.
86.
87.
88.
89.
90.
91.
92.
93.
94.
95.
96.
97.
98.
99.
100.
101.
102.
103.
104.
105.
106.
107.
108.
109.
110.
111.
112.
113.
114.
115.
116.
117.
118.
119.
120.
121.
122.
123.
124.
    [Table("Users")]
    public class User
    {
        public User()
        {
            Users = new List<User> { };
            Documents = new List<Document> { };
            CreatesProblems = new List<Problem> { };
            Problems = new List<Problem> { };
            Dialogs = new List<Dialog> { };
        }

        [Key]
        public int UserID { get; set; }
        [Required, MaxLength(15)]
        public string Login { get; set; }
        [Required, MaxLength(32)]
        public string Password { get; set; }
        [Required, MaxLength(1000)]
        public string Name { get; set; }
        [MaxLength(10)]
        public string sex { get; set; }
        public DateTime DateOfBirth { get; set; }

        //связи
        public int GroupID { get; set; }
        public virtual Group Group { get; set; }

        public int TypeUserID { get; set; }
        public virtual TypeUser TypeUser { get; set; }

        public int PostUserID { get; set; }
        public virtual PostUser PostUser { get; set; }

        public int StatusUserID { get; set; }
        public virtual StatusUser StatusUser { get; set; }

        public int? BossID { get; set; }
        public virtual User Boss { get; set; }

        public virtual List<User> Users { get; set; }
        public virtual List<Document> Documents { get; set; }
        public virtual List<Problem> CreatesProblems { get; set; }
        public virtual List<Problem> Problems { get; set; }
        public virtual List<Dialog> Dialogs { get; set; }
    }

    [Table("Problems")]
    public class Problem
    {
        public Problem()
        {
            Users = new List<User> { };
            Documents = new List<Document> { };
            Files = new List<File> { };
        }

        [Key]
        public int ProblemID { get; set; }
        [Required, MaxLength(150)]
        public string Name { get; set; }
        [MaxLength(2500)]
        public string Description { get; set; }
        public DateTime DateStart { get; set; }
        public DateTime DateFinish { get; set; }

        //связи
        public int TypeProblemID { get; set; }
        public virtual TypeProblem TypeProblem { get; set; }

        public int StatusProblemID { get; set; }
        public virtual StatusProblem StatusProblem{ get; set; }

        public int UserID { get; set; }
        public virtual User User { get; set; }

        public virtual List<User> Users { get; set; }
        public virtual List<Document> Documents { get; set; }
        public virtual List<File> Files { get; set; }
    }

    [Table("Documents")]
    public class Document
    {
        public Document()
        {
            Problems = new List<Problem> { };
            Files = new List<File> { };
        }

        [Key]
        public int DocumentID { get; set; }
        [Required, MaxLength(150)]
        public string Name { get; set; }
        [MaxLength(2500)]
        public string Description { get; set; }
        public DateTime DateCreate { get; set; }
        public DateTime DateChange { get; set; }
        public int Number { get; set; }
        [MaxLength(2500)]
        public string Object { get; set; }
        public DateTime DateStart { get; set; }
        public DateTime DateFinish { get; set; }
        public int Price { get; set; }

        //связи
        public int TypeDocumentID { get; set; }
        public virtual TypeDocument TypeDocument { get; set; } 

        public int StatusDocumentID { get; set; }
        public virtual StatusDocument StatusDocument { get; set; }

        public int UserID { get; set; }
        public virtual User User { get; set; }

        public int Contractor1ID { get; set; }
        public virtual Contractor Contractor1 { get; set; }

        public int Contractor2ID { get; set; }
        public virtual Contractor Contractor2 { get; set; }

        public virtual List<Problem> Problems { get; set; }
        public virtual List<File> Files { get; set; }
    }



При создании базы вылетает ошибка
ОшибкаIntroducing FOREIGN KEY constraint 'FK_dbo.ProblemDocuments_dbo.Documents_Document_DocumentID' on table 'ProblemDocuments' may cause cycles or multiple cascade paths.
Specify ON DELETE NO ACTION or ON UPDATE NO ACTION, or modify other FOREIGN KEY constraints.

Вот прикладываю новую схему и на ней видно, что цикличности не может быть... при удалении Пользователя , сотрутся и Задачи назначенные им и созданные Документы , а он ругается на каскадное удаление между Документами и Задачами ... я что-то не понимаю?
...
Рейтинг: 0 / 0
Двойная связь
    #38608433
__Pavel__
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Забыл приложить схему...
...
Рейтинг: 0 / 0
Двойная связь
    #38608484
Lelouch
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
__Pavel__,

Уберите WillCascadeOnDelete(true);

У вас сейчас удаление пользователя может привести к "cycles or multiple cascade paths."
...
Рейтинг: 0 / 0
Двойная связь
    #38608510
__Pavel__
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Lelouch,

но почему!? так ведь и должно быть, нет? удаляем пользователя и удаляются все задачи и документы... это логично или я не прав?

и если убрать WillCascadeOnDelete(true); , то ошибка остается, нужно явно указывать false как параметр.
...
Рейтинг: 0 / 0
Двойная связь
    #38608723
Lelouch
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
__Pavel__,

"удаляем пользователя и удаляются все задачи и документы"
Удаляем пользователя -> Удаляются привязки пользователей к задачам и задачи -> При удалении задач удаляются привязки пользователей к задачам

Это одна и та же сущность (таблица). Вот он и ругается.
...
Рейтинг: 0 / 0
Двойная связь
    #38611160
ViPRos
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Lelouch,

мозгов не хватает, вт и ругается нет там никакого цикла на экземплярном уровне
...
Рейтинг: 0 / 0
Двойная связь
    #38611275
Lelouch
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
ViPRos,

а как можно при создании таблицы гарантировать отсутствие такого цикла в данных? с точки зрения задачи ТС не сказано, что нельзя самому себе поставить задачу.
...
Рейтинг: 0 / 0
Двойная связь
    #38611760
ViPRos
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
Участник
Lelouch,

ну че посавил да поставил
сообщиь то можно, а вот обявить ошибку - нет
...
Рейтинг: 0 / 0
13 сообщений из 13, страница 1 из 1
Форумы / ADO.NET, LINQ, Entity Framework, NHibernate, DAL, ORM [игнор отключен] [закрыт для гостей] / Двойная связь
Целевая тема:
Создать новую тему:
Автор:
Закрыть
Цитировать
Найденые пользователи ...
Разблокировать пользователей ...
Читали форум (0):
Пользователи онлайн (0):
x
x
Закрыть


Просмотр
0 / 0
Close
Debug Console [Select Text]