1

我正在尝试实现记录视图和点击的自制横幅系统。好吧,几乎一切正常,除了当我打开一个页面并从局部视图调用 RenderBanners() 操作时。它被称为如下:

_Layout.cshtml 通过调用 _Header.cshtml @Html.Partial("_Header"),然后在 _Header.cshtml 中我通过调用此操作@{ Html.RenderAction("RenderBanners", "Home"); }

渲染横幅操作:

        [AllowAnonymous]
        public ActionResult RenderBanners()
        {
            var banners = bannerTimeRepository.ForcefulGetAll().ToList();  

            try
            {
                foreach(var banner in banners)
                {
                    bannerLogRepository.SaveOrUpdate(new BannerLog
                    {
                        Timestamp = DateTime.Now,
                        Action = BannerAction.View,
                        User = membership.LoggedUser,
                        Banner = banner
                    });
                }
            }
            catch(Exception ex)
            {
                var tralala = ex;
            }

            return View(banners);

我得到这个 InvalidOperationException:

无法更改关系,因为一个或多个外键属性不可为空。当对关系进行更改时,相关的外键属性将设置为空值。如果外键不支持空值,则必须定义新关系,必须为外键属性分配另一个非空值,或者必须删除不相关的对象。

看来我的模型不正确,对吧?

好吧,如果我访问 ~/Home/RenderBannersurl,我会得到部分视图,所有内容都正确呈现并且所有内容都保存到数据库中。它也适用于这个动作:

[AllowAnonymous]
public ActionResult FollowBanner(int id)
{
    var banner = bannerRepository.ForcefulGetById(id);

    bannerLogRepository.SaveOrUpdate(new BannerLog
    {
        Timestamp = DateTime.Now,
        Action = BannerAction.Click,
        User = membership.LoggedUser,
        Banner = banner
    });

    return Redirect(banner.Href);
}

我的模型:

public abstract class Entity
{
    [Key]
    public int Id { get; protected set; }
}
public class Banner : Entity
{
    //omited
}

public class BannerLog : Entity
{
    [Key]
    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public new int Id { get; set; }

    public BannerAction Action { get; set; }

    public User User { get; set; }

    public DateTime Timestamp { get; set; }

    public Banner Banner { get; set; }
}

public enum BannerAction
{
    View = 0,
    Click = 1
}

模型构建器:

modelBuilder.Entity<BannerLog>().HasRequired(x => x.Banner).WithMany().WillCascadeOnDelete(true);
modelBuilder.Entity<BannerLog>().HasOptional(x => x.User).WithMany().WillCascadeOnDelete(false);

我相信我的模型和关系是正确的,因为它在大多数情况下都有效......除了通过部分视图调用 Action 时?我勒个去?

编辑:这个项目使用存储库模式,我们有一个接口 IRepository<T> 和 Repository<T> SaveOrUpdate 的实现是这样的:

        public virtual T SaveOrUpdate(T entity)
        {
            var casted = entity as Entity;
            if(casted.Id == 0)
                this.Context.Set<T>().Add(entity);

            this.Context.SaveChanges();

            return entity;
        }

bannerLogRepository 只是 Repository<BannerLog>。

4

1 回答 1

0

因为我仍然不知道到底发生了什么,所以我做了一个快速的破解——@{ Html.RenderAction("RenderBanners", "Home"); }我没有这样做,而是从 JavaScript 调用这个动作:

<div id="bannerDiv"></div>
<script>
    $("#bannerDiv").load('@Url.Action("RenderBanners", "Home")');
</script>

工作正常...

于 2013-07-17T13:16:09.243 回答