Реализация отношения «многие ко многим» в Asp. Net (в замешательстве?) - PullRequest
0 голосов
/ 15 апреля 2020

Моя текущая цель - построить структуру базы данных с использованием классов в Entity Framework & ASP MVC.

У меня в настоящее время есть таблица Users и таблица Posts. То, что я хотел бы сделать, это создать отношения «многие ко многим» для пользователей, которым понравились сообщения (при сохранении того, кто создал сообщение). И иметь возможность доступа для каждого пользователя ко всем понравившимся им сообщениям. В настоящее время у меня есть эти классы, но я не уверен, как их связать, так как все онлайн-примеры связывают первичные ключи из разных баз данных, где я просто хочу использовать параметр имени пользователя. Любая помощь будет отличной. У меня есть это до сих пор.

 public class Posts
    {
        [Key]
        public virtual int PostId { get; set; }
        public virtual string Title { get; set; }
        public virtual string URL { get; set; }
        [DisplayName("Main Text")]
        public virtual string TextBody { get; set; }
        public int PostLikes { get; set; }
        private DateTime Datedata = DateTime.Now;
        public DateTime PostDate { get { return Datedata; } set { Datedata = value; } }
        public virtual Users User { get; set; }
        public ICollection<PostLikes> UsersWhoHaveSigned { get; set; }
    }
{
    public class Users
    {
        [Key]
        public virtual int UserId { get; set; }
        public virtual string Username { get; set; }
        public virtual string FirstName { get; set; }
        public virtual string LastName { get; set; }
        public virtual List<Posts> Post { get; set; }

    }
}

Я еще не создал таблицу UsersWhoHaveSigned. Ранние эксперименты заставили меня вернуться, это было больно. Любая помощь была бы полезна.

Редактировать: Я надеялся попросить о помощи, а затем присвоить эту информацию в соответствии с моим примером, который использует дополнение для отдельных учетных записей. Это создает некоторые дополнительные файлы, которые теперь создают помехи предоставленному вами коду. Вот файл IdentityModels.cs.

using System.Data.Entity;
using System.Security.Claims;
using System.Threading.Tasks;
using Microsoft.AspNet.Identity;
using Microsoft.AspNet.Identity.EntityFramework;

namespace Coursework2.Models
{
    // You can add profile data for the user by adding more properties to your ApplicationUser class, please visit https://go.microsoft.com/fwlink/?LinkID=317594 to learn more.
    public class ApplicationUser : IdentityUser
    {
        public async Task<ClaimsIdentity> GenerateUserIdentityAsync(UserManager<ApplicationUser> manager)
        {
            // Note the authenticationType must match the one defined in CookieAuthenticationOptions.AuthenticationType
            var userIdentity = await manager.CreateIdentityAsync(this, DefaultAuthenticationTypes.ApplicationCookie);
            // Add custom user claims here
            return userIdentity;
        }
    }

    public class ApplicationDbContext : IdentityDbContext<ApplicationUser>
    {
        public ApplicationDbContext()
            : base("DefaultConnection", throwIfV1Schema: false)
        {
        }

        public static ApplicationDbContext Create()
        {
            return new ApplicationDbContext();
        }
    }
}

Я считаю, что директивы сборки в верхней части не позволяют использовать system.data.entity, поэтому при попытке реализовать ApplicationDbContext: DbContext я получаю сообщения об ошибках: /

В идеале я собираюсь использовать файл IdentityModels.cs в качестве замены для класса пользователей. Но все еще очень потерян.

Ответы [ 2 ]

1 голос
/ 15 апреля 2020

Прежде всего, я рекомендую использовать форму единственного числа для имен классов, так как EF автоматически приумножит имена таблиц.

Во-вторых, для ключевого свойства вы можете просто использовать термин Id, без любые аннотации, и EF выберет его в качестве основного ключа.

Наконец, я предполагаю, что вы хотите использовать подход Code-First. Рассмотрим следующие классы (ваши, но для ясности измененные):

public class Post
{

    public virtual Guid Id { get; set; }

    public virtual string UserName { get; set; }

    public virtual User User { get; set; }

    public virtual ICollection<PostLike> Likes { get; set; }

}

public class PostLike
{

    public virtual Guid Id { get; set; }

    public virtual Guid PostId { get; set; }

    public virtual Post Post { get; set; }

    public virtual string UserName { get; set; }

    public virtual User User { get; set; }

}

public class User
{

    public virtual Guid Id { get; set; }

    public virtual string UserName { get; set; }

    public virtual ICollection<Post> Posts { get; set; }

    public virtual ICollection<PostLike> Likes { get; set; }

}

Чтобы это работало, вам понадобится DbContext, например, следующий. Обратите внимание на метод OnModelCreating, где волхвы c происходят:

public class ApplicationDbContext
    : DbContext
{

    public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options) 
        : base(options)
    {

    }

    public DbSet<Post> Posts { get; set; }

    public DbSet<PostLike> PostLikes { get; set; }

    public DbSet<User> Users { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        base.OnModelCreating(modelBuilder);
        modelBuilder.Entity<User>()
            .HasAlternateKey(u => u.UserName);
        modelBuilder.Entity<User>()
            .HasMany(u => u.Posts)
            .WithOne(p => p.User);
        modelBuilder.Entity<Post>()
            .HasOne(p => p.User)
            .WithMany(u => u.Posts)
            .HasForeignKey(p => p.UserName)
            .HasPrincipalKey(u => u.UserName);
        modelBuilder.Entity<Post>()
            .HasMany(p => p.Likes)
            .WithOne(pl => pl.Post);
        modelBuilder.Entity<PostLike>()
            .HasOne(pl => pl.Post)
            .WithMany(p => p.Likes);
        modelBuilder.Entity<PostLike>()
            .HasOne(pl => pl.User)
            .WithMany(u => u.Likes)
            .HasForeignKey(pl => pl.UserName)
            .HasPrincipalKey(u => u.UserName);
    }

}

Вуаля! Я надеюсь, что он отвечает на ваш вопрос;) Если это так, пожалуйста, не забудьте пометить мой пост как ответ!

Редактировать:

Я приведу некоторые объяснения , что я упустил, чтобы ответить на ваш вопрос как можно скорее.

Итак, первое, что вам нужно сделать, это объявить имя пользователя в качестве альтернативного ключа, потому что вы хотите создать отношения в зависимости от него, и вы уже объявить основной ключ 'Id'.

Во-вторых, для каждого объекта, которому должна принадлежать база ссылок пользователей на альтернативном ключе UserName, необходимо объявить свойство UserName объекта как внешний ключ отношения, и свойство UserName пользователя в качестве основного ключа.

Другими словами, внешний ключ - это свойство, которое ссылающийся объект использует для отношения, и главный ключ это свойство, на основе которого указанный объект связан с ссылающимся объектом.

Обратите внимание, что главные ключи должны иметь ключ или * 103 3 * ограничение альтернативного ключа , иначе оно не будет работать.

0 голосов
/ 16 апреля 2020

Просто чтобы прокомментировать ваш ответ. Я обнаружил, что мне пришлось использовать Microsoft.EntityFrameworkCore и удалить System.Data.Entity - это вызывало путаницу в программе относительно того, какой DbContext я хотел использовать. Спасибо!

...