统计图形种类繁多, 有柱状图, 折线图, 扇形图等等, 而统计图形的绘制方法也有很多, 有Flash制作的统计图形, 有水晶报表生成统计图形, 有专门制图软件制作, 也有编程语言自己制作的;这里我们用就C# 制作三款最经典的统计图: 柱状图, 折线图和扇形图;既然是统计, 当然需要数据, 这里演示的数据存于Sql Server2000中, 三款统计图形都是动态生成. 其中柱状图我会附上制作步骤, 其他两款统计图直接附源码.
说明: 需求不一样, 统计图形绘制后的显示效果也不一样, 比如这里柱状图的主要需求是为了比较每一期报名人数与通过人数的差, 因此会把两根柱子放在一起会使比较结果一目了然. 因此大家可以根据需要灵活绘制.
一. 柱状图的绘制.
绘制步骤如下:
1. 定义绘图用到的类.
int height = , width = ;
Bitmap image = new Bitmap(width, height);
Graphics g = Graphics.FromImage(image);
Pen mypen = new Pen(brush, );
2. 绘制图框.
g.FillRectangle(Brushes.WhiteSmoke, , , width, height);
3. 绘制横向坐标线
for (int i = ; i < ; i++)
{
g.DrawLine(mypen, x, , x, );
x = x + ;
}
4. 绘制纵向坐标线
for (int i = ; i < ; i++)
{
g.DrawLine(mypen, , y, , y);
y = y + ;
}
5. 绘制横坐标值
String[] n = { "第一期", "第二期", "第三期", "第四期", "全年" };
for (int i = ; i < ; i++)
{
g.DrawString(n[i].ToString(), font, Brushes.Blue, x, );
x = x + ;
}
6. 绘制纵坐标值
String[] m = {"","", "", "", "", "", "100“};
for (int i = ; i < ; i++)
{
g.DrawString(m[i].ToString(), font, Brushes.Blue, , y);
y = y + ;
}
7. 定义数组存储数据库中统计的数据
int[] Count1 = new int[]; //存储从数据库读取的报名人数
int[] Count2 = new int[]; //存储从数据库读取的通过人数
8. 从数据库中读取报名人数与通过人数
SqlConnection Con = new SqlConnection(
"Server=(Local);Database=committeeTraining;");
Con.Open();
string cmdtxt2 = "SELECT * FROM ##Count
where Company='" + ****+ "'";
SqlDataAdapter da = new SqlDataAdapter(cmdtxt2, Con);
DataSet ds = new DataSet();
da.Fill(ds);
9. 将读取的数据存储到数组中
Count1[] = Convert.ToInt32(ds.Tables[].Rows[][“count1”].ToString());
Count1[] = Convert.ToInt32(ds.Tables[].Rows[][“count3”].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[][“count2”].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count4"].ToString());
10.定义画笔和画刷准备绘图
x = ;
Font font2 = new System.Drawing.Font(
"Arial", , FontStyle.Bold);
SolidBrush mybrush = new SolidBrush(Color.Red);
SolidBrush mybrush2 = new SolidBrush(Color.Green);
11. 根据数组中的值绘制柱状图
()第一期报名人数
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2,
Brushes.Red, x, - Count1[] - ); () 第一期通过人数
x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2,
Brushes.Green, x, - Count2[] - );
12. 将图形输出到页面.
System.IO.MemoryStream ms = new
System.IO.MemoryStream();
image.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg);
Response.ClearContent();
Response.ContentType = "image/Jpeg";
Response.BinaryWrite(ms.ToArray());
最终柱状图的效果图:
柱状图的完整代码:
private void CreateImage()
{
int height = , width = ;
Bitmap image = new Bitmap(width, height);
//创建Graphics类对象
Graphics g = Graphics.FromImage(image); try
{
//清空图片背景色
g.Clear(Color.White); Font font = new Font("Arial", , FontStyle.Regular);
Font font1 = new Font("宋体", , FontStyle.Bold); LinearGradientBrush brush = new LinearGradientBrush(new Rectangle(, , image.Width, image.Height),
Color.Blue, Color.BlueViolet, 1.2f, true);
g.FillRectangle(Brushes.WhiteSmoke, , , width, height);
// Brush brush1 = new SolidBrush(Color.Blue); g.DrawString(this.ddlTaget.SelectedItem.Text + " " + this.ddlYear.SelectedItem.Text +
" 成绩统计柱状图", font1, brush, new PointF(, ));
//画图片的边框线
g.DrawRectangle(new Pen(Color.Blue), , , image.Width - , image.Height - ); Pen mypen = new Pen(brush, );
//绘制线条
//绘制横向线条
int x = ;
for (int i = ; i < ; i++)
{
g.DrawLine(mypen, x, , x, );
x = x + ;
}
Pen mypen1 = new Pen(Color.Blue, );
x = ;
g.DrawLine(mypen1, x, , x, ); //绘制纵向线条
int y = ;
for (int i = ; i < ; i++)
{
g.DrawLine(mypen, , y, , y);
y = y + ;
}
g.DrawLine(mypen1, , y, , y); //x轴
String[] n = { "第一期", "第二期", "第三期", "第四期", "上半年", "下半年", "全年统计" };
x = ;
for (int i = ; i < ; i++)
{
g.DrawString(n[i].ToString(), font, Brushes.Blue, x, ); //设置文字内容及输出位置
x = x + ;
} //y轴
String[] m = {"","", "", "", "", "", "", "",
"", "", ""};
y = ;
for (int i = ; i < ; i++)
{
g.DrawString(m[i].ToString(), font, Brushes.Blue, , y); //设置文字内容及输出位置
y = y + ;
} int[] Count1 = new int[];
int[] Count2 = new int[]; SqlConnection Con = new SqlConnection("Server=(Local);Database=committeeTraining;Uid=sa;Pwd=**");
Con.Open();
string cmdtxt2 = "SELECT * FROM ##Count where Company='" + this.ddlTaget.SelectedItem.Text.Trim() + "'";
SqlDataAdapter da = new SqlDataAdapter(cmdtxt2, Con);
DataSet ds = new DataSet();
da.Fill(ds); Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count1"].ToString());
Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count3"].ToString());
Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count5"].ToString());
Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count7"].ToString()); Count1[] = Count1[] + Count1[];
Count1[] = Count1[] + Count1[]; Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count9"].ToString()); Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count2"].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count4"].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count6"].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count8"].ToString()); Count2[] = Count2[] + Count2[];
Count2[] = Count2[] + Count2[]; Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count10"].ToString()); //绘制柱状图.
x = ;
Font font2 = new System.Drawing.Font("Arial", , FontStyle.Bold);
SolidBrush mybrush = new SolidBrush(Color.Red);
SolidBrush mybrush2 = new SolidBrush(Color.Green); //第一期
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2, Brushes.Red, x, - Count1[] - ); x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2, Brushes.Green, x, - Count2[] - ); //第二期
x = x + ;
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2, Brushes.Red, x, - Count1[] - ); x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2, Brushes.Green, x, - Count2[] - ); //第三期
x = x + ;
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2, Brushes.Red, x, - Count1[] - ); x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2, Brushes.Green, x, - Count2[] - ); //第四期
x = x + ;
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2, Brushes.Red, x, - Count1[] - ); x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2, Brushes.Green, x, - Count2[] - ); //上半年
x = x + ;
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2, Brushes.Red, x, - Count1[] - ); x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2, Brushes.Green, x, - Count2[] - ); //下半年
x = x + ;
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2, Brushes.Red, x, - Count1[] - ); x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2, Brushes.Green, x, - Count2[] - ); //全年
x = x + ;
g.FillRectangle(mybrush, x, - Count1[], , Count1[]);
g.DrawString(Count1[].ToString(), font2, Brushes.Red, x, - Count1[] - ); x = x + ;
g.FillRectangle(mybrush2, x, - Count2[], , Count2[]);
g.DrawString(Count2[].ToString(), font2, Brushes.Green, x, - Count2[] - ); //绘制标识
Font font3 = new System.Drawing.Font("Arial", , FontStyle.Regular);
g.DrawRectangle(new Pen(Brushes.Blue), , , , ); //绘制范围框
g.FillRectangle(Brushes.Red, , , , ); //绘制小矩形
g.DrawString("报名人数", font3, Brushes.Red, , ); g.FillRectangle(Brushes.Green, , , , );
g.DrawString("通过人数", font3, Brushes.Green, , ); System.IO.MemoryStream ms = new System.IO.MemoryStream();
image.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg);
Response.ClearContent();
Response.ContentType = "image/Jpeg";
Response.BinaryWrite(ms.ToArray());
}
finally
{
g.Dispose();
image.Dispose();
}
}
二. 折线统计图的绘制
效果:
折线图的完整代码:
private void CreateImage()
{
int height = , width = ;
Bitmap image = new Bitmap(width, height);
Graphics g = Graphics.FromImage(image); try
{
//清空图片背景色
g.Clear(Color.White); Font font = new System.Drawing.Font("Arial", , FontStyle.Regular);
Font font1 = new System.Drawing.Font("宋体", , FontStyle.Regular);
Font font2 = new System.Drawing.Font("Arial", , FontStyle.Regular);
LinearGradientBrush brush = new LinearGradientBrush(
new Rectangle(, , image.Width, image.Height), Color.Blue, Color.Blue, 1.2f, true);
g.FillRectangle(Brushes.AliceBlue, , , width, height);
Brush brush1 = new SolidBrush(Color.Blue);
Brush brush2 = new SolidBrush(Color.SaddleBrown); g.DrawString(this.ddlTaget.SelectedItem.Text + " " + this.ddlYear.SelectedItem.Text +
" 成绩统计折线图", font1, brush1, new PointF(, ));
//画图片的边框线
g.DrawRectangle(new Pen(Color.Blue), , , image.Width - , image.Height - ); Pen mypen = new Pen(brush, );
Pen mypen2 = new Pen(Color.Red, );
//绘制线条
//绘制纵向线条
int x = ;
for (int i = ; i < ; i++)
{
g.DrawLine(mypen, x, , x, );
x = x + ;
}
Pen mypen1 = new Pen(Color.Blue, );
x = ;
g.DrawLine(mypen1, x, , x, ); //绘制横向线条
int y = ;
for (int i = ; i < ; i++)
{
g.DrawLine(mypen, , y, , y);
y = y + ;
}
// y = 106;
g.DrawLine(mypen1, , y - , , y - ); //x轴
String[] n = { "第一期", "第二期", "第三期", "第四期", "上半年", "下半年", "全年统计" };
x = ;
for (int i = ; i < ; i++)
{
g.DrawString(n[i].ToString(), font, Brushes.Red, x, ); //设置文字内容及输出位置
x = x + ;
} //y轴
String[] m = { "220人", " 200人", " 175人", "150人", " 125人", " 100人", " 75人", " 50人",
" 25人"};
y = ;
for (int i = ; i < ; i++)
{
g.DrawString(m[i].ToString(), font, Brushes.Red, , y); //设置文字内容及输出位置
y = y + ;
} int[] Count1 = new int[];
int[] Count2 = new int[]; SqlConnection Con = new SqlConnection("Server=(Local);Database=committeeTraining;Uid=sa;Pwd=eesoft");
Con.Open();
string cmdtxt2 = "SELECT * FROM ##Count where Company='" + this.ddlTaget.SelectedItem.Text.Trim() + "'";
SqlDataAdapter da = new SqlDataAdapter(cmdtxt2, Con);
DataSet ds = new DataSet();
da.Fill(ds); //报名人数
Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count1"].ToString());
Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count3"].ToString());
Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count5"].ToString());
Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count7"].ToString()); Count1[] = Convert.ToInt32(ds.Tables[].Rows[]["count9"].ToString()); //全年 Count1[] = Count1[] + Count1[];
Count1[] = Count1[] + Count1[]; Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count2"].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count4"].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count6"].ToString());
Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count8"].ToString()); Count2[] = Convert.ToInt32(ds.Tables[].Rows[]["count10"].ToString()); //全年 Count2[] = Count2[] + Count2[];
Count2[] = Count2[] + Count2[]; //显示折线效果
Font font3 = new System.Drawing.Font("Arial", , FontStyle.Bold);
SolidBrush mybrush = new SolidBrush(Color.Red);
Point[] points1 = new Point[];
points1[].X = ; points1[].Y = - Count1[]; //从106纵坐标开始, 到(0, 0)坐标时
points1[].X = ; points1[].Y = - Count1[];
points1[].X = ; points1[].Y = - Count1[];
points1[].X = ; points1[].Y = - Count1[]; points1[].X = ; points1[].Y = - Count1[];
points1[].X = ; points1[].Y = - Count1[]; points1[].X = ; points1[].Y = - Count1[];
g.DrawLines(mypen2, points1); //绘制折线 //绘制数字
g.DrawString(Count1[].ToString(), font3, Brushes.Red, , points1[].Y - );
g.DrawString(Count1[].ToString(), font3, Brushes.Red, , points1[].Y - );
g.DrawString(Count1[].ToString(), font3, Brushes.Red, , points1[].Y - );
g.DrawString(Count1[].ToString(), font3, Brushes.Red, , points1[].Y - ); g.DrawString(Count1[].ToString(), font3, Brushes.Red, , points1[].Y - );
g.DrawString(Count1[].ToString(), font3, Brushes.Red, , points1[].Y - ); g.DrawString(Count1[].ToString(), font3, Brushes.Red, , points1[].Y - ); Pen mypen3 = new Pen(Color.Green, );
Point[] points2 = new Point[];
points2[].X = ; points2[].Y = - Count2[];
points2[].X = ; points2[].Y = - Count2[];
points2[].X = ; points2[].Y = - Count2[];
points2[].X = ; points2[].Y = - Count2[]; points2[].X = ; points2[].Y = - Count2[];
points2[].X = ; points2[].Y = - Count2[]; points2[].X = ; points2[].Y = - Count2[];
g.DrawLines(mypen3, points2); //绘制折线 //绘制通过人数
g.DrawString(Count2[].ToString(), font3, Brushes.Green, , points2[].Y - );
g.DrawString(Count2[].ToString(), font3, Brushes.Green, , points2[].Y - );
g.DrawString(Count2[].ToString(), font3, Brushes.Green, , points2[].Y - );
g.DrawString(Count2[].ToString(), font3, Brushes.Green, , points2[].Y - ); g.DrawString(Count2[].ToString(), font3, Brushes.Green, , points2[].Y - );
g.DrawString(Count2[].ToString(), font3, Brushes.Green, , points2[].Y - ); g.DrawString(Count2[].ToString(), font3, Brushes.Green, , points2[].Y - ); //绘制标识
g.DrawRectangle(new Pen(Brushes.Red), , , , ); //绘制范围框
g.FillRectangle(Brushes.Red, , , , ); //绘制小矩形
g.DrawString("报名人数", font2, Brushes.Red, , ); g.FillRectangle(Brushes.Green, , , , );
g.DrawString("通过人数", font2, Brushes.Green, , ); System.IO.MemoryStream ms = new System.IO.MemoryStream();
image.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg);
Response.ClearContent();
Response.ContentType = "image/Jpeg";
Response.BinaryWrite(ms.ToArray());
}
finally
{
g.Dispose();
image.Dispose();
}
}
三. 扇形统计图的绘制
效果图:
扇形图完整代码:
private void CreateImage()
{
//把连接字串指定为一个常量
SqlConnection Con = new SqlConnection("Server=(Local);
Database=committeeTraining;Uid=sa;Pwd=**");
Con.Open();
string cmdtxt = selectString; // "select * from ##Count"; //
//SqlCommand Com = new SqlCommand(cmdtxt, Con);
DataSet ds = new DataSet();
SqlDataAdapter Da = new SqlDataAdapter(cmdtxt, Con);
Da.Fill(ds);
Con.Close();
float Total = 0.0f, Tmp; //转换成单精度。也可写成Convert.ToInt32
Total = Convert.ToSingle(ds.Tables[].Rows[][this.count[]]); // Total=Convert.ToSingle(ds.Tables[0].Rows[0][this.count[0]]);
//设置字体,fonttitle为主标题的字体
Font fontlegend = new Font("verdana", );
Font fonttitle = new Font("verdana", , FontStyle.Bold); //背景宽
int width = ;
int bufferspace = ;
int legendheight = fontlegend.Height * + bufferspace; //高度
int titleheight = fonttitle.Height + bufferspace;
int height = width + legendheight + titleheight + bufferspace;//白色背景高
int pieheight = width;
Rectangle pierect = new Rectangle(, titleheight, width, pieheight); //加上各种随机色
ArrayList colors = new ArrayList();
Random rnd = new Random();
for (int i = ; i < ; i++)
colors.Add(new SolidBrush(Color.FromArgb(rnd.Next(), rnd.Next(), rnd.Next()))); //创建一个bitmap实例
Bitmap objbitmap = new Bitmap(width, height);
Graphics objgraphics = Graphics.FromImage(objbitmap); //画一个白色背景
objgraphics.FillRectangle(new SolidBrush(Color.White), , , width, height); //画一个亮黄色背景
objgraphics.FillRectangle(new SolidBrush(Color.Beige), pierect); //以下为画饼图(有几行row画几个)
float currentdegree = 0.0f; //画通过人数
objgraphics.FillPie((SolidBrush)colors[], pierect, currentdegree,
Convert.ToSingle(ds.Tables[].Rows[][this.count[]]) / Total * );
currentdegree += Convert.ToSingle(ds.Tables[].Rows[][this.count[]]) / Total * ; //未通过人数饼状图
objgraphics.FillPie((SolidBrush)colors[], pierect, currentdegree,
((Convert.ToSingle(ds.Tables[].Rows[][this.count[]]))-(Convert.ToSingle(ds.Tables[].Rows[][this.count[]]))) / Total * );
currentdegree += ((Convert.ToSingle(ds.Tables[].Rows[][this.count[]])) -
(Convert.ToSingle(ds.Tables[].Rows[][this.count[]]))) / Total * ; //以下为生成主标题
SolidBrush blackbrush = new SolidBrush(Color.Black);
SolidBrush bluebrush = new SolidBrush(Color.Blue);
string title = " 机关单位成绩统计饼状图: "
+ "\n \n\n";
StringFormat stringFormat = new StringFormat();
stringFormat.Alignment = StringAlignment.Center;
stringFormat.LineAlignment = StringAlignment.Center; objgraphics.DrawString(title, fonttitle, blackbrush,
new Rectangle(, , width, titleheight), stringFormat); //列出各字段与得数目
objgraphics.DrawRectangle(new Pen(Color.Red, ), , height + - legendheight, width, legendheight + ); objgraphics.DrawString("----------------统计信息------------------",
fontlegend, bluebrush, , height - legendheight + fontlegend.Height * + );
objgraphics.DrawString("统计单位: " + this.ddlTaget.SelectedItem.Text,
fontlegend, blackbrush, , height - legendheight + fontlegend.Height * + );
objgraphics.DrawString("统计年份: " + this.ddlYear.SelectedItem.Text,
fontlegend, blackbrush, , height - legendheight + fontlegend.Height * + );
objgraphics.DrawString("统计期数: " + this.ddlSpan.SelectedItem.Text,
fontlegend, blackbrush, , height - legendheight + fontlegend.Height * + ); objgraphics.FillRectangle((SolidBrush)colors[], ,height - legendheight + fontlegend.Height * + , , );
objgraphics.DrawString("报名总人数: " + Convert.ToString(Convert.ToSingle(ds.Tables[].Rows[][this.count[]])),
fontlegend, blackbrush, , height - legendheight + fontlegend.Height * + );
objgraphics.FillRectangle((SolidBrush)colors[], , height - legendheight + fontlegend.Height * + , , );
objgraphics.DrawString("通过总人数: " + Convert.ToString(Convert.ToSingle(ds.Tables[].Rows[][this.count[]])),
fontlegend, blackbrush, , height - legendheight + fontlegend.Height * + );
objgraphics.DrawString("未通过人数: " + ((Convert.ToSingle(ds.Tables[].Rows[][this.count[]])) -
(Convert.ToSingle(ds.Tables[].Rows[][this.count[]]))), fontlegend, blackbrush, , height - legendheight + fontlegend.Height * + ); objgraphics.DrawString("通过率: " + Convert.ToString((Convert.ToSingle(ds.Tables[].Rows[][this.count[]]) /
Convert.ToSingle(ds.Tables[].Rows[][this.count[]])) * )+ " %", fontlegend,
blackbrush, , height - legendheight + fontlegend.Height * + ); Response.ContentType = "image/Jpeg";
objbitmap.Save(Response.OutputStream, System.Drawing.Imaging.ImageFormat.Jpeg);
objgraphics.Dispose();
objbitmap.Dispose(); }
这里的统计图直接输出到网页, 如果大家需要制作 winForm 上需要显示的统计图。可参考我之前写的:
C#制作艺术字
C#完成超酷的图像效果 (附demo)
出处:http://www.cnblogs.com/ziyiFly/archive/2008/09/24/1297841.html