画函数图形的C#程序(改进版)

我在10月份发表过一篇随笔“ 画函数图形的C#程序,兼论一个病态函数”,在那篇随笔中写道:
这个画函数图形的C#程序有一个严重的缺点,就是函数表达式是直接写的源程序中的,不能象SciLab和Matlab那样交互式地输入。
后来,根据“ 空间/IV”的评论,我写了个动态生成用户输入的函数表达式的类,用以改进这个画函数图形的C#程序。下面是该程序的运行效果:
画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)
可以看到,不但要画的函数的表达式可以由用户动态地输入,而且函数自变量的范围也可以是常量表达式。 下面就是源程序:
画函数图形的C#程序(改进版) //  plot.cs: 画函数图形, 编译方法: csc /t:winexe plot.cs Expression.cs
画函数图形的C#程序(改进版)
using  System;
画函数图形的C#程序(改进版)
using  System.Drawing;
画函数图形的C#程序(改进版)
using  System.Windows.Forms;
画函数图形的C#程序(改进版)
using  Skyiv.Util;
画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)
namespace  Skyiv.Ben.Plot
画函数图形的C#程序(改进版)
{
画函数图形的C#程序(改进版)  
sealed class PlotForm : Form
画函数图形的C#程序(改进版)  
{
画函数图形的C#程序(改进版)    
const int yBase = 24// 屏幕保留区域的高度
画函数图形的C#程序(改进版)

画函数图形的C#程序(改进版)    TextBox tbxX0, tbxX1;  
// 函数自变量的取值范围
画函数图形的C#程序(改进版)
    TextBox tbxExpression; // 函数的表达式
画函数图形的C#程序(改进版)
    
画函数图形的C#程序(改进版)    PlotForm()
画函数图形的C#程序(改进版)    
{
画函数图形的C#程序(改进版)      SuspendLayout();
画函数图形的C#程序(改进版)      
画函数图形的C#程序(改进版)      Button btnSubmit 
= new Button();
画函数图形的C#程序(改进版)      btnSubmit.Text 
= "刷新";
画函数图形的C#程序(改进版)      btnSubmit.Location 
= new Point(00);
画函数图形的C#程序(改进版)      btnSubmit.Size 
= new Size(4824);
画函数图形的C#程序(改进版)      btnSubmit.Click 
+= new EventHandler(BtnSubmit_Click);
画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)      tbxX0 
= new TextBox();
画函数图形的C#程序(改进版)      tbxX0.Text 
= "-Math.PI";
画函数图形的C#程序(改进版)      tbxX0.Location 
= new Point(553);
画函数图形的C#程序(改进版)      tbxX0.Size 
= new Size(10020);
画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)      tbxX1 
= new TextBox();
画函数图形的C#程序(改进版)      tbxX1.Text 
= "Math.PI";
画函数图形的C#程序(改进版)      tbxX1.Location 
= new Point(1603);
画函数图形的C#程序(改进版)      tbxX1.Size 
= new Size(10020);
画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)      tbxExpression 
= new TextBox();
画函数图形的C#程序(改进版)      tbxExpression.Text 
= "Math.Sin(x)";
画函数图形的C#程序(改进版)      tbxExpression.Location 
= new Point(2653);
画函数图形的C#程序(改进版)      tbxExpression.Size 
= new Size(33520);
画函数图形的C#程序(改进版)      tbxExpression.Anchor 
= (AnchorStyles.Top | AnchorStyles.Left | AnchorStyles.Right);
画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)      Controls.AddRange(
new Control[]{btnSubmit, tbxX0, tbxX1, tbxExpression});
画函数图形的C#程序(改进版)      Text 
= "Plot";
画函数图形的C#程序(改进版)      BackColor 
= Color.White;
画函数图形的C#程序(改进版)      ClientSize 
= new Size(600600 + yBase);
画函数图形的C#程序(改进版)      
// WindowState = FormWindowState.Maximized;
画函数图形的C#程序(改进版)

画函数图形的C#程序(改进版)      ResumeLayout(
false);
画函数图形的C#程序(改进版)    }

画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)    
// 点击“刷新”按钮时重绘程序主窗口
画函数图形的C#程序(改进版)
    void BtnSubmit_Click(object sender, EventArgs e)
画函数图形的C#程序(改进版)    
{
画函数图形的C#程序(改进版)      Invalidate();
画函数图形的C#程序(改进版)    }

画函数图形的C#程序(改进版)    
画函数图形的C#程序(改进版)    
/*
画函数图形的C#程序(改进版)    // 因为本程序使用 C# 的反射功能动态生成数学表达式并计算其值
画函数图形的C#程序(改进版)    // 所以重画时有点慢,如果你的计算机的速度不是非常快的,
画函数图形的C#程序(改进版)    // 就不要在窗口改变大小时强制重绘,而是通过点击发“刷新”按钮重绘。
画函数图形的C#程序(改进版)    protected override void OnSizeChanged(EventArgs e)
画函数图形的C#程序(改进版)    {
画函数图形的C#程序(改进版)      Invalidate();
画函数图形的C#程序(改进版)      base.OnSizeChanged(e);
画函数图形的C#程序(改进版)    }
画函数图形的C#程序(改进版)    
*/

画函数图形的C#程序(改进版)    
画函数图形的C#程序(改进版)    
protected override void OnPaint(PaintEventArgs e)
画函数图形的C#程序(改进版)    
{
画函数图形的C#程序(改进版)      Graphics gc 
= e.Graphics;
画函数图形的C#程序(改进版)      
try
画函数图形的C#程序(改进版)      
{
画函数图形的C#程序(改进版)        
double x0 = new Expression(tbxX0.Text).Compute(0);
画函数图形的C#程序(改进版)        
double x1 = new Expression(tbxX1.Text).Compute(0);
画函数图形的C#程序(改进版)        Size size 
= ClientSize;
画函数图形的C#程序(改进版)        
int i0 = 0;
画函数图形的C#程序(改进版)        
int i1 = size.Width - 1;
画函数图形的C#程序(改进版)        
int j0 = yBase;
画函数图形的C#程序(改进版)        
int j1 = size.Height - 1;
画函数图形的C#程序(改进版)        Pen pen 
= new Pen(Color.Black, 1);
画函数图形的C#程序(改进版)        gc.DrawLine(pen, i0, j0, i1, j0); 
// 画图区和保留区的分界线
画函数图形的C#程序(改进版)
        double rx = (x1 - x0) / (i1 - i0);
画函数图形的C#程序(改进版)        
double y0, y1;
画函数图形的C#程序(改进版)        Expression fx 
= new Expression(tbxExpression.Text);
画函数图形的C#程序(改进版)        GetFunctionValueRange(fx, x0, rx, i0, i1, 
out y0, out y1);
画函数图形的C#程序(改进版)        
double ry = (y1 - y0) / (j1 - j0);
画函数图形的C#程序(改进版)        Out(gc, 
0"ClientSize: {0}x{1}", i1 - i0 + 1, j1 - j0 + 1);
画函数图形的C#程序(改进版)        Out(gc, 
1"f(x): " + tbxExpression.Text);
画函数图形的C#程序(改进版)        Out(gc, 
2"x:[{0}, {1}] range:{2}", x0, x1, x1 - x0);
画函数图形的C#程序(改进版)        Out(gc, 
3"y:[{0}, {1}] range:{2}", y0, y1, y1 - y0);
画函数图形的C#程序(改进版)        Out(gc, 
4"rx:{0}"1 / rx);  // 函数自变量每单位值用多少个象素表示
画函数图形的C#程序(改进版)
        Out(gc, 5"ry:{0}"1 / ry);  // 函数的值每单位值用多少个象素表示
画函数图形的C#程序(改进版)
        Out(gc, 6"r :{0}", rx / ry); // 该值如果小于1表示图形纵向被压扁,反之则被拉伸
画函数图形的C#程序(改进版)
        pen.Color = Color.Green;
画函数图形的C#程序(改进版)        
int j = j1 + (int)(y0 / ry);
画函数图形的C#程序(改进版)        
if (j >= j0 && j <= j1) gc.DrawLine(pen, i0, j, i1, j); // x坐标轴
画函数图形的C#程序(改进版)
        int i = i0 - (int)(x0 / rx);
画函数图形的C#程序(改进版)        
if (i >= i0 && i <= i1) gc.DrawLine(pen, i, j0, i, j1); // y坐标轴
画函数图形的C#程序(改进版)
        pen.Color = Color.Red;
画函数图形的C#程序(改进版)        
for (i = i0; i <= i1; i++)
画函数图形的C#程序(改进版)        
{
画函数图形的C#程序(改进版)          
double x = x0 + (i - i0) * rx;
画函数图形的C#程序(改进版)          
double y = fx.Compute(x);
画函数图形的C#程序(改进版)          
if (double.IsInfinity(y) || double.IsNaN(y)) continue;
画函数图形的C#程序(改进版)          j 
= j1 - (int)((y - y0) / ry);
画函数图形的C#程序(改进版)          
if (j > j1 || j < j0) continue;
画函数图形的C#程序(改进版)          gc.DrawLine(pen, i, j, i 
+ 1, j); // 画函数的图形
画函数图形的C#程序(改进版)
        }

画函数图形的C#程序(改进版)      }

画函数图形的C#程序(改进版)      
catch (Exception ex)
画函数图形的C#程序(改进版)      
{
画函数图形的C#程序(改进版)        Out(gc, 
0, ex.Message);
画函数图形的C#程序(改进版)      }

画函数图形的C#程序(改进版)      
base.OnPaint(e);
画函数图形的C#程序(改进版)    }

画函数图形的C#程序(改进版)    
画函数图形的C#程序(改进版)    
// 函数值的取值范围
画函数图形的C#程序(改进版)
    void GetFunctionValueRange(Expression fx, double x0, double rx, int i0, int i1, out double y0, out double y1)
画函数图形的C#程序(改进版)    
{
画函数图形的C#程序(改进版)      y0 
= double.MaxValue;
画函数图形的C#程序(改进版)      y1 
= double.MinValue;
画函数图形的C#程序(改进版)      
for (int i = i0; i <= i1; i++)
画函数图形的C#程序(改进版)      
{
画函数图形的C#程序(改进版)        
double x = x0 + (i - i0) * rx;
画函数图形的C#程序(改进版)        
double y = fx.Compute(x);
画函数图形的C#程序(改进版)        
if (double.IsInfinity(y) || double.IsNaN(y)) continue;
画函数图形的C#程序(改进版)        
if (y0 > y) y0 = y;
画函数图形的C#程序(改进版)        
if (y1 < y) y1 = y;
画函数图形的C#程序(改进版)      }

画函数图形的C#程序(改进版)    }

画函数图形的C#程序(改进版)    
画函数图形的C#程序(改进版)    
// 在指定的位置写字符串
画函数图形的C#程序(改进版)
    void Out(Graphics gc, int line, string fmt, params object [] args)
画函数图形的C#程序(改进版)    
{
画函数图形的C#程序(改进版)      gc.DrawString(
string.Format(fmt, args), new Font("Courier New"10), Brushes.Blue, new PointF(5, yBase + 15 * line));
画函数图形的C#程序(改进版)    }

画函数图形的C#程序(改进版)
画函数图形的C#程序(改进版)    
static void Main()
画函数图形的C#程序(改进版)    
{
画函数图形的C#程序(改进版)      Application.Run(
new PlotForm());
画函数图形的C#程序(改进版)    }

画函数图形的C#程序(改进版)  }

画函数图形的C#程序(改进版)}

画函数图形的C#程序(改进版)
其中的“Expression.cs”程序请参看我的另一篇随笔:“ 动态地生成用户输入的函数表达式(C#)”。
这里的表达式是使用C#语法。如需要使用 VisualBasic 语法,请参阅该随笔的评论。

你可能感兴趣的:(C#)