OpenFileDialog 无法加载 CSV 文件但可以加载 xls/xlsx Excel 个文件

OpenFileDialog can't load CSV files but can load xls/xlsx Excel files

在我的 Windows 表单应用程序中加载 Excel 文件时,我可以加载 .xls.xlsx 格式,但是当我 select 一个 .CSV 我收到以下错误:

System.NullReferenceException: 'Object reference not set to an instance of an object.' sConnectionString was null.

错误发生在行:

if (sConnectionString.Length > 0)

来自完整的代码部分:

public string sConnectionString;
public void FillData()
{
    if (sConnectionString.Length > 0)
    {
        OleDbConnection cn = new OleDbConnection(sConnectionString);
        {
            cn.Open();
            DataTable dt = new DataTable();
            OleDbDataAdapter Adpt = new OleDbDataAdapter("select * from [sheet1$]", cn);
            Adpt.Fill(dt);
            dataGridView1.DataSource = dt;
        }
    }
}

在按钮代码之前:

private void Browse_Click(object sender, EventArgs e)
{
    OpenFileDialog op = new OpenFileDialog();
    op.InitialDirectory = @"C:\";
    op.Title = "Browse Excel Files";
    op.CheckFileExists = true;
    op.CheckPathExists = true;
    op.DefaultExt = "csv";
    op.Filter = "CSV Files (*.csv)|*.csv";
    op.FilterIndex = 2;
    op.RestoreDirectory = true;
    op.ReadOnlyChecked = true;
    op.ShowReadOnly = true;

    if (op.ShowDialog() == System.Windows.Forms.DialogResult.OK)
    {
        if (File.Exists(op.FileName))
        {
            string[] Arr = null;
            Arr = op.FileName.Split('.');
            if (Arr.Length > 0)
            {
                if (Arr[Arr.Length - 1] == "xls")
                    sConnectionString = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" +
                    op.FileName + ";Extended Properties='Excel 8.0;HDR=Yes;IMEX=1'";
            }
            else if (Arr[Arr.Length - 1] == "xlsx")
            {
                sConnectionString = "Provider=Microsoft.ACE.OLEDB.12.0;Data Source=" + op.FileName + ";Extended Properties='Excel 12.0 Xml;HDR=YES';";
            }
        }
        FillData();
        fileTextBox.Text = op.FileName;
    }
}

编辑

已添加:

else if (Arr[Arr.Length - 1] == "csv")
    {
    sConnectionString = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + op.FileName + 
                        ";Extended Properties='Excel 8.0;HDR=Yes;IMEX=1'";
    }

仍然得到同样的错误。

关于报错:

System.NullReferenceException: Object reference not set to an instance of an object. sConnectionString was null.

异常产生是因为Connection字符串声明为:

public string sConnectionString;

因为它从未初始化,因为连接字符串的初始化只针对某些文件类型执行,而不是 OpenFileDialog.Filter 中包含的所有文件类型。当代码测试字符串长度时,字符串仍然是null。这样可以避免设置初始值:

public string sConnectionString = string.Empty;

关于使用 OleDbConnection 操作 .CSV 文件所需的连接字符串:

  • 所有 OleDb 提供程序都可以:
    • Microsoft.Jet.OLEDB.4.0
    • Microsoft.ACE.OLEDB.12.0
    • Microsoft.ACE.OLEDB.16.0
  • 如果某些遗留格式(旧 Access .mdb 文件)需要 Microsoft.Jet.OLEDB.4.0,则应用程序必须编译为 32 位,因此安装其他提供程序的相应 32 位版本:

Microsoft Database Engine 2010 Redistributable
Microsoft Database Engine 2016 Redistributable

要读取 CSV 文件,所有提供程序的连接字符串组成如下:

{Provider};Data Source={Catalog}; Extended Properties="text; HDR=Yes; IMEX=1; FMT=Delimited;

其中:

  • {Provider} => OleDb 提供程序之一。他们中的任何一个都可以。

  • {Catalog} => 包含要打开的文件的目录。

  • HDR=Yes/No => CSV 文件包含一个 Header:如果 Yes,则 Header 是文件的第一行

  • IMEX=1 => Import/Export 模式设置为 1(导出模式 = 0;导入模式 = 1,链接模式 = 2) , 忽略数值并仅使用字符串。这里实际上不相关。最好保留它,作为一般帮助(以防文件中没有 Header 和 HDR=Yes)。

  • FMT=Delimited => 文件格式:定界。 Header/Fields 由定界符分隔。公认的分隔符是逗号 (,)。此设置可能是 System-dependant(第 3 部分应用程序可能出于自身原因修改了注册表)。要指定不同于默认的分隔符(CSV中的C表示逗号),Catalog[=105中必须有一个Schema.ini文件=] 为特定文件定义特定分隔符的文件夹:

      [MyFile.csv]
      Format=Delimited(;)
    
  • 由于Data Source是一个目录名(将其视为数据库),要打开的文件的文件名在查询:

      SELECT * FROM MyFile.csv
    

使用 Microsoft.ACE.OLEDB.12.0 作为提供者的示例连接字符串:

string connectionString = $@"Provider=Microsoft.ACE.OLEDB.12.0;Data Source={dirName};" +
                            "Extended Properties=\"text; HDR=Yes; IMEX=1; FMT=Delimited\";";

有关其他可用的连接字符串格式,请参阅 The Connection Strings Reference 网站。

测试结果的示例代码(本例中使用Microsoft.Jet.OLEDB.4.0:

private void Browse_Click(object sender, EventArgs e)
{
    string userFileName = string.Empty;
    using (var ofd = new OpenFileDialog()) {
        ofd.Filter = "CSV Files|*.csv|Excel '97-2003|*.xls|Excel 2007-2019|*.xlsx";
        if (ofd.ShowDialog(this) == DialogResult.OK) {
            userFileName = ofd.FileName;
        }
    }
    
    if (userFileName.Length == 0) return;
    dataGridView1.DataSource = GetData(userFileName);
}

private DataTable GetData(string userFileName)
{
    string dirName = Path.GetDirectoryName(userFileName);
    string fileName = Path.GetFileName(userFileName);
    string fileExtension = Path.GetExtension(userFileName);
    string conString = string.Empty;
    string query = string.Empty;

    switch (fileExtension)
    {
        // Can also use Microsoft.ACE.OLEDB.12 or Microsoft.ACE.OLEDB.16
        case ".xls":
            conString = $@"Provider=Microsoft.Jet.OLEDB.4.0;Data Source={userFileName};" +
                           "Extended Properties=\"Excel 8.0; HDR=Yes; IMEX=1\"";
            query = "SELECT * FROM [Sheet1$]";
            break;
        // Can also use Microsoft.ACE.OLEDB.16
        case ".xlsx":
            conString = $@"Provider=Microsoft.ACE.OLEDB.12.0;Data Source={userFileName};" +
                           "Extended Properties=\"Excel 12.0; HDR=Yes; IMEX=1\"";
            query = "SELECT * FROM [Sheet1$]";
            break;
        // Can also use Microsoft.ACE.OLEDB.16
        case ".csv":
            conString = $@"Provider=Microsoft.ACE.OLEDB.12.0;Data Source={dirName};" +
                           "Extended Properties=\"text; HDR=Yes; IMEX=1; FMT=Delimited\"";
            query = $"SELECT * FROM {fileName}";
            break;
    }
    return FillData(conString, query);
}

private DataTable FillData(string conString, string query)
{
    var dt = new DataTable();
    using (var con = new OleDbConnection(conString)) { 
        con.Open();
        using (var cmd = new OleDbCommand(query, con))
        using (var reader = cmd.ExecuteReader()) {
            dt.Load(reader);
        };
    }
    return dt;
}