存在被降级的风险,我想问一下,使用C#,使用批次查看表中的数据的最佳机制(最好对于此处固有的实践违规显然是主观的)列。很多,我的意思是1000。
现在,在你点击所有点击之前,或者抛出一些回答,比如“你为什么会有一个包含那么多列的表”,让我说它实际上是设计要求的一部分。我们正在从1000个数据点尽快收集数据。我们需要尽可能快地存储这些,因此平台。数据需要直接从SQL Server访问,因此数据库(我们正在使用SQL Compact with table-direct)。
所以,让我们暂时忘记我们所学到的关于正确的数据库设计,规范化规则等等,并且只关注我有一个包含1000列的表的事实,我希望能够显示屏幕上的数据用于验证数据是否实际进入那里。
我尝试过数据网格。它是因为(毫不奇怪)它不是为处理那么多列而设计的。
我尝试在Studio中使用查看器。它在256之后呕吐,加上最终用户无论如何都不会安装Studio。
现在结果不一定非常好,它不需要是可更新的,也不需要对数据更改敏感 - 只是在给定时间点表中数据的静态快照。
相关(或半相关)信息:
答案 0 :(得分:15)
如果您要实现自己的自定义用户控件,可以像这样做一个鱼眼网格:
此示例显示了在9x10表格中移动的全尺寸3x4面板。由于(我假设)您不需要编辑此数据,因此UI可能只是用户抓取面板并拖动它的东西。如果你真的是自虐和/或有很多空闲时间,你甚至可以在同一个网格上有多个鱼眼面板,让你可以同时比较网格的一个或多个区域。
更新:Silverlight has one of these,显然。排序。
答案 1 :(得分:14)
您可以将所有数字格式化为带有空格的n个字符的字符串,然后以固定宽度的字体显示它们。
1 2 3 4 6 36 436 6346
2 3 4 6 36 436 6346 0
3 4 6 36 436 6346 3 4
4 6 36 436 6346 333 222 334
答案 2 :(得分:12)
好的,对我来说最合适的答案是使用ReportViewer control,但不是以MSDN中记录的任何方式。问题是我有动态数据,所以我需要一个动态报告,所有教程等似乎都假设您可以在设计时了解所有内容,这样您就可以通过向导指向并点击您的方式。 / p>
解决方案最终需要几件。首先,我必须创建代码来动态生成RDLC,ReportViewer用它来描述报告布局以及哪些数据字段映射到什么。这就是我想出的:
public static Stream BuildRDLCStream(
DataSet data, string name, string reportXslPath)
{
using (MemoryStream schemaStream = new MemoryStream())
{
// save the schema to a stream
data.WriteXmlSchema(schemaStream);
schemaStream.Seek(0, SeekOrigin.Begin);
// load it into a Document and set the Name variable
XmlDocument xmlDomSchema = new XmlDocument();
xmlDomSchema.Load(schemaStream);
xmlDomSchema.DocumentElement.SetAttribute("Name", data.DataSetName);
// load the report's XSL file (that's the magic)
XslCompiledTransform xform = new XslCompiledTransform();
xform.Load(reportXslPath);
// do the transform
MemoryStream rdlcStream = new MemoryStream();
XmlWriter writer = XmlWriter.Create(rdlcStream);
xform.Transform(xmlDomSchema, writer);
writer.Close();
rdlcStream.Seek(0, SeekOrigin.Begin);
// send back the RDLC
return rdlcStream;
}
}
第二部分是我从Dan Shipe's blog开始的XSL文件。那里的RDLC代码非常没用,因为它全部用于Web使用,但XSL是纯金。我已经把它放在这篇文章的底部,以便在博客离线时完整。
一旦我拥有这两个部分,只需要创建一个带有ReportViewer控件的Form,然后使用这段代码进行设置:
ds.DataSetName = name;
Stream rdlc = RdlcEngine.BuildRDLCStream(
ds, name, "c:\\temp\\rdlc\\report.xsl");
reportView.LocalReport.LoadReportDefinition(rdlc);
reportView.LocalReport.DataSources.Clear();
reportView.LocalReport.DataSources.Add(
new ReportDataSource(ds.DataSetName, ds.Tables[0]));
reportView.RefreshReport();
这里的关键是'ds'是一个DataSet对象,其中包含一个DataTable,其中包含要显示的数据。
同样,为了完整性,这里是XSL - 对不起大小:
<?xml version="1.0"?>
<!-- Stylesheet for creating ReportViewer RDLC documents -->
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:msxsl="urn:schemas-microsoft-com:xslt"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"
xmlns:rd="http://schemas.microsoft.com/SQLServer/reporting/reportdesigner" xmlns="http://schemas.microsoft.com/sqlserver/reporting/2005/01/reportdefinition"
>
<xsl:variable name="mvarName" select="/xs:schema/@Name"/>
<xsl:variable name="mvarFontSize">8pt</xsl:variable>
<xsl:variable name="mvarFontWeight">500</xsl:variable>
<xsl:variable name="mvarFontWeightBold">700</xsl:variable>
<xsl:template match="/">
<xsl:apply-templates select="/xs:schema/xs:element/xs:complexType/xs:choice/xs:element/xs:complexType/xs:sequence">
</xsl:apply-templates>
</xsl:template>
<xsl:template match="xs:sequence">
<Report xmlns:rd="http://schemas.microsoft.com/SQLServer/reporting/reportdesigner" xmlns="http://schemas.microsoft.com/sqlserver/reporting/2005/01/reportdefinition">
<BottomMargin>1in</BottomMargin>
<RightMargin>1in</RightMargin>
<LeftMargin>1in</LeftMargin>
<TopMargin>1in</TopMargin>
<InteractiveHeight>11in</InteractiveHeight>
<InteractiveWidth>8.5in</InteractiveWidth>
<Width>6.5in</Width>
<Language>en-US</Language>
<rd:DrawGrid>true</rd:DrawGrid>
<rd:SnapToGrid>true</rd:SnapToGrid>
<rd:ReportID>7358b654-3ca3-44a0-8677-efe0a55c7c45</rd:ReportID>
<xsl:call-template name="BuildDataSource">
</xsl:call-template>
<xsl:call-template name="BuildDataSet">
</xsl:call-template>
<Body>
<Height>0.50in</Height>
<ReportItems>
<Table Name="table1">
<DataSetName><xsl:value-of select="$mvarName" /></DataSetName>
<Top>0.5in</Top>
<Height>0.50in</Height>
<Header>
<TableRows>
<TableRow>
<Height>0.25in</Height>
<TableCells>
<xsl:apply-templates select="xs:element" mode="HeaderTableCell">
</xsl:apply-templates>
</TableCells>
</TableRow>
</TableRows>
</Header>
<Details>
<TableRows>
<TableRow>
<Height>0.25in</Height>
<TableCells>
<xsl:apply-templates select="xs:element" mode="DetailTableCell">
</xsl:apply-templates>
</TableCells>
</TableRow>
</TableRows>
</Details>
<TableColumns>
<xsl:apply-templates select="xs:element" mode="TableColumn">
</xsl:apply-templates>
</TableColumns>
</Table>
</ReportItems>
</Body>
</Report>
</xsl:template>
<xsl:template name="BuildDataSource">
<DataSources>
<DataSource Name="DummyDataSource">
<ConnectionProperties>
<ConnectString/>
<DataProvider>SQL</DataProvider>
</ConnectionProperties>
<rd:DataSourceID>84635ff8-d177-4a25-9aa5-5a921652c79c</rd:DataSourceID>
</DataSource>
</DataSources>
</xsl:template>
<xsl:template name="BuildDataSet">
<DataSets>
<DataSet Name="{$mvarName}">
<Query>
<rd:UseGenericDesigner>true</rd:UseGenericDesigner>
<CommandText/>
<DataSourceName>DummyDataSource</DataSourceName>
</Query>
<Fields>
<xsl:apply-templates select="xs:element" mode="Field">
</xsl:apply-templates>
</Fields>
</DataSet>
</DataSets>
</xsl:template>
<xsl:template match="xs:element" mode="Field">
<xsl:variable name="varFieldName">
<xsl:value-of select="@name" />
</xsl:variable>
<xsl:variable name="varDataType">
<xsl:choose>
<xsl:when test="@type='xs:int'">System.Int32</xsl:when>
<xsl:when test="@type='xs:string'">System.String</xsl:when>
<xsl:when test="@type='xs:dateTime'">System.DateTime</xsl:when>
<xsl:when test="@type='xs:boolean'">System.Boolean</xsl:when>
</xsl:choose>
</xsl:variable>
<Field Name="{$varFieldName}">
<rd:TypeName><xsl:value-of select="$varDataType"/></rd:TypeName>
<DataField><xsl:value-of select="$varFieldName"/></DataField>
</Field>
</xsl:template>
<xsl:template match="xs:element" mode="HeaderTableCell">
<xsl:variable name="varFieldName">
<xsl:value-of select="@name" />
</xsl:variable>
<TableCell>
<ReportItems>
<Textbox Name="textbox{position()}">
<rd:DefaultName>textbox<xsl:value-of select="position()"/>
</rd:DefaultName>
<Value><xsl:value-of select="$varFieldName"/></Value>
<CanGrow>true</CanGrow>
<ZIndex>7</ZIndex>
<Style>
<TextAlign>Center</TextAlign>
<PaddingLeft>2pt</PaddingLeft>
<PaddingBottom>2pt</PaddingBottom>
<PaddingRight>2pt</PaddingRight>
<PaddingTop>2pt</PaddingTop>
<FontSize><xsl:value-of select="$mvarFontSize"/></FontSize>
<FontWeight><xsl:value-of select="$mvarFontWeightBold"/></FontWeight>
<BackgroundColor>#000000</BackgroundColor>
<Color>#ffffff</Color>
<BorderColor>
<Default>#ffffff</Default>
</BorderColor>
<BorderStyle>
<Default>Solid</Default>
</BorderStyle>
</Style>
</Textbox>
</ReportItems>
</TableCell>
</xsl:template>
<xsl:template match="xs:element" mode="DetailTableCell">
<xsl:variable name="varFieldName">
<xsl:value-of select="@name" />
</xsl:variable>
<TableCell>
<ReportItems>
<Textbox Name="{$varFieldName}">
<rd:DefaultName><xsl:value-of select="$varFieldName"/></rd:DefaultName>
<Value>=Fields!<xsl:value-of select="$varFieldName"/>.Value</Value>
<CanGrow>true</CanGrow>
<ZIndex>7</ZIndex>
<Style>
<TextAlign>Left</TextAlign>
<PaddingLeft>2pt</PaddingLeft>
<PaddingBottom>2pt</PaddingBottom>
<PaddingRight>2pt</PaddingRight>
<PaddingTop>2pt</PaddingTop>
<FontSize><xsl:value-of select="$mvarFontSize"/></FontSize>
<FontWeight><xsl:value-of select="$mvarFontWeight"/></FontWeight>
<BackgroundColor>#e0e0e0</BackgroundColor>
<Color>#000000</Color>
<BorderColor>
<Default>#ffffff</Default>
</BorderColor>
<BorderStyle>
<Default>Solid</Default>
</BorderStyle>
</Style>
</Textbox>
</ReportItems>
</TableCell>
</xsl:template>
<xsl:template match="xs:element" mode="TableColumn">
<TableColumn>
<Width>0.75in</Width>
</TableColumn>
</xsl:template>
<xsl:template name="replace-string">
<xsl:param name="text"/>
<xsl:param name="from"/>
<xsl:param name="to"/>
<xsl:choose>
<xsl:when test="contains($text, $from)">
<xsl:variable name="before" select="substring-before($text, $from)"/>
<xsl:variable name="after" select="substring-after($text, $from)"/>
<xsl:variable name="prefix" select="concat($before, $to)"/>
<xsl:value-of select="$before"/>
<xsl:value-of select="$to"/>
<xsl:call-template name="replace-string">
<xsl:with-param name="text" select="$after"/>
<xsl:with-param name="from" select="$from"/>
<xsl:with-param name="to" select="$to"/>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$text"/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
答案 3 :(得分:6)
如何将数据存储在csv文件中,这样可以为您提供查看选项。如果您的用户有excel或Open Office Calc,他们可以轻松导入数据(不确定Calc是否有列限制,但excel 2007可以容纳16384列)并通过该程序查看它?
答案 4 :(得分:3)
你需要在一张桌子上查看多行吗?
我的猜测是这个数据是数字的,有没有什么方法可以将单行数据显示为20 * 50网格或类似的东西,然后只是通过行分页?
例如,第1行,第1列=数据库的第1列,第2行,第1列=数据库的第21列等
Id = 1
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
----|--------------------------------------------------------
0 |
20 |
40 |
60 |
80 |
100 |
120 |
etc |
答案 5 :(得分:2)
尝试使用非常小的字体
的html表如果您担心格式化表格,请使用CSS:
td { font-size: 0.2em; text-align: right; }
或者,如果你的所有数字都是相同的大小,你也可以只生成一个“数字墙”显示,例如使用固定宽度的字体并在滚动面板中显示5个字符宽的列
答案 6 :(得分:1)
DataGrid(甚至是ListView)应该能够处理一个包含32列和32行的表,这样您就可以一次显示整个数据库行的数据。这样您就可以立即查看某些单元格是否缺少数据。
答案 7 :(得分:1)
这取决于它需要多漂亮。如果这只是一个调试/点检查工具,你可以并排放置几个DataGrids,每个DataGrids都显示一列选择。会有点丑,但是可行。
OTOH,如果你需要一个半抛光工具,你可能想要一个自定义控件来处理它。基本上,您将使用一些缓冲区加载正在查看的数据库部分,当用户滚动当前加载的数据时,运行新查询。答案 8 :(得分:0)
鉴于用户无论如何都必须水平滚动,您可以使用显示合理列数(例如50)的常规数据网格。然后,您在网格下方有一个水平滚动条,用于选择要显示的列的子集。当滚动条在左侧时,您会显示1-50列,当您单击右箭头时,您会转到2-51,等等。
这为您提供了滚动功能,而无需使用数据重载网格控件。虽然您将失去在表格中自由光标或进行大矩形选择的能力,但听起来不会对此应用程序造成问题。
答案 9 :(得分:0)
..来验证数据是否实际进入那里。
可能它已经过时了,但您可以使用像素地图,其中单个像素代表表格的单个单元格(屏幕大于1000)或单个像素代表10个单元格,点击缩放区域。
像素的颜色将取决于数据。空/数据可能是黑/白。或者它可能是颜色显示值随着每一行增长或减少。或者红色表示数据突然跳跃。您可以在数据网格中正常捕捉所有异常情况。
然后您需要的是捕获感兴趣区域中的点击坐标,并使用小表来显示该部分表格而不进行任何滚动。
只需点击即可返回像素图。
答案 10 :(得分:0)
我建议调查平面布局以外的其他内容。根据我的经验,数据库对列数和行字节大小有限制。
每个数据库实现都有一个页面大小(4k / 8k),并且单个行必须符合此数据大小。 NULL通常是免费赠品。这意味着1000英寸1000 x 4字节将只适合4k页面大小。
如果您正在与varchars讨论数据,那么问题就更糟了。每列中有多少个字符?可以填写多少列?如果您平均有10个字符,并且页面大小为8k,则会因SQL错误而丢失数据。
如果你必须笑,但这种情况确实发生在我知道正在推动极限的扁平数据表中特别长的啰嗦打字员。
答案 11 :(得分:0)
从斜角开始,我会询问用户是否需要一次“加载”所有列?
如果用户愿意一次显示一列子列(例如,一次显示100个列,或者一次显示一个特定的列),那么我会使用某种数据网格(内置的一个) ,或者ListView,或者可能是第三方)来显示子集,CheckedListView停靠在一边,允许显示感兴趣的子集。
或者,您是否可以显示某种类型的摘要数据,显示100列组的计数/平均值/ xxx?
答案 12 :(得分:0)
如果您刚刚进行验证,则无法以编程方式检查每个字段并报告整行是否正常!然后你需要一个非常简单的数据网格,列出那些不太好的行。
然后可以通过您可以应用于单行的任何技术来检查它们,因为在大多数情况下您不需要浏览字段。我在这里假设您可以以某种方式查看整行,并且正在寻找同时浏览多行以查找缺失数据的方法(自动化这将使其更加可靠)。
答案 13 :(得分:0)
我会把它作为深入研究。在第一页(或页面顶部),您将拥有选择行的控件。在下一页(或页面底部)中,您将显示所选行中的数据。例如,根据所需的单元格宽度,您可以将此行设置为100行10列,或1000行1列。
这可以很容易地做为动态客户端javascript - 您甚至可以通过这种方式使其可编辑。我不确定这在C#中是如何工作的。
答案 14 :(得分:0)
也许你应该研究一种不同类型的数据库。我听说面向列的数据库很适合这种事情(而典型的RDBMS是面向行的)。另外,如果您在第一次插入行后不再更新行,那么二进制平面文件可能比巨型表更好吗?
答案 15 :(得分:0)
谁将阅读1000列表???试着想一想过滤或可视化数据的方法。
答案 16 :(得分:0)
即使建议这样做,我也觉得很脏,但你可以做一些事情:
SELECT Field1 + ' - ' + Field2 + ... AS EvilMegaColumn FROM Table
但实际上我认为这属于“如果你遇到这个限制,你做错了什么”的类别。我真的看不出任何理由,速度或其他需要1000列......
答案 17 :(得分:-1)
如果您只需确保填充数据,那么为什么不让每列都有默认值,例如'void','blank'等。
然后你可以在计算非默认值/总数时进行迭代以显示百分比。
现在,您可以使用百分比值可视化数据完整性,甚至可以记录哪些列具有默认值(如列表/数组)以供进一步调查。
答案 18 :(得分:-1)
您可以考虑与您的用户群进行核对并查看他们真正需要查看的内容,然后针对每个不同的需求设置视图,以便减少列数。
另一个选择是读取数据,并从中创建一个高大的静态html页面集。然后,您可以从程序中调用浏览器来查看它。
答案 19 :(得分:-1)
具有可滚动窗格并一次显示10列(可以主动加载或缓存这些列或任何您需要的列)。向左滚动时,显示前十个。向右滚动时,显示后面的列序列。总而言之,在任何给定点上只有10列有效。在我看来,尝试实际显示1000列将是任何其他方式。 PS:这只不过是一个理想的猜测;我不确定它是否可以远程实现。
答案 20 :(得分:-1)
有多少数据对初始视图至关重要?我可以看到做类似主/细节类型网格的事情,你将关键列(比如10)放到数据网格上,当用户点击查看细节时,你可以取剩下的列并将它们显示在“属性区域“或那方面的东西。