[C#]關於DataDirectory的一些思考

来源:https://www.cnblogs.com/cncc/archive/2018/08/22/9515991.html
-Advertisement-
Play Games

筆者在使用Entity Framework中的Scaffolding機制自動創建拓展名為mdf的資料庫及表單時,遇到如下的錯誤: 首先回顧一下創建這個程式的步驟: 1、創建一個Console控制台應用程式,程式集名稱及命名空間為ConsoleApp; 2、使用程式包控制台管理器將Entity Fra ...


筆者在使用Entity Framework中的Scaffolding機制自動創建拓展名為mdf的資料庫及表單時,遇到如下的錯誤:

A file activation error occurred. 
The physical file name '\\MusicDBContext.mdf' may be incorrect. 
Diagnose and correct additional errors, and retry the operation.
CREATE DATABASE failed. Some file names listed could not be created. 
Check related errors.

 

首先回顧一下創建這個程式的步驟:

1、創建一個Console控制台應用程式,程式集名稱及命名空間為ConsoleApp;

2、使用程式包控制台管理器將Entity Framework包含到此程式中,代碼如下:

PM> install-package Entity Framework

3、在App.Config文件中將以下內容插入到configuration節點:

<connectionStrings>
    <add name="MusicDBContext"
       connectionString="Data Source=(LocalDb)\MSSQLLocalDB;
         Initial Catalog=MusicDBContext;Integrated Security=SSPI;
         AttachDBFilename=|DataDirectory|\MusicDBContext.mdf"
       providerName="System.Data.SqlClient" />
</connectionStrings> 

4、在控制台編寫以下代碼:

using System;
using System.Linq;
using System.Data.Entity;
namespace ConsoleApp
{
    class Program
    {
        static void Main(string[] args)
        {
            try
            {
                MusicDbContext db = new MusicDbContext();
                Music music = new Music { Title = "Far Away From Home", 
                                          ReleaseDate = DateTime.Now };
                db.Musics.Add(music);
                db.SaveChanges();
                db.Musics.ToList().ForEach(x => Console.WriteLine($"{x.ID},
                                                {x.Title},{x.ReleaseDate}"));
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
                if(ex.InnerException != null)
                {
                    Console.WriteLine(ex.InnerException.Message);
                }
            }
            Console.ReadKey();
        }
    }
    public class Music
    {
        public int ID { get; set; }
        public string Title { get; set; }
        public DateTime ReleaseDate { set; get; }
    }
    public class MusicDbContext : DbContext
    {
        public MusicDbContext() : base("MusicDBContext") { }
        public DbSet<Music> Musics { set; get; }
    }
}

5、運行此程式,發現程式不能按自己想要的結果運行,出現在最前面出現的錯誤。


通過查看出錯的信息,發現

AttachDBFilename=|DataDirectory|\MusicDBContext.mdf

有問題,而這又是沒有問題的,這到底是怎麼回事?為什麼會出現錯誤?

於是,通過MSDN查找相關資料,通過以下方法獲得DataDirectory指定的路徑是什麼:

object path = AppDomain.CurrentDomain.GetData("DataDirectory");

運行此行代碼,發現path居然是null!!!什麼?一般控制台或者Windows Form程式根據是Debug還是Release決定DataDirectory的初始化路徑為Bebug文件夾還是Release文件夾嗎?

這個錯了。

如果原先的Bebug文件夾或Release文件夾存在資料庫文件,使用類似"AttachDBFilename=|DataDirectory|\MusicDBContext.mdf"的寫法是沒有問題的,

即使path = null,它也知道是在Bebug文件夾或Release文件夾下。

如果原先的Bebug文件夾或Release文件夾不存在資料庫文件,上面的寫法就有問題,也就會出現最開始出現的那種錯誤。

那麼,我們該如何解決呢?細心的人可以發現,既然可以使用AppDomain.CurrentDomain.GetData來獲得DataDirectory指定的路徑,

那及可以使用AppDomain.CurrentDomain.SetData來指定DataDirectory的初始化路徑,代碼如下:

AppDomain.CurrentDomain.SetData("DataDirectory", Environment.CurrentDirectory);

通過以上的方法,就可以解決最開始前面的問題。


 

通過以上的介紹,最終的代碼修改如下:

using System;
using System.Linq;
using System.IO;
using System.Data.Entity;

namespace ConsoleApp
{
    class Program
    {
        static void Main(string[] args)
        {
            string dbPath = Environment.CurrentDirectory + @"\MusicDBContext.mdf";
            if(!File.Exists(dbPath))
            {
                AppDomain.CurrentDomain.SetData("DataDirectory", Environment.CurrentDirectory);
            }
            try
            {
                MusicDbContext db = new MusicDbContext();
                Music music = new Music { Title = "Far Away From Home", ReleaseDate = DateTime.Now };
                db.Musics.Add(music);
                db.SaveChanges();
                db.Musics.ToList().ForEach(x => Console.WriteLine($"{x.ID},{x.Title},{x.ReleaseDate}"));
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
                if(ex.InnerException != null)
                {
                    Console.WriteLine(ex.InnerException.Message);
                }
            }
            Console.ReadKey();
        }
    }

    public class Music
    {
        public int ID { get; set; }
        public string Title { get; set; }
        public DateTime ReleaseDate { set; get; }

    }

    public class MusicDbContext : DbContext
    {
        public MusicDbContext() : base("MusicDBContext") { }
        public DbSet<Music> Musics { set; get; }
    }
}

程式就可以正常運行了。


 

註:

1)AttachDBFilename=|DataDirectory|\MusicDBContext.mdf 

     其中的“\”可以省略掉,即為:AttachDBFilename=|DataDirectory|MusicDBContext.mdf 

2)如果是ASP.NET程式,DataDirectory的初始化目錄為App_Data。

3)關於更多的|DataDirectory|知識,請參考如下:

     https://docs.microsoft.com/en-us/dotnet/framework/data/adonet/ef/connection-strings

     https://stackoverflow.com/questions/1409358/ado-net-datadirectory-where-is-this-documented/1409378#1409378

     https://stackoverflow.com/questions/51948028/a-file-activation-error-occurred-when-using-entity-framework

   

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 真覺得自己的知識面還是比較窄,在此之前,居然還不知道SqlLocalDb。 SqlLocalDb是啥?其實就是簡化SQL Server的本地資料庫,可以這樣子說,SQL Server既可以作為遠程,也可以做本地, 而SqlLocalDb只能作為本地使用。說的直接一點,SqlLocalDb就是一個輕量 ...
  • 項目屬性-->生成-->允許不安全代碼勾上。 代碼: /// <summary> /// 反色處理 /// </summary> private Bitmap Inverse(Bitmap bmp) { BitmapData srcdat = bmp.LockBits(new Rectangle(P ...
  • 同大陸身份證驗證一樣,該部分是按照國家增值稅發票代碼的定製規則,進行發票代碼驗證,如果需要查驗發票信息是否正確,應該通過第三方介面(大約一毛錢查驗一次),或者直接上 "國家稅務總局全國增值稅發票查驗平臺" 進行查驗。 目前能識別的增值稅發票代碼包含以下幾類: 增值稅專用發票 , 增值稅普通發票(紙質 ...
  • 上一篇文章中介紹了關於設置Excel條件格式,包括基於單元格值、自定義公式等應用條件格式、應用數據條條件類型格式、刪除條件格式等內容。在本篇文章中將繼續介紹C# 設置條件格式的方法。 要點概述: 1. 應用條件格式用於高亮重覆、唯一數值 2. 應用條件格式用於高亮峰值(最高、最低) 3. 應用條件格 ...
  • 在本教程中,我們來學習如何進行分組。 在 書籍管理系統的“關於”頁上,將顯示每個出版社有多少書籍。 這要求在分組上再進行分組和簡單計算。 要完成此操作,需要執行以下操作: 創建一個視圖模型類,該視圖類是需要傳遞到該視圖的數據的抽象。 修改關於視圖。 ...
  • C#/WPF/WinForm/.NET程式代碼實現軟體程式開機自動啟動的兩種常用方法函數的示例與實例帶詳細註釋 方法一:將軟體的快捷方式創建到電腦的自動啟動目錄下(不需要管理員許可權) 1.必要引用 2.代碼實現-只需要調用SetMeAutoStart(bool onOff)方法就可以了,參數onO ...
  • 作者:暴王 個人博客:http://www.boydwang.com/2017/12/net core in memory cache/ 這兩天在看.net core的in memory cache,這裡記錄一下用法,主要涉及MemoryCache的Get/Set/Expire/Flush。 首先我 ...
  • All在項目文件裡面,這一行代表引用包,可是有三個子項目,需要說一下:IncludeAssetsExcludeAssetsPrivateAssets這三個是什麼意思?中文翻譯,Assets是資產,反正無法理解說的是什麼鬼。一開始我以為它可以定製引用庫裡面那個類,搞了好久才發覺這是一個部署選項。可以取... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...