网站导航免费论文 原创论文 论文搜索 原创论文 网学软件 学术大家 资料中心 会员中心 问题解答 原创论文 大学论文导航 设计下载 最新论文 下载排行 原创论文 论文源代码
返回网学首页
网学联系
最新论文 推荐专题 热门论文 素材专题
当前位置: 网学 > 编程文档 > ASP.net > 正文

.net多语言和数据集内多数据表的处理(3)

来源:http://myeducs.cn 联系QQ:点击这里给我发消息 作者: 用户投稿 来源: 网络 发布时间: 14/02/27

文章导读:在新的一年中,各位网友都进入紧张的学习或是工作阶段。网学的各位小编整理了ASP.net-.net多语言和数据集内多数据表的处理(3)的相关内容供大家参考,祝大家在新的一年里工作和学习顺利!

3 考虑以上两种情况的综合
当以上两种情况凑在一块的时候,情况还会复杂一些,因为在我们的这个解决方案中,多语言和信息的主体是采用的松耦合,如果不采用松耦合就不能保证其通用性和可扩展性,但是采用了松耦合在数据集中多表操作时又会产生麻烦。
因为松耦合,所以在数据集中自动级连更新的时候并不能够自动更新,修改还无所谓,我们只要保证和多语言表关联的那个Guid不变就可以了,但是删除呢?我们总不能把信息主体删除了却又把多语言数据留着吧,更麻烦的是因为数据已经删除,我们很难知道删除之前与多语言数据表关联的Guid是多少。而且我们需要把对信息主体的删除和多语言数据的删除放到一个事务中,相信谁都不会希望程序偶然出错信息主体未被删除的时候多语言数据都被删除了。而且,我们还要实现前面曾说过的一个目标:每个用户都只需要维护自己用的这种语言的记录信息就可以了,而不用考虑其他语言的问题,也可以非常方便的即使系统运行了一段时间后再次添加支持的语言,不需要在添加记录的事后就需要添加所有语言的版本,只在需要的时候才添加相应语言的版本,从而使数据库记录数尽量的少。
好了,我们知道了需求,然后该怎么做呢?看下面的代码:
在从数据库获取数据时:
public AddressData GetAddress(string languageCode)
{
AddressData ds = new AddressData();

SqlDataAdapter DARegion = new SqlDataAdapter(AddressSQL.strGetRegion,SQLConfig.DataBaseConnection);
DARegion.Fill(ds.Region);

SqlDataAdapter DACountry = new SqlDataAdapter(AddressSQL.strGetCountry,SQLConfig.DataBaseConnection);
DACountry.Fill(ds.Country);

SqlDataAdapter DAProvince = new SqlDataAdapter(AddressSQL.strGetProvince,SQLConfig.DataBaseConnection);
DAProvince.Fill(ds.Province);

SqlDataAdapter DACity = new SqlDataAdapter(AddressSQL.strGetCity,SQLConfig.DataBaseConnection);
DACity.Fill(ds.City);

SqlDataAdapter DAPort = new SqlDataAdapter(AddressSQL.strGetPort,SQLConfig.DataBaseConnection);
DAPort.Fill(ds.Port);

RegionTypeData regionTypeDS = GetRegionType(languageCode);

foreach ( AddressData.RegionRow region in ds.Region)
{
if (!region.IsNameGuidNull())
{
region.Name = DBDisplayString.GetDisplay(region.NameGuid,languageCode,vDisplayTable);
}
if (regionTypeDS.RegionType.FindByRegionTypeID(region.RegionTypeID) != null)
{
region.RegionTypeName = regionTypeDS.RegionType.FindByRegionTypeID(region.RegionTypeID).Abbr;
}
}

foreach ( AddressData.CountryRow country in ds.Country)
{
if (!country.IsNameGuidNull())
{
country.Name = DBDisplayString.GetDisplay(country.NameGuid,languageCode,vDisplayTable);
}
}

foreach (AddressData.ProvinceRow province in ds.Province)
{
if (!province.IsNameGuidNull())
{
province.Name = DBDisplayString.GetDisplay(province.NameGuid,languageCode,vDisplayTable);
}
}

foreach (AddressData.CityRow city in ds.City)
{
if (!city.IsNameGuidNull())
{
city.Name = DBDisplayString.GetDisplay(city.NameGuid, languageCode,vDisplayTable);
}
}

foreach (AddressData.PortRow port in ds.Port)
{
if (!port.IsNameGuidNull())
{
port.Name = DBDisplayString.GetDisplay(port.NameGuid,languageCode,vDisplayTable);
}
}

return ds;
}
这样,只需要通过一个单一的Name字段便可以获取用户所用语言的数据了,而不用考虑用户使用的到底是哪一种语言。这里不在数据库里直接做join的原因是:这个获取多语言数据的方法并不是直接把数据库资料取出来那么简单,假如用户要取的那种语言的某条资料恰好没有呢?我们取了一个系统默认语言的数据,那么哪一种是系统默认语言呢?可见如果这些都放在SQL中做的话,要传入的参数会非常多,SQL语句会变得非常复杂,也就不利于封装了,而且考虑到SQL语句调试的复杂性,我没有选择这么做。
那么把数据更新回数据库又该怎么办呢?看下面的代码:
private void DeleteDisplay(AddressData ds, Guid refGuid)
{
foreach (AddressData.AddressDisStrRow dis in ds.AddressDisStr.Select("RefGuid = ''" +refGuid+ "''"))
{
dis.Delete();
}
}
private void DeletePort(AddressData ds, int portID)
{
if (!ds.Port.FindByPortID(portID).IsNameGuidNull())
{
DeleteDisplay(ds,ds.Port.FindByPortID(portID).NameGuid);
}
ds.Port.FindByPortID(portID).Delete();
}

private void DeleteCity(AddressData ds, int cityID)
{
if (!ds.City.FindByCityID(cityID).IsNameGuidNull())
{
DeleteDisplay(ds,ds.City.FindByCityID(cityID).NameGuid);
}
foreach (AddressData.PortRow port in ds.Port)
{
if ((int)port["CityID",DataRowVersion.Original] == cityID)
{
port.RejectChanges();
DeletePort(ds,port.PortID);
}
}
ds.City.FindByCityID(cityID).Delete();
}

private void DeleteProvince(AddressData ds, int provinceID)
{
if (!ds.Province.FindByProvinceID(provinceID).IsNameGuidNull())
{
DeleteDisplay(ds,ds.Province.FindByProvinceID(provinceID).NameGuid);
}
foreach (AddressData.CityRow city in ds.City)
{
if ((int)city["ProvinceID",DataRowVersion.Original] == provinceID)
{
city.RejectChanges();
DeleteCity(ds,city.CityID);
}
}
ds.Province.FindByProvinceID(provinceID).Delete();
}
public void UpdateProvince(AddressData ds, string languageCode)
{
SqlDataAdapter DAAddressDisStr = new SqlDataAdapter(AddressSQL.strGetAddressDisStr,SQLConfig.DataBaseConnection);
DAAddressDisStr.Fill(ds.AddressDisStr);

foreach(AddressData.ProvinceRow province in ds.Province)
{
if (province.RowState == DataRowState.Deleted)
{
province.RejectChanges();
DeleteProvince(ds,province.ProvinceID);
}
else if (province.RowState == DataRowState.Added || province.RowState == DataRowState.Modified)
{
if (province.IsNameNull() && !province.IsNameGuidNull())
{
DBDisplayString.SaveDisplay(languageCode,province.NameGuid,null,vDisplayTable);
}
else if (!province.IsNameNull())
{
if (province.IsNameGuidNull())
{
province.NameGuid = Guid.NewGuid();
}
DBDisplayString.SaveDisplay(languageCode,province.NameGuid,province.Name,vDisplayTable);
}
}
}
DataTableExtend[] dts = new DataTableExtend[4];
dts[0] = new DataTableExtend(ds.Port,"Port");
dts[1] = new DataTableExtend(ds.City, "City");
dts[2] = new DataTableExtend(ds.Province,"Province");
dts[3] = new DataTableExtend(ds.AddressDisStr,vDisplayTable);
SQLModify.ModifyDataBase(dts);
}
可以看出,这个操作会很麻烦,对于删除操作而言,我们需要把该条记录所有的多语言数据都删掉,然后很不幸,对于数据集中已删除了的记录而言,虽然它还存在但却不能访问,更不幸的是,因为多数据表关联的原因,我们甚至都不知道有多少数据被级连删除了:(没办法,RejectChanges()吧,然后通过关联找到所有被级连删除的数据,把多语言数据都删除掉,然后再删除自身,注意到我这里用到了Select方法,这是一个很方便的做法,减少了到数据库的往返操作,从而以牺牲一些内存的代价节省了与数据库痛心的时间。对于新增和修改操作,我们则负担着另一项任务,即把Name字段里的数据根据用户的当前语言,更新会数据库相应的记录中,这里也会有很多种情况,如该字段数据是新增的,缺省语言的数据也没有,当前语言的数据也没有,或者是缺省语言的数据有了当前语言的数据却没有,再或者用户是把这个字段的数据清空了,我们必须要找到多语言表中相应的记录并删除掉……等等,无奈,我们只好让DBDisplayString.SaveDisplay()包办掉这一切。

网学推荐

免费论文

原创论文

设为首页 | 加入收藏 | 论文首页 | 论文专题 | 设计下载 | 网学软件 | 论文模板 | 论文资源 | 程序设计 | 关于网学 | 站内搜索 | 网学留言 | 友情链接 | 资料中心
版权所有 QQ:3710167 邮箱:3710167@qq.com 网学网 [Myeducs.cn] 您电脑的分辨率是 像素
Copyright 2008-2015 myeducs.Cn www.myeducs.Cn All Rights Reserved 湘ICP备09003080号