问题描述:数据库B需要数据库A中的某些详细信息,且需要定时更新数据库B的数据,最好能做到实时更新
整体方法描述:通过webService来实现。首先是CLIENT端发送请求到SERVICE端,收到请求后将需要的数据全部打包发送给CLIENT端,CLIENT端收到数据
后插入或更新目标数据库的临时表中,临时表通过触发器来更新或插入到表中,同时更新源数据库中临时表的时间字段信息。
具体方法如下:
1、首先在数据库A中通过建立视图来查找出数据库B需要的字段信息。
此时需注意字段结构要一样。
2、在源数据库A中建立一张临时表
create table t_temp(
t_temp_id number(18) primary key not null,
updatetime datetime,
tablename varchar2(20),
status char(1)
);
3、service 端传送数据方法,为确保数据的安全性,在传送数据的过程中需要对数据进行压缩传送方法
数据打包压缩成一个DataSet后发送给CLIENT 端
数据压缩与解压缩方法:
public class ZipUnZip
{
public static byte[] ZipData(DataSet ds)
{
byte[] zipData = null;
//DataSet ds = LoadUserInfo(q).Copy();
ds.RemotingFormat = SerializationFormat.Binary;
BinaryFormatter ser = new BinaryFormatter();
MemoryStream unMS = new MemoryStream();
ser.Serialize(unMS, ds);
byte[] bytes = unMS.ToArray();
int lenbyte = bytes.Length;
MemoryStream compMs = new MemoryStream();
GZipStream compStream = new GZipStream(compMs, CompressionMode.Compress, true);
compStream.Write(bytes, 0, lenbyte);
compStream.Close();
unMS.Close();
compMs.Close();
zipData = compMs.ToArray();
return zipData;
}
public static DataSet UnzipData(byte[] da)
{
DataSet ds = new DataSet();
try
{
//WS.Service1 wss = new WSZipDemo.WS.Service1();//WebReference
//byte[] da = wss.getZipData();
MemoryStream input = new MemoryStream();
input.Write(da, 0, da.Length);
input.Position = 0;
GZipStream gzip = new GZipStream(input, CompressionMode.Decompress, true);
MemoryStream output = new MemoryStream();
byte[] buff = new byte[4096];
int read = -1;
read = gzip.Read(buff, 0, buff.Length);
while (read > 0)
{
output.Write(buff, 0, read);
read = gzip.Read(buff, 0, buff.Length);
}
gzip.Close();
byte[] rebytes = output.ToArray();
output.Close();
input.Close();
MemoryStream ms = new MemoryStream(rebytes);
BinaryFormatter bf = new BinaryFormatter();
object obj = bf.Deserialize(ms);
ds = (DataSet)obj;
}
catch (Exception ex)
{
//MessageBox.Show(ex.Message);
}
return ds;
}
}
4、CLIENT端收到数据后,对目标数据库中的临时表进行操作(临时表的结构与正式表结构一样),将数据全部插入到临时表中,
插入数据到临时表中时, 最好先清空临时表中的信息。(truncate table table_name;)
ds是SERVICE端传送过来的DataSet数据集.
CLIENT 端可调用方法:Boolean b = oracleHelper.GetSqlBulkCopy(ds, oracleConn);
public class OracleHelper
{
/// <summary>
/// 大批量入库处理
/// </summary>
/// <param name="dtData"></param>
/// <returns></returns>
public Boolean GetSqlBulkCopy(DataSet dsData, OracleConnection conn)
{
Boolean bolReturn = false;
foreach (DataTable dtData in dsData.Tables)
{
bolReturn = GetSqlBulkCopyByTable(dtData, conn);
}
return bolReturn;
}
public Boolean GetSqlBulkCopyByTable(DataTable dtData, OracleConnection oracleConn)
{
int BulkCopySize = 500;
int BulkCopyTimeOut = 1000 * 60 * 60 * 60;
Boolean bolReturn = false;
OracleBulkCopy bulkCopy = null;
try
{
bulkCopy = new OracleBulkCopy(oracleConn);
bulkCopy.DestinationTableName = dtData.TableName + "_TEMP";
bulkCopy.BatchSize = BulkCopySize;
bulkCopy.BulkCopyOptions = OracleBulkCopyOptions.UseInternalTransaction;
bulkCopy.BulkCopyTimeout = BulkCopyTimeOut;
bulkCopy.WriteToServer(dtData);
bolReturn = true;
}
catch
{
bolReturn = false;
throw;
}
finally
{
bulkCopy = null;
}
return bolReturn;
}
}
5、通过临时表触发器来更新或插入目标数据库B中的数据
触发器格式案例如下:
CREATE OR REPLACE TRIGGER T_TEMP_trg
AFTER INSERT OR UPDATE ON T_TEMP
FOR EACH ROW
DECLARE
PRAGMA AUTONOMOUS_TRANSACTION;
resultCount number :=0;
maxID number :=0;
BEGIN
-- 查询是否有记录 以此判断是否更新插入
SELECT count(*) INTO resultCount FROM T_TDJY WHERE YWLSH = :NEW.YWLSH AND YXTYWLSH = :NEW.YXTYWLSH AND BDBH =
:NEW.BDBH AND BJJE =:NEW.BJJE;
--下一条记录的ID
SELECT max(T_TDJY_ID) INTO maxID FROM T_TDJY;
--maxID :=maxID+1;
IF(resultCount=0) THEN
INSERT INTO T_TDJY VALUES(
nvl(maxID,0)+1,
:NEW.YWLSH,
:NEW.BBH,
:NEW.XZQH,
:NEW.YXTYWLSH,
:NEW.JMRBH,
:NEW.BDBH,
:NEW.JMRIP,
:NEW.BJJE
);
ELSE
UPDATE T_TDJY SET
YWLSH= :NEW.YWLSH,
BBH= :NEW.BBH,
XZQH= :NEW.XZQH,
YXTYWLSH=:NEW.YXTYWLSH,
JMRBH= :NEW.JMRBH,
BDBH= :NEW.BDBH,
JMRIP= :NEW.JMRIP,
BJJE= :NEW.BJJE
WHERE YWLSH = :NEW.YWLSH AND YXTYWLSH = :NEW.YXTYWLSH AND BDBH =:NEW.BDBH AND BJJE =:NEW.BJJE;
END IF;
-- DELETE FROM T_TDJY;
commit;
END;
6、如果不进行一定的处理时,在通过临时表的触发器来进行插入或更新数据目标数据库数据时,会出现 :直接路径不支持使用触发器的错误。
因为OracleBulkCopy类不支持事务处理。
解决方法:
1、将源数据库中的数据插入到目标数据库临时表中时,先将目标数据库中临时表的触发器禁用掉,然后调用下面方法:
Boolean b = oracleHelper.GetSqlBulkCopy(ds, oracleConn);
将数据全部插入到目标数据库中,启用触发器,然后通过更新临时表中的字段(doupdate)来触发。此时需要在源数据库A的视图和目标数据库B
的临时表中增加一个临时的字段(doupdate)。
禁用和启用触发器的存储过程方法如下:
create or replace procedure Trigger_OnOff(p_OnOff number) is
cursor c is select TABLE_NAME from user_all_tables WHERE TABLE_NAME LIKE '%_TEMP';
tablename varchar2(100);
sqlstr varchar2(1000);
begin
open c;
loop
fetch c into tablename;
exit when c%notfound;
if p_OnOff=0 then --0禁用 1启用
sqlstr:='alter table '||tablename||' disable all triggers';
else
sqlstr:='alter table '||tablename||' enable all triggers';
end if;
execute immediate sqlstr;
-- dbms_output.put_line(sqlstr);
end loop;
close c;
end;
CLIENT端调用存储过程来禁用触发器
OracleCommand OracleCmd2 = new OracleCommand("TRIGGER_ONOFF", oracleConn);
OracleCmd2.CommandType = CommandType.StoredProcedure;
OracleParameter id = OracleCmd2.Parameters.Add("@b", OracleDbType.Int32);
id.Value = 0;
OracleCmd2.ExecuteNonQuery();
//调用存储过程来启用触发器
OracleCommand OracleCmd3 = new OracleCommand("TRIGGER_ONOFF", oracleConn);
OracleCmd3.CommandType = CommandType.StoredProcedure;
OracleParameter id1 = OracleCmd3.Parameters.Add("@b", OracleDbType.Int16);
id1.Value = 1;
OracleCmd3.ExecuteNonQuery();
7、CLIENT端添加定时器
private System.Timers.Timer timer1;
timer1 = new System.Timers.Timer(1000 * 60 * 2);
timer1.Elapsed += new System.Timers.ElapsedEventHandler(Timer1Action);
timer1.AutoReset = true;
timer1.Enabled = true;public void Timer1Action(object sender, System.Timers.ElapsedEventArgs e)
{
//事务处理方法
}