TSQL foreign keys on views?

后端 未结 11 982
臣服心动
臣服心动 2021-01-01 13:51

I have a SQL-Server 2008 database and a schema which uses foreign key constraints to enforce referential integrity. Works as intended. Now the user creates views on the orig

相关标签:
11条回答
  • 2021-01-01 14:20

    Using Rob Farley's schema:

    CREATE TABLE dbo.testtable(
    id int IDENTITY(1,1) PRIMARY KEY,
    val int NOT NULL); 
    go
    INSERT dbo.testtable(val)
    VALUES(20),(30),(40),(50),(60),(70);
    go 
    CREATE TABLE dbo.secondtable(
    id int NOT NULL,
    CONSTRAINT FK_SecondTable FOREIGN KEY(id) REFERENCES dbo.TestTable(id)); 
    go
    
    CREATE TABLE z(n tinyint PRIMARY KEY);
    INSERT z(n)
    VALUES(0),(1);
    go
    CREATE VIEW dbo.SecondTableCheck WITH SCHEMABINDING AS
    SELECT 1 n
    FROM dbo.TestTable AS t JOIN dbo.SecondTable AS s ON t.Id = s.Id
    CROSS JOIN dbo.z
    WHERE t.Val < 50;
    go
    CREATE UNIQUE CLUSTERED INDEX NoSmallIds ON dbo.SecondTableCheck(n);
    go
    

    I had to create a tiny helper table (dbo.z) in order to make this work, because indexed views cannot have self joins, outer joins, subqueries, or derived tables (and TVCs count as derived tables).

    0 讨论(0)
  • 2021-01-01 14:21

    Peter already hit on this, but the best solution is to:

    1. Create the "main" logic (that filtering the referenced table) once.
    2. Have all views on related tables join to the view created for (1), not the original table.

    I.e.,

    CREATE VIEW v1 AS SELECT * FROM table1 WHERE blah
    
    CREATE VIEW v2 AS SELECT * FROM table2 WHERE EXISTS
      (SELECT NULL FROM v1 WHERE v1.id = table2.FKtoTable1)
    

    Sure, syntactic sugar for propagating filters for views on one table to views on subordinate tables would be handy, but alas, it's not part of the SQL standard. That said, this solution is still good enough -- efficient, straightforward, maintainable, and guarantees the desired state for the consuming code.

    0 讨论(0)
  • 2021-01-01 14:22

    Another approach, depending on your requirements, would be to use a stored procedure to return two recordsets. You pass it filtering criteria and it uses the filtering criteria to query table 1, and then those results can be used to filter the query to table 2 so that it's results are also consistent. Then you return both results.

    0 讨论(0)
  • 2021-01-01 14:27

    If rolling over tables so that Identity columns will not clash, one possibility would be to use a lookup table that referenced the different data tables by Identity and a table reference.

    Foreign keys on this table would work down the line for referencing tables.

    This would be expensive in a number of ways Referential integrity on the lookup table would have to be be enforced using triggers. Additional storage of the lookup table and indexing in addition to the data tables. Data reading would almost certainly involve a Stored Procedure or three to execute a filtered UNION. Query plan evaluation would also have a development cost.

    The list goes on but it might work on some scenarios.

    0 讨论(0)
  • 2021-01-01 14:27

    You could stage the filtered table 1 data to another table. The contents of this staging table are your view 1, and then you build view 2 via a join of the staging table and table 2. This way the proccessing for filtering table 1 is done once and reused for both views.

    Really what it boils down to is that view 2 has no idea what kind of filtering you performed in view 1, unless you tell view 2 the filtering criteria, or make it somehow dependent on the results of view 1, which means emulating the same filtering that occurs on view1.

    Constraints don't perform any kind of filtering, they only prevent invalid data, or cascade key changes and deletes.

    0 讨论(0)
  • 2021-01-01 14:28

    From a purely data integrity perspective (and nothing to do with the Query Optimizer), I had considered an Indexed View. I figured you could make a unique index on it, which could be broken when you try to have broken integrity in your underlying tables.

    But... I don't think you can get around the restrictions of indexed views well enough.

    For example:

    You can't use outer joins, or sub-queries. That makes it very hard to find the rows that don't exist in the view. If you use aggregates, you can't use HAVING, so that cuts out some options you could use there too. You can't even have constants in an indexed view if you have grouping (whether or not you use a GROUP BY clause), so you can't even try putting an index on a constant field so that a second row will fall over. You can't use UNION ALL, so the idea of having a count which will break a unique index when it hits a second zero won't work.

    I feel like there should be an answer, but I'm afraid you're going to have to take a good look at your actual design and work out what you really need. Perhaps triggers (and good indexes) on the tables involved, so that any changes that might break something can roll it all that.

    But I was really hoping to be able to suggest something that the Query Optimizer might be able to leverage to help the performance of your system, but I don't think I can.

    0 讨论(0)
提交回复
热议问题