MS Access, связанный с представлениями SQL server
-
10-07-2019 - |
Вопрос
у нас возникла проблема с базой данных Access, которую мы обновляем, чтобы использовать SQL Server в качестве хранилища данных.
Эта конкретная база данных связана с двумя базами данных sql, поэтому я подумал, что для упрощения у нас есть представление в основной базе данных, которое связано с каждой таблицей во вторичной базе данных.Таким образом, access должен был бы взаимодействовать напрямую только с одной базой данных SQL.
Когда мы связали access с представлениями базы данных, мы выбрали, какие поля были первичными ключами, чтобы представления не были доступны только для чтения.У нас есть стандартный код, который обновляет все ссылки, когда база данных открывается для получения любых изменений, и связанные представления становятся доступными только для чтения, поскольку информация о первичном ключе теряется.
Есть ли способ обновить ссылки на представления с сохранением информации о первичном ключе?
Джон
Решение
Я включил всю свою функцию повторного подключения ODBC ниже.Эта функция основана на идее, что у меня есть таблица с именем rtblODBC, в которой хранится вся информация, необходимая мне для повторного подключения.Если вы реализуете эту функцию, вам НЕ нужно будет беспокоиться о подключении к нескольким базам данных SQL, поскольку это выполняется плавно, и каждая таблица, подлежащая повторному подключению, имеет свою собственную строку подключения.
Когда вы дойдете до конца, вы увидите, что я использую DAO для воссоздания первичных ключей с помощью db.Выполните "СОЗДАТЬ ИНДЕКС " & sPrimaryKeyName & " НА " & sLocalTableName & "(" & sPrimaryKeyField & ")С ПОМОЩЬЮ PRIMARY;"
Если у вас есть какие-либо вопросы, пожалуйста, задавайте.
Public Function fnReconnectODBC( _
Optional bForceReconnect As Boolean _
) As Boolean
' Comments :
' Parameters: bForceReconnect -
' Returns : Boolean -
' Modified :
' --------------------------------------------------'
On Error GoTo Err_fnReconnectODBC
Dim db As DAO.Database
Dim rs As DAO.Recordset
Dim tdf As DAO.TableDef
Dim sPrimaryKeyName As String
Dim sPrimaryKeyField As String
Dim sLocalTableName As String
Dim strConnect As String
Dim varRet As Variant
Dim con As ADODB.Connection
Dim rst As ADODB.Recordset
Dim sSQL As String
If IsMissing(bForceReconnect) Then
bForceReconnect = False
End If
sSQL = "SELECT rtblODBC.LocalTableName, MSysObjects.Name, MSysObjects.ForeignName, rtblODBC.SourceTableName, MSysObjects.Connect, rtblODBC.ConnectString " _
& "FROM MSysObjects RIGHT JOIN rtblODBC ON MSysObjects.Name = rtblODBC.LocalTableName " _
& "WHERE (((rtblODBC.ConnectString)<>'ODBC;' & [Connect]));"
Set con = Access.CurrentProject.Connection
Set rst = New ADODB.Recordset
rst.Open sSQL, con, adOpenDynamic, adLockOptimistic
'Test the recordset to see if any tables in rtblODBC (needed tables) are missing from the MSysObjects (actual tables)
If rst.BOF And rst.EOF And bForceReconnect = False Then
'No missing tables identified
fnReconnectODBC = True
Else
'Table returned information, we don't have a perfect match, time to relink
Set db = CurrentDb
Set rs = db.OpenRecordset("rtblODBC", dbOpenSnapshot)
'For each table definition in the database collection of tables
For Each tdf In db.TableDefs
'Set strConnect variable to table connection string
strConnect = tdf.Connect
If Len(strConnect) > 0 And Left(tdf.Name, 1) <> "~" Then
If Left(strConnect, 4) = "ODBC" Then
'If there is a connection string, and it's not a temp table, and it IS an odbc table
'Delete the table
DoCmd.DeleteObject acTable, tdf.Name
End If
End If
Next
'Relink tables from rtblODBC
With rs
.MoveFirst
Do While Not .EOF
Set tdf = db.CreateTableDef(!localtablename, dbAttachSavePWD, !SourceTableName, !ConnectString)
varRet = SysCmd(acSysCmdSetStatus, "Relinking '" & !SourceTableName & "'")
db.TableDefs.Append tdf
db.TableDefs.Refresh
If Len(!PrimaryKeyName & "") > 0 And Len(!PrimaryKeyField & "") > 0 Then
sPrimaryKeyName = !PrimaryKeyName
sPrimaryKeyField = !PrimaryKeyField
sLocalTableName = !localtablename
db.Execute "CREATE INDEX " & sPrimaryKeyName & " ON " & sLocalTableName & "(" & sPrimaryKeyField & ")WITH PRIMARY;"
End If
db.TableDefs.Refresh
.MoveNext
Loop
End With
subTurnOffSubDataSheets
fnReconnectODBC = True
End If
rst.Close
Set rst = Nothing
con.Close
Set con = Nothing
Exit_fnReconnectODBC:
Set tdf = Nothing
Set rs = Nothing
Set db = Nothing
varRet = SysCmd(acSysCmdClearStatus)
Exit Function
Err_fnReconnectODBC:
fnReconnectODBC = False
sPrompt = "Press OK to continue."
vbMsg = MsgBox(sPrompt, vbOKOnly, "Error Reconnecting")
If vbMsg = vbOK Then
Resume Exit_fnReconnectODBC
End If
End Function
Другие советы
Значительная часть кода без DSN, который повторно связывает таблицы access с SQL server, часто сначала удаляет ссылки, а затем создает их заново.Затем код настраивает строку подключения.Таким образом, именно удаление приводит к тому, что вы теряете то, чем был / является первичный ключ.
На самом деле я рекомендую вам изменить свой код повторной ссылки, чтобы не удалять ссылки на таблицы.
Попробуйте что-то вроде:
For Each tdfCurrent In dbCurrent.TableDefs
If Len(tdfCurrent.Connect) > 0 Then
If Left$(tdfCurrent.Connect, 5) = "ODBC;" Then
strCon = "ODBC;DRIVER={sql server};" & _
"SERVER=" & ServerName & ";" & _
"DATABASE=" & DatabaseName & ";" & _
"UID=" & UserID & ";" & _
"PWD=" & USERpw & ";" & _
"APP=Microsoft Office 2003;" & _
"WSID=" & WSID & ";"
End If
End If
tdfCurrent.Connect = strCon
tdfCurrent.RefreshLink
End If
Next tdfCurrent
Для меня это работает немного лучше (обратите внимание на перенесенный конец, если):
Dim dbCurrent As Database
Set dbCurrent = CurrentDb()
StatusList.SetFocus
StatusList.AddItem ("starting... ")
I = DoEvents()
Dim tdfCurrent As DAO.TableDef
For Each tdfCurrent In dbCurrent.TableDefs
If Len(tdfCurrent.Connect) > 0 Then
If Left$(tdfCurrent.Connect, 5) = "ODBC;" Then
strCon = "ODBC;DRIVER={sql server};" & _
"SERVER=" & ServerName & ";" & _
"DATABASE=" & DatabaseName & ";" & _
"UID=" & UserID & ";" & _
"PWD=" & USERpw & ";" & _
"APP=Microsoft Office 2003;" & _
"WSID=" & WSID & ";"
StatusList.AddItem ("fixing " & tdfCurrent.Name)
tdfCurrent.Connect = strCon
tdfCurrent.RefreshLink
End If
End If
I = DoEvents()
Next tdfCurrent
StatusList.AddItem ("----Done.")
Проверка ODBC верна, даже несмотря на то, что часть "ODBC;" не отображается в представлении MSysObjects.