C#中是否有任何连接字符串解析器?


Answers:


305

是的,有System.Data.Common.DbConnectionStringBuilder课。

DbConnectionStringBuilder类提供基类,强类型连接字符串构建器(SqlConnectionStringBuilder,OleDbConnectionStringBuilder等)从该基类派生。连接字符串构建器允许开发人员以编程方式创建语法正确的连接字符串,并解析和重建现有的连接字符串。

感兴趣的子类是:

System.Data.EntityClient.EntityConnectionStringBuilder
System.Data.Odbc.OdbcConnectionStringBuilder
System.Data.OleDb.OleDbConnectionStringBuilder
System.Data.OracleClient.OracleConnectionStringBuilder
System.Data.SqlClient.SqlConnectionStringBuilder

例如,要从SQL服务器连接字符串“窥视数据源”,可以执行以下操作:

var builder = new SqlConnectionStringBuilder(connectionString);
var dataSource = builder.DataSource;

6
基类DbConnectionStringBuilder具有可使用的通用处理功能,而无需使用子类:if (builder.TryGetValue("Password", out var pwd)) { string decrypted = SomehowDecrypt(pwd); builder["Password"] = decrypted; }
Olivier Jacot-Descombes,

41

有来自不同供应商,如供应商特定的连接字符串的建设者SqlConnectionStringBuilderMySqlConnectionStringBuilderSQLiteConnectionStringBuilder等(遗憾的是,从MS这次没有公共接口)。否则,您将拥有DbProviderFactory.CreateConnectionStringBuilder,这将为您提供一种与提供商无关的书写方式。您需要在配置文件中指定提供程序,并具有可用的正确版本的dll。例如

var c = "server=localhost;User Id=root;database=ppp";
var f = DbProviderFactories.GetFactory("MySql.Data.MySqlClient"); //your provider
var b = f.CreateConnectionStringBuilder();
b.ConnectionString = c;
var s = b["data source"];
var d = b["database"];

我曾经为自己编写过手动解析,这并没有给我带来任何麻烦。扩展它以提供其他参数的信息将是微不足道的(目前仅适用于简单的事情,例如数据库名称,数据源,用户名和密码)。大概这样:

static readonly string[] serverAliases = { "server", "host", "data source", "datasource", "address", 
                                           "addr", "network address" };
static readonly string[] databaseAliases = { "database", "initial catalog" };
static readonly string[] usernameAliases = { "user id", "uid", "username", "user name", "user" };
static readonly string[] passwordAliases = { "password", "pwd" };

public static string GetPassword(string connectionString)
{
    return GetValue(connectionString, passwordAliases);
}

public static string GetUsername(string connectionString)
{
    return GetValue(connectionString, usernameAliases);
}

public static string GetDatabaseName(string connectionString)
{
    return GetValue(connectionString, databaseAliases);
}

public static string GetServerName(string connectionString)
{
    return GetValue(connectionString, serverAliases);
}

static string GetValue(string connectionString, params string[] keyAliases)
{
    var keyValuePairs = connectionString.Split(';')
                                        .Where(kvp => kvp.Contains('='))
                                        .Select(kvp => kvp.Split(new char[] { '=' }, 2))
                                        .ToDictionary(kvp => kvp[0].Trim(),
                                                      kvp => kvp[1].Trim(),
                                                      StringComparer.InvariantCultureIgnoreCase);
    foreach (var alias in keyAliases)
    {
        string value;
        if (keyValuePairs.TryGetValue(alias, out value))
            return value;
    }
    return string.Empty;
}

为此,您不需要任何特殊的配置文件或任何dll。仅当您需要绕过格式不正确的连接字符串(如where 什么都不添加)时,Containsin Where子句才重要。要像正常的生成器一样工作(在这种情况下会爆炸),请将更改为server = localhost;pp;ppWhere

.Where(kvp => !string.IsNullOrWhitespace(kvp))

@Icarus并不是真的,因为字典的键比较器是StringComparer.InvariantCultureIgnoreCase。看到ToDictionary超载
nawfal

1
是的,您是对的,我只是写了一个快速测试。由于错误,将删除我的原始评论。
伊卡洛斯

3
在以下情况下,您的GetValue()方法将无法使用: ';''='密码或的。我写了一个类似的实现,并了解到它不是很难实现的。天哪,连接字符串解析实际上比我想象的要难得多!
菲利普·阿兹

@Joshua我希望您正在谈论手动解析部分。请以此处的答案为起点,而不是简单且经过战斗验证的解决方案。我希望它们比不留下任何信息的评论更有价值。您也可以自由投票。据我所知,需要进一步做的是坚持解析标准。MS在msdn上有一个,在我心中已经想修改了很长时间。如果我们所有人都有时间的话。“ @”所有人都请记住紧要关头,尤其是菲利普·阿兹(Philip Atz)评论中的一个。
nawfal

@nawfal:解决后,我走了过来,留下了答案。
约书亚记

15

这是几行代码,可以将任何连接字符串解析为字典:

Dictionary<string, string> connStringParts = connString.Split(';')
    .Select(t => t.Split(new char[] { '=' }, 2))
    .ToDictionary(t => t[0].Trim(), t => t[1].Trim(), StringComparer.InvariantCultureIgnoreCase);

然后您可以访问任何部分:

string dataSource = connStringParts["Data Source"];

3
聪明,我唯一要更改的内容将包括StringSplitOptions.RemoveEmptyEntries在第一次拆分中,因为IndexOutOfRange如果出现拖尾,它将导致异常;
Scott Chamberlain

2
这可行,但是连接字符串构建器更健壮。在无效连接字符串的情况下,这样的代码将引发低级异常,而不是更有意义的解析错误。
2013年

帮我分析一个ADO连接字符串,这样我就可以建立一个相当于SqlConnectionSqlConnectionStringBuilder
Holistic Developer'Apr

这里有一些警告。例如,连接字符串值可以用引号引起来。
Seva Alekseyev '18年



4

是的,您可以使用ConnectionStringBuilder类来实现。这是标准数据提供者可用的DbConnectionStringBuilder实现的列表:

System.Data.Odbc.OdbcConnectionStringBuilder
System.Data.OleDb.OleDbConnectionStringBuilder
System.Data.OracleClient.OracleConnectionStringBuilder
System.Data.SqlClient.SqlConnectionStringBuilder

这是解析连接字符串并显示其元素的示例示例。

 string conString = @"Data Source=.\sqlexpress;" +
                        "Database=Northwind;Integrated Security=SSPI;" +
                        "Min Pool Size=5;Max Pool Size=15;Connection Reset=True;" +
                        "Connection Lifetime=600;";
    // Parse the SQL Server connection string and display it's properties

    SqlConnectionStringBuilder objSB1 = new SqlConnectionStringBuilder(conString);
    Response.Write("<b>Parsed SQL Connection String Parameters:</b>");
    Response.Write(" <br/>  Database Source = " + objSB1.DataSource);
    Response.Write(" <br/>  Database = " + objSB1.InitialCatalog);
    Response.Write(" <br/>  Use Integrated Security = " + objSB1.IntegratedSecurity);
    Response.Write(" <br/>  Min Pool Size = " + objSB1.MinPoolSize);
    Response.Write(" <br/>  Max Pool Size = " + objSB1.MaxPoolSize);
    Response.Write(" <br/>  Lifetime = " + objSB1.LoadBalanceTimeout);

4

您可以使用DbConnectionStringBuilder,并且不需要任何特定的提供程序:

如下代码:

var cnstr = "Data Source=data source value;Server=ServerValue";
var builder = new DbConnectionStringBuilder();
builder.ConnectionString = cnstr;
Console.WriteLine("Data Source: {0}", builder["Data Source"]);
Console.WriteLine("Server: {0}", builder["Server"]);

输出到控制台:

Data Source: data source value
Server: ServerValue

编辑:

由于DbConnectionStringBuilder实现IDictionary,因此您可以枚举连接字符串参数:

foreach (KeyValuePair<string, object> kv in builder)
{
    Console.WriteLine("{0}: {1}", kv.Key, kv.Value);
}

假定您已经知道连接字符串具有“数据源”值等,但并不总是如此,如上面stackoverflow.com/a/15529085/534109中所述。
Tieson T.

我的回答addrees具体是什么运问:“我希望能偷看例如,‘数据源’”
赫苏斯·洛佩斯

我对其进行了编辑以显示所有连接字符串参数。
赫苏斯·洛佩斯

1

我不太喜欢这里的所有答案。所以这就是我发现的。

使用.NET Core

您可以DbConnectionStringBuilder直接使用:

var builder = new System.Data.Common.DbConnectionStringBuilder();
builder.ConnectionString = settings.ConnectionString;
var server = builder["server"];

0

因此,我发现所有现有答案或多或少都是错误的。我最终得到了以下简单的解决方案:

class ConnectionStringParser: DbConnectionStringBuilder {
    ConnectionStringParser(string c) { Connection = c; }
    public override bool ShouldSerialize(string keyword) => true;
}

解析器位于DbConnectionStringBuilder中,非常容易获得。我们要做的唯一愚蠢的事情是将ShouldSerialize设置为始终返回true,以防止在尝试往返任意连接字符串时丢失组件。


2
现有答案出了什么问题?或者您的解决方案可以解决什么。说明会有所帮助。
nawfal
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.