苏飞论坛

 找回密码
 马上注册

QQ登录

只需一步,快速开始

分布式系统框架(V2.0) 轻松承载百亿数据,千万流量!讨论专区 - 源码下载 - 官方教程

HttpHelper爬虫框架(V2.7-含.netcore) HttpHelper官方出品,爬虫框架讨论区 - 源码下载 - 在线测试和代码生成

HttpHelper爬虫类(V2.0) 开源的爬虫类,支持多种模式和属性 源码 - 代码生成器 - 讨论区 - 教程- 例子

查看: 51169|回复: 26
打印 上一主题 下一主题

[小工具] C#多线程|匿名委托传参数|测试网站压力|附源代码

[复制链接]
跳转到指定楼层
楼主
发表于 2012-7-1 12:02:53 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式
源代码下载: AutoFor1.0.zip (62.99 KB, 下载次数: 1269)

引言

我们一直在做网站,但在我河南这块,对测试工作,特别是压力测试一般都不怎么在意,都是自己访问一下速度不错就行了,再就是数据库访问速度测试也是同样情况
程序员在写Sql代码时,一般是一个人写完之后,一运行可快完事
其实这些是不够的,我们根本没有进行过多用户多线程的测试,如果是100个,一千个要同时访问,还会有这样的速度吗?
我们自己反思一下是不是有这样的经历呢,我做的网站刚上传服务器,打开很快,调数据库1000条以内一秒用不了,感觉非常好,但过了不几天,就会感觉到网站很慢很慢,于是去检查测试
其实这些可以提前做的,我下面来实现一个多线程测试网站访问速度的功能。
效果


说明:
1.一次可以开N多个线程;
2.可以设置要访问的地址;
3.可以设置要循环访问的次数;
相关技术点:
1.C# Winform;
2.[url=http://www.sufeinet.com/thread-3-1-1.html]httpHelper类[/url];这是我之前自己写的一个类,大家可以参考一下(带证书,无视编码,设置代理等)
3.多线程;
4.线程之间的传参;
5.委托与匿名委托的使用方法;
实现步骤:
1.新建一个CS项目,AutoFor,新建一个窗体为TextFor
2.自己拉几个控件实现如下界面

3.定义一个委托用来修改DataGridview的值,代码如下
[C#] 纯文本查看 复制代码
//修改表格的委托
        private delegate void UpDateDgvDelegate(string msg, int rowId, string columnName);

        private UpDateDgvDelegate _upDateStateDelegate;
        //构造器
        public TextFor()
        {
            InitializeComponent();
            _upDateStateDelegate = new UpDateDgvDelegate(UpDateDgv);
        }

        /// <summary>
        /// 修改表格的行数据
        /// </summary>
        /// <param name="msg">要修改为的数据</param>
        /// <param name="rowId">行号</param>
        /// <param name="columnName">列名</param>
        private void UpDateDgv(string msg, int rowId, string columnName)
        {
            try
            {
                dgvTextFor.Rows[rowId].Cells[columnName].Value = msg.ToString();
            }
            catch { }
        }



4.在单击开始时先生成对应的线程表格式,就是界面上的DataGridview,大家看下代码
[C#] 纯文本查看 复制代码
/// <summary>
        /// 创建表格
        /// </summary>
        /// <param name="rows">生成多少行数</param>
        private void CreateTable(int rows)
        {
            DataTable dt_Sale = new DataTable();
            DataColumn dc = null;
            //线程ID
            dc = new DataColumn();
            dc.ColumnName = "线程ID";
            dc.DefaultValue = "1";
            dc.DataType = Type.GetType("System.String");
            dt_Sale.Columns.Add(dc);

            //循环类型
            dc = new DataColumn();
            dc.ColumnName = "循环类型";
            dc.DefaultValue = " ";
            dc.DataType = Type.GetType("System.String");
            dt_Sale.Columns.Add(dc);

            //当前循环次数
            dc = new DataColumn();
            dc.ColumnName = "当前循环次数";
            dc.DefaultValue = " ";
            dc.DataType = Type.GetType(" System.String");
            dt_Sale.Columns.Add(dc);

            //开始时间
            dc = new DataColumn();
            dc.ColumnName = "开始时间";
            dc.DefaultValue = " ";
            dc.DataType = Type.GetType("System.String");
            dt_Sale.Columns.Add(dc);

            //结束时间
            dc = new DataColumn();
            dc.ColumnName = "结束时间";
            dc.DefaultValue = " ";
            dc.DataType = Type.GetType("System.String");
            dt_Sale.Columns.Add(dc);

            //总用时(毫秒)
            dc = new DataColumn();
            dc.ColumnName = "总用时(毫秒)";
            dc.DefaultValue = " ";
            dc.DataType = Type.GetType("System.String");
            dt_Sale.Columns.Add(dc);


            DataRow dr = dt_Sale.NewRow();
            for (int i = 1; i < rows; i++)
            {
                dr["线程ID"] = i.ToString();
                dr["循环类型"] = "For循环";
                dr["当前循环次数"] = "0";
                dr["开始时间"] = "00:00:00";
                dr["结束时间"] = "00:00:00";
                dr["总用时(毫秒)"] = "0";
                dt_Sale.Rows.Add(dr);
                dr = dt_Sale.NewRow();
            }
            dgvTextFor.DataSource = dt_Sale;
        }

5.定义一个方法用来访问指定的网站就是我们的实际测试这块,
一起来看下代码
[C#] 纯文本查看 复制代码
/// <summary>
        /// 执行数据
        /// </summary>
        /// <param name="dgvrowid"> 线程号行号</param>
        /// <param name="number">循环总次数</param>
        private void PingTask(int dgvrowid, int number)
        {
            //获取开始时间
            DateTime st = DateTime.Now;

            //开始时间
            this.BeginInvoke(_upDateStateDelegate, st.ToString("hh-mm-ss"), dgvrowid, "开始时间");

            for (int i = 0; i < number; i++)
            {
                try
                {
                    HttpHelps hh = new HttpHelps();

                    //自动访问百度,主要是延长时间
                    hh.GetHttpRequestStringByNUll_Get("www.baidu.com", null);

                    //当前循环次数
                    this.BeginInvoke(_upDateStateDelegate, i.ToString(), dgvrowid, "当前循环次数");

                    //获取结束时间
                    DateTime et = DateTime.Now;

                    //结束时间
                    this.BeginInvoke(_upDateStateDelegate, et.ToString("hh-mm-ss"), dgvrowid, "结束时间");

                    //总用时(毫秒)
                    this.BeginInvoke(_upDateStateDelegate, ExecDateDiff(st, et), dgvrowid, "总用时(毫秒)");
                }
                catch { }
            }

        }


我来解释下这句 //总用时(毫秒) this.BeginInvoke(_upDateStateDelegate, ExecDateDiff(st, et), dgvrowid, "总用时(毫秒)");
第一个BeginInvoke方法,是用来异步执行委托的,系统自带方法。
upDateStateDelegate是要执行的委托我们前面有定义
ExecDateDiff计算时间差的方法自己写的如下代码
[C#] 纯文本查看 复制代码
/// <summary>
        /// 程序执行时间测试
        /// </summary>
        /// <param name="dateBegin">开始时间</param>
        /// <param name="dateEnd">结束时间</param>
        /// <returns>返回(秒)单位,比如: 0.00239秒</returns>
        public static string ExecDateDiff(DateTime dateBegin, DateTime dateEnd)
        {
            TimeSpan ts1 = new TimeSpan(dateBegin.Ticks);
            TimeSpan ts2 = new TimeSpan(dateEnd.Ticks);
            TimeSpan ts3 = ts1.Subtract(ts2).Duration();
            return ts3.TotalMilliseconds.ToString();
        }

注意,大家一定要记着这里的类型要和委托的类型是一样的,否则为出错,而且它不会自动转化,如果你定义的是String,传的是int是不可以的, 要手动的去转,这点请大家注意一下。

dgvrowid启动线程所在DataGridview行,一行是一个线程的变化情况
"总用时(毫秒)" 列名,这里是为了方便 大家看,使用的汉语大家见谅。
6.启动线程,我们只要执行一个For就可以循环启动了,大家一起来看看方法吧,
[C#] 纯文本查看 复制代码
private void button3_Click(object sender, EventArgs e)
        {
            int count = Convert.ToInt32(txtCount.Text.Trim());
            int number = Convert.ToInt32(txtNumber.Text.Trim());
            CreateTable(count + 1);
            //开启number个线程
            for (int i = 0; i < count; i++)
            {
                Thread pingTask = new Thread(new ThreadStart(delegate
               {
                   PingTask(i, number);
               }));
                pingTask.Start();
                Thread.Sleep(100);
            }
        }



我们都知道线程是不能直接传参数的,只能传Object,但使用匿名委托就可以解决 这个问题,方法如上面,大家看不明白的可以留言给我。
Thread.Sleep(100); 是为了让线程正常启动做了一个时间间隔。
大家可以根据自己的情况调整

其实这个例子不但可以实现这样测试,大家还可以用来访问数据库,开上几千个线程,看看你的Sql代码访问速度有多快。
个人感觉很不错的一种压力测试方法
希望大家多提提建议哦

本帖被以下淘专辑推荐:



1. 开通SVIP会员,免费下载本站所有源码,不限次数据,不限时间
2. 加官方QQ群,加官方微信群获取更多资源和帮助
3. 找站长苏飞做网站、商城、CRM、小程序、App、爬虫相关、项目外包等点这里
沙发
发表于 2012-12-21 16:16:53 | 只看该作者
很不错,想做测试,但以前在网站开发时就有这个意识,但后来因种种原因就没做下来,以后试试。。。
板凳
 楼主| 发表于 2012-12-21 16:17:57 | 只看该作者
weiyz2011 发表于 2012-12-21 16:16
很不错,想做测试,但以前在网站开发时就有这个意识,但后来因种种原因就没做下来,以后试试。。。

这个只是练手行,实际使用,还需要你自己根据情况进行修改
地板
发表于 2013-1-5 17:49:44 | 只看该作者
你好,耽误你的时间,我想询问一个问题。
在这个程序开启的那个位置就是button3的click事件中
那个for循环里面每次都声明一个pingTask的线程,这个循环中和上次生成的那个有影响吗?
比如我们开启了5个线程,这5个都是交pingTask吗?不知道我说的对不对,本人新手能力有限,请见谅、
Thread pingTask = new Thread(new ThreadStart(delegate
               {
                   PingTask(i, number);
               }));
5
 楼主| 发表于 2013-1-5 18:26:34 | 只看该作者
xoyo 发表于 2013-1-5 17:49
你好,耽误你的时间,我想询问一个问题。
在这个程序开启的那个位置就是button3的click事件中
那个for循环 ...

各是各的没有任何影响,相当于你在一同的电脑上开的不同的几个程序
6
发表于 2013-1-6 17:11:02 | 只看该作者
admin 发表于 2013-1-5 18:26
各是各的没有任何影响,相当于你在一同的电脑上开的不同的几个程序

万分感谢。
7
 楼主| 发表于 2013-1-6 17:12:04 | 只看该作者
xoyo 发表于 2013-1-6 17:11
万分感谢。

不客气
8
发表于 2013-1-7 10:45:53 | 只看该作者
这个太邪恶了
9
 楼主| 发表于 2013-1-7 11:18:04 | 只看该作者
Eagle 发表于 2013-1-7 10:45
这个太邪恶了

http://www.sufeinet.com/thread-13-1-1.html那这个呢
10
发表于 2013-4-21 11:18:05 | 只看该作者
正在找这方面得内容,谢谢飞哥
您需要登录后才可以回帖 登录 | 马上注册

本版积分规则

QQ|手机版|小黑屋|手机版|联系我们|关于我们|广告合作|苏飞论坛 ( 豫ICP备18043678号-2)

GMT+8, 2025-1-23 01:01

© 2014-2021

快速回复 返回顶部 返回列表