關於SQL Server中修改“用戶自定義表類型”的問題

来源:http://www.cnblogs.com/wy123/archive/2017/08/03/7282682.html
-Advertisement-
Play Games

本文出處:http://www.cnblogs.com/wy123/p/7282682.html (保留出處並非什麼原創作品權利,本人拙作還遠遠達不到,僅僅是為了鏈接到原文,因為後續對可能存在的一些錯誤進行修正或補充,無他) SQL Server開發過程中,為了傳入數據集類型的變數(比如接受C#中的 ...


 

本文出處:http://www.cnblogs.com/wy123/p/7282682.html 
(保留出處並非什麼原創作品權利,本人拙作還遠遠達不到,僅僅是為了鏈接到原文,因為後續對可能存在的一些錯誤進行修正或補充,無他)

 

SQL Server開發過程中,為了傳入數據集類型的變數(比如接受C#中的DataTable類型變數),需要定義“用戶自定義表類型”,通過“用戶自定義表類型”可以接收二維數據集作為參數,
在需要修改“用戶自定義表類型”的時候,增加欄位,刪除欄位,修改欄位類型等,它沒有像表一樣的alter table語法來進行修改。
只能通過刪除重建來實現,但是在刪除“用戶自定義表類型”的時候會提示有對象引用它(某些存儲過程用到了這個“用戶自定義表類型”),因此無法刪除。
為了達到公用的目的,有時候一個TableType可以在多個地方分別被引用到,這樣的話,勢必要先刪除所有的引用了這個“用戶自定義表類型”的對象(存儲過程等)
如果這個“用戶自定義表類型”被多個存儲過程引用,那麼就要分別刪除多個引用了“用戶自定義表類型”的存儲過程,然後修改“用戶自定義表類型”,在重建存儲過程,這樣做起來似乎有點繞,
這個問題可以用過EXEC sys.sp_refreshsqlmodule這個系統函數來簡介實現“用戶自定義表類型”的定義

 

TableType的基本使用

 如下創建一個用戶自定義表類型

定義的TableType可以在用戶自定義表類型中找到

 

 創建兩個存儲過程,分別用到了上面定義的用戶自定義表類型,模擬用戶自定義表類型被引用的情況

 此時的存儲過程可以接收TableType參數並正常運行

 

TableType的修改

 TableType類型不支持alter語法,也即無法直接修改TableType的定義

那麼只能通過刪除TableType的方法來重建這個TableType,當刪除的時候,仍然報錯,提示“因為它正由對象 '***' 引用。可能還有其他對象在引用此類型。”

此時只能刪除引用了這個TableType的對象來解決,下麵可以查到那些對象引用了某一個TableType,然後分別刪除,重建TableType,再重建存儲過程,有點繞彎子。

 可以先將自定義的某個TableType重命名,重命名的過程中有一個警告,這裡先忽略它,隨後可以直接Drop Type dbo.MyTableType

 

 刪除原TableType之後,重建(重定義)TableType

 

重建TableType之後,先前存儲過程中用到這個TableType的存儲過程是無法編譯通過的

此時就需要重新刷新引用對象的定義

刷新完成之後,原存儲過程就可以正常編譯了

 最後刪除原始的TableType被重命名的TableType(被第一步重名的那個)

 

   這樣子,整個過程就無需因為修改TableType的定義而刪除引用了TableType的對象了,在修改了TableType的定義之後,引用了這個TableType的對象可以正常運行,也可以根據修改之後的TableType做具體的使用

 

  完整的腳本如下

--判斷Type是否存在,如果存在,重命名,隨後之後才再刪除,否則無法直接刪除
IF EXISTS (SELECT 1 FROM sys.types t join sys.schemas s on t.schema_id=s.schema_id 
                        and t.name='MyTableType' and s.name='dbo')
   EXEC sys.sp_rename 'dbo.MyTableType', 'obsoleting_MyTableType';
GO


--重建TYPE,比如原來是四個欄位,現在想修改為三個欄位,或者原來有三個欄位想加一個欄位變成四個欄位
CREATE TYPE dbo.MyTableType AS TABLE(
    Id INT NOT NULL,
    Name VARCHAR(255) NOT NULL,
   Remark
VARCHAR(255)
)
GO

--將原來引用將要刪除的TYPE全部重建一遍,否則原始存儲過程會報錯
DECLARE @Name NVARCHAR(500);
DECLARE REF_CURSOR CURSOR FOR
SELECT referencing_schema_name + '.' + referencing_entity_name
FROM sys.dm_sql_referencing_entities('dbo.MyTableType', 'TYPE');
    OPEN REF_CURSOR;
    FETCH NEXT FROM REF_CURSOR INTO @Name;
    WHILE (@@FETCH_STATUS = 0)
    BEGIN
        EXEC sys.sp_refreshsqlmodule @name = @Name;
        FETCH NEXT FROM REF_CURSOR INTO @Name;
    END;
CLOSE REF_CURSOR;
DEALLOCATE REF_CURSOR;
GO

--最後刪除原始的被重命名的TableType(被第一步重名的那個)
IF EXISTS (SELECT 1 FROM sys.types t 
            join sys.schemas s on t.schema_id=s.schema_id 
            and t.name='obsoleting_MyTableType' and s.name='dbo')
   DROP TYPE dbo.obsoleting_MyTableType
GO

--最後執行授權
GRANT EXECUTE ON TYPE::dbo.MyTableType TO public
GO

 

 

總結:

  TableType可以方便地接受二維數據作為參數,從而可以達到批量處理數據的目的,避免傳遞進去一大堆字元串,然後在對字元串解析的做法,從而可以在一定程度上提高sql的運行效率。
   不過TableType的修改確實存在一定的問題,直接修改TableType會存在級聯刪除資料庫對象的情況,可以通過“曲線救國”的方式,來減小工作量的情況下修改TableType。


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

-Advertisement-
Play Games
更多相關文章
  • TextView兩種顯示link的方法 一、簡介 也是TextView顯示文本控制項兩種方法 也是顯示豐富的文本 二、方法 TextView兩種顯示link的方法 1)通過TextView裡面的類html標簽 * 1、設置好html標簽的文本 String text1="<font color='re ...
  • View繪製的三部曲,測量,佈局,繪畫現在我們分析佈局部分測量部分在上篇文章中已經分析過了。不瞭解的可以去我的博客里找一下View的佈局和測量一樣,都是從ViewRootImpl中發起,ViewRootImpl先通過measure來初始化整個的view樹之後會調用onLayout方法來佈局,View ...
  • 1.UIButton UIButton的類是一個UIControl子類,它實現了在觸摸屏上的按鈕。觸摸一個按鈕攔截事件和動作消息發送到目標對象時,它的挖掘。設定的目標和行動方法都繼承自UIControl。這個類提供了方法來設置標題,圖像,按鈕等外觀屬性。通過使用set方法,你可以指定一個不同的外觀為 ...
  • 想要掌握一樣東西,最好的方式就是閱讀理解它的源碼。想要掌握Android Binder,最好的方式就是寫一個AIDL文件,然後查看其生成的代碼。本文的思路也是來自於此。 ...
  • iOS開發中有時候需要拿到字元串的行數來實現特定的功能,下麵代碼用來實現根據字元串內容和字元串的顯示寬度(即label的長度)來計算行數 1.引入#import <CoreText/CoreText.h> 頭文件 2.項目開發需要實現在朋友圈內容右下角加展開按鈕,避免文字遮擋按鈕,就把區分好的字元串 ...
  • 使用Java代碼來創建view 一、簡介 需要瞭解的知識 二、方法 1)java代碼創建view方法 * 1、先建view對象 View view= View.inflate(this, R.layout.activity01, null); * 2、在view中填充R.layout.activit ...
  • 相對佈局relativeLayout 一、簡介 二、實例 代碼 /Test_FrameLayout/res/layout/relativelayout.xml ...
  • 程式猿是否應該接私活? 我大約從去年 11 月份開始接私活做,到目前為止也有大半年了。自己在這大半年的時間里,也學習了很多,在此記錄一下這大半年來的感受以及一些不成熟的建議。 在我看來,在你又打算接私活的時候,你應該考慮以下幾個問題 時間是否充裕? 個人認為這個問題非常重要,如果時間不夠充裕的話,接 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...