Entity Framework CTP 4 - Code First Custom Database Initializer

后端 未结 6 635
傲寒
傲寒 2020-11-28 23:53

I would like to implement a custom database initialization strategy so that I can generate the database schema and apply it to an EXISTING EMPTY SQL database using a supplie

相关标签:
6条回答
  • 2020-11-29 00:26

    I too was looking for a good solution since godaddy does not allow drop/creation of database and thus no tables created. Since the newer version of Entity Framework has obsoleted EDMData, I modified Alex's code to see if a DropMeToRecreateDatabase table exists or not, if it doesnt exist, it deletes all tables and recreates new tables.

    using System.Data.Entity;
    using System.Data.Entity.Infrastructure;
    using System.Data.Objects;
    using System.Linq;
    
    namespace LadyTreble.DatabaseInitializer
    {
    
        public class DontDropExistingDbCreateTablesIfTableDropped<T> : IDatabaseInitializer<T> where T : DbContext
        {
            public bool TryInitializeDatabase(T context)
            {
                var objectContext = ((IObjectContextAdapter)context).ObjectContext;
                if (objectContext.ExecuteStoreQuery<int>(GetTableCount).FirstOrDefault() == 0)
                {
                    this.DeleteExistingTables(objectContext);
                    this.CreateTables(objectContext);
                }
                return true;
            }
    
            public void InitializeDatabase(T context)
            {
                this.TryInitializeDatabase(context);
            }
    
            private void CreateTables(ObjectContext objectContext)
            {
                string dataBaseCreateScript = objectContext.CreateDatabaseScript();
                objectContext.ExecuteStoreCommand(dataBaseCreateScript);
            }
    
            private void DeleteExistingTables(ObjectContext objectContext)
            {
    
                objectContext.ExecuteStoreCommand(DeleteAllTablesScript);
            }
    
            private const string DeleteAllTablesScript =
                @"declare @cmd varchar(4000)
    
                  DECLARE cmds0 CURSOR FOR 
                  SELECT 'ALTER TABLE ' + TABLE_NAME + ' DROP CONSTRAINT ' + CONSTRAINT_NAME FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS WHERE CONSTRAINT_TYPE = 'FOREIGN KEY'
    
                  DECLARE cmds1 CURSOR FOR 
                  SELECT 'ALTER TABLE ' + TABLE_NAME + ' DROP CONSTRAINT ' + CONSTRAINT_NAME FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS
    
                  DECLARE cmds2 CURSOR FOR 
                  SELECT 'TRUNCATE TABLE ' + TABLE_NAME FROM INFORMATION_SCHEMA.TABLES
    
                  DECLARE cmds3 CURSOR FOR 
                  SELECT 'DROP TABLE [' + TABLE_NAME + ']' FROM INFORMATION_SCHEMA.TABLES
    
                  open cmds0
                  while 1=1
                  begin
                      fetch cmds0 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds0
                  deallocate cmds0
    
                  open cmds1
                  while 1=1
                  begin
                      fetch cmds1 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds1
                  deallocate cmds1
    
                  open cmds2
                  while 1=1
                  begin
                      fetch cmds2 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds2
                  deallocate cmds2
    
                  open cmds3
                  while 1=1
                  begin
                      fetch cmds3 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds3
                  deallocate cmds3
    
                 CREATE TABLE DropMeToRecreateDatabase(id int IDENTITY(1,1) NOT NULL)";
    
            private const string GetTableCount =
                @"SELECT COUNT(*) FROM INFORMATION_SCHEMA.TABLES T WHERE T.TABLE_NAME = 'DropMeToRecreateDatabase'";
        }
    
    }
    
    0 讨论(0)
  • 2020-11-29 00:30

    I took a slightly different approach to this problem. This seems like as good a place as any to share the results.

    I want to only create tables that don't already exist in the database. This has the benefit of being able to roll out new tables without erasing the rest of the database.

    This also helps if you have multiple data contexts in an inheritance chain. For example, if you split your application into different assemblies. You might have a data context in a "core" module, and then inherit it in a different assembly for add-on modules. This configuration works fine, but the built-in Drop/Create initializers don't like it because the model hash is changing all the time. By checking table existance, initialization takes slightly longer, but then you have none of these issues.

    Anyway, here's the code:

    /// <summary>
    /// Database Initializer to create tables only if they don't already exist.
    /// It will never drop the database.  Does not check the model for compatibility.
    /// </summary>
    /// <typeparam name="TContext">The data context</typeparam>
    public class CreateTablesOnlyIfTheyDontExist<TContext> : IDatabaseInitializer<TContext>
      where TContext : DataContext
    {
      public void InitializeDatabase(TContext context)
      {
        using (new TransactionScope(TransactionScopeOption.Suppress))
        {
          // If the database doesn't exist at all then just create it like normal.
          if (!context.Database.Exists())
          {
            context.Database.Create();
            return;
          }
    
          // get the object context
          var objectContext = ((IObjectContextAdapter)context).ObjectContext;
    
          // get the database creation script
          var script = objectContext.CreateDatabaseScript();
    
    
          if (context.Database.Connection is SqlConnection)
          {
            // for SQL Server, we'll just alter the script
    
            // add existance checks to the table creation statements
            script = Regex.Replace(script,
              @"create table \[(\w+)\]\.\[(\w+)\]",
              "if not exists (select * from INFORMATION_SCHEMA.TABLES " +
              "where TABLE_SCHEMA='$1' and TABLE_NAME = '$2')\n$&");
    
            // add existance checks to the table constraint creation statements
            script = Regex.Replace(script,
              @"alter table \[(\w+)\]\.\[(\w+)\] add constraint \[(\w+)\]",
              "if not exists (select * from INFORMATION_SCHEMA.TABLE_CONSTRAINTS " +
              "where TABLE_SCHEMA='$1' and TABLE_NAME = '$2' " +
              "and CONSTRAINT_NAME = '$3')\n$&");
    
            // run the modified script
            objectContext.ExecuteStoreCommand(script);
          }
          else if (context.Database.Connection is SqlCeConnection)
          {
            // SQL CE doesn't let you use inline existance checks,
            // so we have to parse each statement out and check separately.
    
            var statements = script.Split(new[] { ";\r\n" },
                            StringSplitOptions.RemoveEmptyEntries);
            foreach (var statement in statements)
            {
              var quoteSplitStrings = statement.Split('"');
              if (statement.StartsWith("CREATE TABLE"))
              {
                // Create a table if it does not exist.
                var tableName = quoteSplitStrings[1];
                const string sql = 
                  "SELECT COUNT(*) FROM INFORMATION_SCHEMA.TABLES " +
                  "WHERE TABLE_NAME='{0}'"
                var checkScript = string.Format(sql, tableName);
                if (objectContext.ExecuteStoreQuery<int>(checkScript).First() == 0)
                  objectContext.ExecuteStoreCommand(statement);
              }
              else if (statement.Contains("ADD CONSTRAINT"))
              {
                // Add a table constraint if it does not exist.
                var tableName = quoteSplitStrings[1];
                var constraintName = quoteSplitStrings[3];
                const string sql = 
                  "SELECT COUNT(*) FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS " +
                  "WHERE TABLE_NAME='{0}' AND CONSTRAINT_NAME='{1}'";
                var checkScript = string.Format(sql, tableName, constraintName);
                if (objectContext.ExecuteStoreQuery<int>(checkScript).First() == 0)
                  objectContext.ExecuteStoreCommand(statement);
              }
              else
              {
                // Not sure what else it could be. Just run it.
                objectContext.ExecuteStoreCommand(statement);
              }
            }
          }
          else
          {
            throw new InvalidOperationException(
              "This initializer is only compatible with SQL Server or SQL Compact Edition"
              );
          }
        }
      }
    }
    
    0 讨论(0)
  • 2020-11-29 00:32

    This is the easiest way to get EF Code First running on AppHarbor!

    Using the EdmMetadata.TryGetModelHash(context) function to check when the model doesn't match the database and showing an error with the new code that needs to be used after you run alteration scripts.

    PopulateOnly : Only creates objects when the database is empty

    I thought I'd post my own version of the Initializer which I'm currently using on appharbor to populate an existing database. It will also try to do a create if the database doesn't exists and throws if a change is detected (sorry no automatic updating yet). I hope someone finds it useful.

        using System;
        using System.Data.Entity;
        using System.Data.Entity.Infrastructure;
        using System.Data.Objects;
        using System.Transactions;
    
        namespace Deskspace.EntityFramework
        {
    
            /// <summary> A Database Initializer for appharbor </summary>
            /// <typeparam name="T">Code first context</typeparam>
            public class PopulateOnly<T> : IDatabaseInitializer<T> where T : DbContext
            {
                private EdmMetadata metadata;
    
                private enum Status
                {
                    Compatable,
                    Invalid,
                    Missing
                }
    
                /// <summary> Initializer that supports creating or populating a missing or empty database </summary>
                /// <param name="context"> Context to create for </param>
                public void InitializeDatabase(T context)
                {
                    // Get metadata hash
                    string hash = EdmMetadata.TryGetModelHash(context);
    
                    bool exists;
                    using (new TransactionScope( TransactionScopeOption.Suppress )) {
                        exists = context.Database.Exists();
                    }
    
                    if (exists) {
    
                        ObjectContext objectContext = ((IObjectContextAdapter)context).ObjectContext;
    
                        var dbHash = GetHashFromDatabase( objectContext );
    
                        Status compatability = 
                                string.IsNullOrEmpty( dbHash )? 
                            Status.Missing : 
                                (dbHash != hash)? 
                            Status.Invalid :
                            Status.Compatable;
    
                        if (compatability == Status.Missing) {
    
                            // Drop all database objects
                            ClearDatabase( objectContext );
    
                            // Recreate database objects
                            CreateTables( objectContext );
    
                            // Save the new hash
                            SaveHash( objectContext,  hash );
    
                        } else if (compatability == Status.Invalid) {
    
                            throw new Exception( 
                                "EdmMetadata does not match, manually update the database, expected: " + 
                                Environment.NewLine + 
                                "<[(" + hash + ")}>"
                            );
                        }
                    } else {
                        context.Database.Create();
                        context.SaveChanges();
                    }
                }
    
                private void ClearDatabase(ObjectContext objectContext)
                {
                    objectContext.ExecuteStoreCommand( DropAllObjects );
                }
    
                private void CreateTables(ObjectContext objectContext)
                {
                    string dataBaseCreateScript = objectContext.CreateDatabaseScript();
                    objectContext.ExecuteStoreCommand( dataBaseCreateScript );
                }
    
                private void SaveHash(ObjectContext objectContext, string hash)
                {
                    objectContext.ExecuteStoreCommand( string.Format(UpdateEdmMetaDataTable, hash.Replace( "'", "''" )) );
                }
    
                private string GetHashFromDatabase(ObjectContext objectContext)
                {
                    foreach (var item in objectContext.ExecuteStoreQuery<string>( GetEdmMetaDataTable )) {
                        return item;
                    }
    
                    return string.Empty;
                }
    
                private const string UpdateEdmMetaDataTable = @"
        Delete From EdmMetadata;
        Insert Into EdmMetadata (ModelHash) Values ('{0}');";
    
                private const string GetEdmMetaDataTable = @"
        If Exists (Select * From INFORMATION_SCHEMA.TABLES tables where tables.TABLE_NAME = 'EdmMetaData')
            Select Top 1 ModelHash From EdmMetadata;
        Else
            Select '';";
    
                private const string DropAllObjects = @"
        declare @n char(1)
        set @n = char(10)
    
        declare @stmt nvarchar(max)
    
        -- procedures
        select @stmt = isnull( @stmt + @n, '' ) +
            'drop procedure [' + name + ']'
        from sys.procedures
    
        -- check constraints
        select @stmt = isnull( @stmt + @n, '' ) +
            'alter table [' + object_name( parent_object_id ) + '] drop constraint [' + name + ']'
        from sys.check_constraints
    
        -- functions
        select @stmt = isnull( @stmt + @n, '' ) +
            'drop function [' + name + ']'
        from sys.objects
        where type in ( 'FN', 'IF', 'TF' )
    
        -- views
        select @stmt = isnull( @stmt + @n, '' ) +
            'drop view [' + name + ']'
        from sys.views
    
        -- foreign keys
        select @stmt = isnull( @stmt + @n, '' ) +
            'alter table [' + object_name( parent_object_id ) + '] drop constraint [' + name + ']'
        from sys.foreign_keys
    
        -- tables
        select @stmt = isnull( @stmt + @n, '' ) +
            'drop table [' + name + ']'
        from sys.tables
    
        -- user defined types
        select @stmt = isnull( @stmt + @n, '' ) +
            'drop type [' + name + ']'
        from sys.types
        where is_user_defined = 1
    
        exec sp_executesql @stmt";
    
            }
        }
    
    0 讨论(0)
  • 2020-11-29 00:38

    Just to contribute to @Luhmann's solution, here's mine but slightly changed to drop the FK and PK properly.

    using System.Data.Entity;
    using System.Data.Entity.Design;
    using System.Data.Entity.Infrastructure;
    using System.Data.Metadata.Edm;
    using System.Data.Objects;
    using System.Globalization;
    using System.Linq;
    using System.Security.Cryptography;
    using System.Text;
    using System.Xml;
    
    namespace SISQuote.Server.Persistence
    {
        public class DontDropExistingDbCreateTablesIfModelChanged<T> : IDatabaseInitializer<T> where T : DbContext
        {
            private EdmMetadata edmMetaData;
    
            public bool TryInitializeDatabase(T context)
            {
                ObjectContext objectContext = ((IObjectContextAdapter)context).ObjectContext;
                string modelHash = GetModelHash(objectContext);
    
                if (CompatibleWithModel(modelHash, context, objectContext))
                    return false;
    
                DeleteExistingTables(objectContext);
                CreateTables(objectContext);
                SaveModelHashToDatabase(context, modelHash, objectContext);
    
                return true;
            }
    
            public void InitializeDatabase(T context)
            {
                TryInitializeDatabase(context);
            }
    
            private void SaveModelHashToDatabase(T context, string modelHash, ObjectContext objectContext)
            {
                if (edmMetaData != null) 
                    objectContext.Detach(edmMetaData);
    
                edmMetaData = new EdmMetadata();
                context.Set<EdmMetadata>().Add(edmMetaData);
    
                edmMetaData.ModelHash = modelHash;
                context.SaveChanges();
            }
    
            private void CreateTables(ObjectContext objectContext)
            {
                string dataBaseCreateScript = objectContext.CreateDatabaseScript();
                objectContext.ExecuteStoreCommand(dataBaseCreateScript);
            }
    
            private void DeleteExistingTables(ObjectContext objectContext)
            {
                objectContext.ExecuteStoreCommand(DeleteAllTablesScript);
            }
    
            private string GetModelHash(ObjectContext context)
            {
                var csdlXmlString = GetCsdlXmlString(context).ToString();
                return ComputeSha256Hash(csdlXmlString);
            }
    
            public bool CompatibleWithModel(DbContext context)
            {
                ObjectContext objectContext = ((IObjectContextAdapter)context).ObjectContext;
                return CompatibleWithModel(GetModelHash(objectContext), context, objectContext);
            }
    
            private bool CompatibleWithModel(string modelHash, DbContext context, ObjectContext objectContext)
            {
                var isEdmMetaDataInStore = objectContext.ExecuteStoreQuery<int>(LookupEdmMetaDataTable).FirstOrDefault();
                if (isEdmMetaDataInStore == 1)
                {
                    edmMetaData = context.Set<EdmMetadata>().FirstOrDefault();
                    if (edmMetaData != null)
                    {
                        return modelHash == edmMetaData.ModelHash;
                    }
                }
                return false;
            }
    
            private string GetCsdlXmlString(ObjectContext context)
            {
                if (context != null)
                {
                    var entityContainerList = context.MetadataWorkspace.GetItems<EntityContainer>(DataSpace.SSpace);
                    if (entityContainerList != null)
                    {
                        EntityContainer entityContainer = entityContainerList.FirstOrDefault();
                        var generator = new EntityModelSchemaGenerator(entityContainer);
                        var stringBuilder = new StringBuilder();
                        var xmlWRiter = XmlWriter.Create(stringBuilder);
                        generator.GenerateMetadata();
                        generator.WriteModelSchema(xmlWRiter);
                        xmlWRiter.Flush();
                        return stringBuilder.ToString();
                    }
                }
                return string.Empty;
            }
    
            private static string ComputeSha256Hash(string input)
            {
                byte[] buffer = new SHA256Managed().ComputeHash(Encoding.ASCII.GetBytes(input));
                var builder = new StringBuilder(buffer.Length * 2);
                foreach (byte num in buffer)
                {
                    builder.Append(num.ToString("X2", CultureInfo.InvariantCulture));
                }
                return builder.ToString();
            }
    
            private const string DeleteAllTablesScript =
                @"declare @cmd varchar(4000)
    
                  DECLARE cmds0 CURSOR FOR 
                  SELECT 'ALTER TABLE ' + TABLE_NAME + ' DROP CONSTRAINT ' + CONSTRAINT_NAME FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS WHERE CONSTRAINT_TYPE = 'FOREIGN KEY'
    
                  DECLARE cmds1 CURSOR FOR 
                  SELECT 'ALTER TABLE ' + TABLE_NAME + ' DROP CONSTRAINT ' + CONSTRAINT_NAME FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS
    
                  DECLARE cmds2 CURSOR FOR 
                  SELECT 'TRUNCATE TABLE ' + TABLE_NAME FROM INFORMATION_SCHEMA.TABLES
    
                  DECLARE cmds3 CURSOR FOR 
                  SELECT 'DROP TABLE [' + TABLE_NAME + ']' FROM INFORMATION_SCHEMA.TABLES
    
                  open cmds0
                  while 1=1
                  begin
                      fetch cmds0 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds0
                  deallocate cmds0
    
                  open cmds1
                  while 1=1
                  begin
                      fetch cmds1 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds1
                  deallocate cmds1
    
                  open cmds2
                  while 1=1
                  begin
                      fetch cmds2 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds2
                  deallocate cmds2
    
                  open cmds3
                  while 1=1
                  begin
                      fetch cmds3 into @cmd
                      if @@fetch_status != 0 break
                      print @cmd
                      exec(@cmd)
                  end
                  close cmds3
                  deallocate cmds3";
    
            private const string LookupEdmMetaDataTable =
                @"Select COUNT(*) 
                  FROM INFORMATION_SCHEMA.TABLES T 
                  Where T.TABLE_NAME = 'EdmMetaData'";
        }
    }
    
    0 讨论(0)
  • 2020-11-29 00:40

    The Entity Designer Database Generation Power Pack will do this. Not sure if it works with Code First yet, but worth a shot.

    0 讨论(0)
  • 2020-11-29 00:42

    I ran into the same problem. I didn't really solve it, but I managed to get a little nasty workaround running, so i can deploy my solution to AppHarbor ;)

    Its a IDatabaseInitializer implementation, that doesn't delete the db, but just nukes all the constraints and tables, and then uses the ObjectContext.CreateDatabaseScript() method to generate the sql, and then I execute it as a storecommand. A lot like the above implementation in the question.

    But i also added functionality to create a hash from the model and save it in db, and when it runs again it checks if the current model-hash matches the one i db. Just like the real code-first implementation.

    I couldn't make it work with the build in context.Database.CompatibleWithModel(true) - but this should work just as well, and seeing as its a temporary workaround it should be fine.

    using System;
    using System.Data.Entity;
    using System.Data.Entity.Database;
    using System.Data.Entity.Design;
    using System.Data.Entity.Infrastructure;
    using System.Data.Metadata.Edm;
    using System.Data.Objects;
    using System.Globalization;
    using System.Security.Cryptography;
    using System.Text;
    using System.Xml;
    using System.Linq;
    
    namespace Devtalk
    {
        public class DontDropDbJustCreateTablesIfModelChanged<T> : IDatabaseInitializer<T> where T : DbContext
        {
            private EdmMetadata _edmMetaData;
    
            public void InitializeDatabase(T context)
            {
                ObjectContext objectContext = ((IObjectContextAdapter)context).ObjectContext;
                string modelHash = GetModelHash(objectContext);
    
                if (CompatibleWithModel(modelHash, context, objectContext)) return;
    
                DeleteExistingTables(objectContext);
                CreateTables(objectContext);
    
                SaveModelHashToDatabase(context, modelHash, objectContext);
            }
    
            private void SaveModelHashToDatabase(T context, string modelHash, ObjectContext objectContext)
            {
                if (_edmMetaData != null) objectContext.Detach(_edmMetaData);
    
                _edmMetaData = new EdmMetadata();
                context.Set<EdmMetadata>().Add(_edmMetaData);
    
                _edmMetaData.ModelHash = modelHash;
                context.SaveChanges();
            }
    
            private void CreateTables(ObjectContext objectContext)
            {
                string dataBaseCreateScript = objectContext.CreateDatabaseScript();
                objectContext.ExecuteStoreCommand(dataBaseCreateScript);
            }
    
            private void DeleteExistingTables(ObjectContext objectContext)
            {
                objectContext.ExecuteStoreCommand(Dropallconstraintsscript);
                objectContext.ExecuteStoreCommand(Deletealltablesscript);
            }
    
            private string GetModelHash(ObjectContext context)
            {
                var csdlXmlString = GetCsdlXmlString(context).ToString();
                return ComputeSha256Hash(csdlXmlString);
            }
    
            private bool CompatibleWithModel(string modelHash, DbContext context, ObjectContext objectContext)
            {
                var isEdmMetaDataInStore = objectContext.ExecuteStoreQuery<int>(LookupEdmMetaDataTable).FirstOrDefault();
                if (isEdmMetaDataInStore == 1)
                {            
                    _edmMetaData = context.Set<EdmMetadata>().FirstOrDefault();
                    if (_edmMetaData != null)
                    {
                        return modelHash == _edmMetaData.ModelHash;
                    }
                }
                return false;
            }
    
            private string GetCsdlXmlString(ObjectContext context)
            {
                if (context != null)
                {
                    var entityContainerList = context.MetadataWorkspace.GetItems<EntityContainer>(DataSpace.SSpace);
                    if (entityContainerList != null)
                    {
                        EntityContainer entityContainer = entityContainerList.FirstOrDefault();
                        var generator = new EntityModelSchemaGenerator(entityContainer);
                        var stringBuilder = new StringBuilder();
                        var xmlWRiter = XmlWriter.Create(stringBuilder);
                        generator.GenerateMetadata();
                        generator.WriteModelSchema(xmlWRiter);
                        xmlWRiter.Flush();
                        return stringBuilder.ToString();
                    }
                }
                return string.Empty;
            }
    
            private static string ComputeSha256Hash(string input)
            {
                byte[] buffer = new SHA256Managed().ComputeHash(Encoding.ASCII.GetBytes(input));
                var builder = new StringBuilder(buffer.Length * 2);
                foreach (byte num in buffer)
                {
                    builder.Append(num.ToString("X2", CultureInfo.InvariantCulture));
                }
                return builder.ToString();
            }
    
            private const string Dropallconstraintsscript =
                @"select  
                    'ALTER TABLE ' + so.table_name + ' DROP CONSTRAINT ' + so.constraint_name  
                    from INFORMATION_SCHEMA.TABLE_CONSTRAINTS so";
    
            private const string Deletealltablesscript =
                @"declare @cmd varchar(4000)
                    declare cmds cursor for 
                    Select
                        'drop table [' + Table_Name + ']'
                    From
                        INFORMATION_SCHEMA.TABLES
    
                    open cmds
                    while 1=1
                    begin
                        fetch cmds into @cmd
                        if @@fetch_status != 0 break
                        print @cmd
                        exec(@cmd)
                    end
                    close cmds
                    deallocate cmds";
    
            private const string LookupEdmMetaDataTable =
                @"Select COUNT(*) 
                  FROM INFORMATION_SCHEMA.TABLES T 
                  Where T.TABLE_NAME = 'EdmMetaData'";
        }
    }
    
    0 讨论(0)
提交回复
热议问题