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

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

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

InBlock.gif      ResumeLayout(
false);
ExpandedSubBlockEnd.gif    }

InBlock.gif
InBlock.gif    
// 点击“刷新”按钮时重绘程序主窗口
InBlock.gif
    void BtnSubmit_Click(object sender, EventArgs e)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif      Invalidate();
ExpandedSubBlockEnd.gif    }

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

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

ExpandedSubBlockEnd.gif      }

InBlock.gif      
catch (Exception ex)
ExpandedSubBlockStart.gifContractedSubBlock.gif      
dot.gif{
InBlock.gif        Out(gc, 
0, ex.Message);
ExpandedSubBlockEnd.gif      }

InBlock.gif      
base.OnPaint(e);
ExpandedSubBlockEnd.gif    }

InBlock.gif    
InBlock.gif    
// 函数值的取值范围
InBlock.gif
    void GetFunctionValueRange(Expression fx, double x0, double rx, int i0, int i1, out double y0, out double y1)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif      y0 
= double.MaxValue;
InBlock.gif      y1 
= double.MinValue;
InBlock.gif      
for (int i = i0; i <= i1; i++)
ExpandedSubBlockStart.gifContractedSubBlock.gif      
dot.gif{
InBlock.gif        
double x = x0 + (i - i0) * rx;
InBlock.gif        
double y = fx.Compute(x);
InBlock.gif        
if (double.IsInfinity(y) || double.IsNaN(y)) continue;
InBlock.gif        
if (y0 > y) y0 = y;
InBlock.gif        
if (y1 < y) y1 = y;
ExpandedSubBlockEnd.gif      }

ExpandedSubBlockEnd.gif    }

InBlock.gif    
InBlock.gif    
// 在指定的位置写字符串
InBlock.gif
    void Out(Graphics gc, int line, string fmt, params object [] args)
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif      gc.DrawString(
string.Format(fmt, args), new Font("Courier New"10), Brushes.Blue, new PointF(5, yBase + 15 * line));
ExpandedSubBlockEnd.gif    }

InBlock.gif
InBlock.gif    
static void Main()
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif      Application.Run(
new PlotForm());
ExpandedSubBlockEnd.gif    }

ExpandedSubBlockEnd.gif  }

ExpandedBlockEnd.gif}

None.gif

转载于:https://www.cnblogs.com/gjahead/archive/2007/08/18/861164.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/363939.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Swift-UITextView占位文字及占位文字颜色扩展(可在xib中直接使用)

使用案例 extension UITextView {private struct RuntimeKey {static let hw_placeholderLabelKey UnsafeRawPointer.init(bitPattern: "hw_placeholderLabelKey".hashValue)/// ...其他Key声明}/// 占位文字IBInspectable public var placeholder: String {get {re…

Java 8中Lambda表达式的阴暗面

这篇文章可能不会使我成为任何新朋友。 哦&#xff0c;好吧&#xff0c;无论如何我从来没有真正在学校受到欢迎。 但是&#xff0c;让我们说清楚。 就语言而言&#xff0c;Java 8的最大特色无疑是Lambda表达式。 几年来&#xff0c;它一直是功能语言&#xff08;例如Scala和Clo…

JQ实现弹幕效果

JQ实现弹幕效果&#xff0c;快来吐糟你的想法吧 效果图&#xff1a; 代码如下&#xff0c;复制即可使用&#xff1a; <!DOCTYPE html> <html><head><meta charset"utf-8" /><title>JQ实现弹幕效果</title><style type"t…

获取表的所有列名

selectname fromsyscolumns whereidobject_id(表名)下面这样也可以:selectcolumn_name frominformation_schema.columns wheretable_name news转载于:https://www.cnblogs.com/zhuboxingzbx/archive/2007/09/03/880049.html

天下武功唯快不破------实验吧

题目地址&#xff1a;http://www.shiyanbar.com/ctf/1854 打开链接 全是英文&#xff0c;能力有限&#xff0c;翻译一下&#xff0c;好像没其他东西了&#xff0c;查看一下源码 让用post请求&#xff0c;丢到burp改一下 看到response有一个FLAG: UDBTVF9USElTX1QwX0NINE5HRV9GT…

JS实现文本中查找并替换字符

JS实现文本中查找并替换字符 效果图&#xff1a; 代码如下&#xff0c;复制即可使用&#xff1a; <!DOCTYPE html><html> <head><style type"text/css">*{font-family:"微软雅黑";font-size:16px;margin:0;padding:0;letter-spaci…

在没有IDE的情况下编译和运行Java

最近一个名为“ 不使用IDE编译Java软件包 ”的Java subreddit线程提出了一个问题&#xff1a;“是否有一个命令将软件包内的一组Java文件编译到一个单独的文件夹中&#xff08;以下简称为bin&#xff09;&#xff0c;以及如何我会去运行新的类文件吗&#xff1f;” 该帖子的作者…

[转]MySQL 表锁和行锁机制

本文转自&#xff1a;http://www.cnblogs.com/itdragon/p/8194622.html MySQL 表锁和行锁机制 行锁变表锁&#xff0c;是福还是坑&#xff1f;如果你不清楚MySQL加锁的原理&#xff0c;你会被它整的很惨&#xff01;不知坑在何方&#xff1f;没事&#xff0c;我来给你们标记几个…

使用JS实现文字搬运工

使用JS实现文字搬运工 效果图&#xff1a; 代码如下&#xff0c;复制即可使用&#xff1a; <!DOCTYPE html> <html><head><meta http-equiv"Content-Type" content"text/html; charsetUTF-8"> <title>使用JS实现文字搬运工&…

ActiveMQ –经纪人网络解释–第3部分

现在&#xff0c;我们已经在本博客系列的第1部分和第2 部分中了解了ActiveMQ网络连接器的基础&#xff0c;在第3部分中&#xff0c;我们将研究ActiveMQ如何平衡连接到代理网络的使用者。 介绍 当可以无序处理队列中的消息时通常使用并发使用者&#xff0c;通常可以提高消息吞吐…

Linux下用户组、文件权限详解=------转载文

转载自-----原文地址&#xff1a; https://www.cnblogs.com/123-/p/4189072.html Linux下用户组、文件权限详解 用户组 在linux中的每个用户必须属于一个组&#xff0c;不能独立于组外。在linux中每个文件有所有者、所在组、其它组的概念 - 所有者 - 所在组 - 其它组 - 改变用…

JS实现逼真的雪花飘落特效

逼真的雪花飘落特效 效果图&#xff1a; 图片素材 &#xff1a; --> ParticleSmoke.png 代码如下&#xff0c;复制即可使用&#xff1a; <!doctype html> <html> <head> <meta charset"UTF-8"> <meta name"renderer" conte…

阴影及定位

阴影及定位 隐藏及阴影 标签隐藏 1、显示方式 display display: none; /*表示在页面中隐藏&#xff0c;并且不占位&#xff0c;但是重新显示出来又会占位*/ 2、透明度 opacity opacity: 0; /* 0 代表完全透明 1 代表完全显示 但是即使是透明了也会在页面中占位*/ /* 显示方式透…

ActiveMQ –经纪人网络解释–第2部分

在此博客中&#xff0c;我们将看到双工网络连接器如何工作。 在上一部分中&#xff0c;我们从broker-1和broker-2创建了一个网络连接器。 我们能够看到当代理2上有一个使用者使用队列“ foo.bar”时&#xff0c;代理1上的队列“ foo.bar”的消息如何转发到代理2上的队列“ foo…

JQ实现情人节表白程序

JQ实现情人节表白页面 效果图&#xff1a; 表白利页&#xff0c;你值得拥有哦&#xff01; 代码如下&#xff0c;复制即可使用&#xff1a; <!doctype html> <html> <head> <meta charset"utf-8"> <title>JQ实现情人节表白程序<…

消息队列使用的四种场景介绍(转)

消息队列中间件是分布式系统中重要的组件&#xff0c;主要解决应用耦合&#xff0c;异步消息&#xff0c;流量削锋等问题 实现高性能&#xff0c;高可用&#xff0c;可伸缩和最终一致性架构 使用较多的消息队列有ActiveMQ&#xff0c;RabbitMQ&#xff0c;ZeroMQ&#xff0c;Ka…

一步一步SharePoint 2007之三十一:实现文档Event Handler(3)——附加Handler程序

摘要  本篇文章将介绍实现文档Event Handler的第三部分——附加Handler程序。正文  下面将记录每一步的操作过程。  1、首先打开我的网站&#xff0c;依次点击Document Center、Documents&#xff0c;进入Documents列表页面。  2、在Documents列表界面中点击Settings&a…

Day2 HTML基本标签元素

Day2 HTML基本标签元素 HTML: 超文本标记语言(HyperText Mark-up Language ) 1.作用&#xff1a;写网页结构    2.HTML不区分大小写&#xff0c;建议小写   3.文件后缀 .html 或者 .htm   4.html由浏览器解析执行. 由上往下&#xff0c;由左往右 1) HTML标…

作用域、执行环境、闭包(四)

本文也同步发表在我的公众号“我的天空” 上一期我们已经介绍了闭包&#xff0c;由于闭包可以延长函数内部的变量的生存周期&#xff0c;因此我们可以将不需要暴露在全局的变量封装成函数的内部变量&#xff0c;从而避免代码污染。 譬如要实现一个简单的累加器&#xff0c;为了…

php static_castunsigned int,C++ static_cast、dynamic_cast、const_cast和reinterpret_cast(四种类型转换运算符)...

上节讲到&#xff0c;隐式类型转换是安全的&#xff0c;显式类型转换是有风险的&#xff0c;C语言之所以增加强制类型转换的语法&#xff0c;就是为了强调风险&#xff0c;让程序员意识到自己在做什么。但是&#xff0c;这种强调风险的方式还是比较粗放&#xff0c;粒度比较大&…