C#委託+回調詳解

今天寫不完,明天會接着寫的,,,,編程

學習C#有一段時間了,不過C#的委託+回調才這兩天才會用,之前只是知道怎麼用.前面的一篇文章,函數指針,實際上是爲這個作鋪墊的,說白了委託就至關於C語言中的函數指針,具體說用委託有什麼好處,我也不知道,可是由於你只要學C#那麼回調就必定要會,回調是委託的一種.回調多用於線程間....仍是用代碼一點點的說明吧!如今認爲本身之前不會用委託是由於手太懶,再者沒有太多的必須用C#作的Demo,本身學東西都是用到什麼學什麼,想作什麼東西須要什麼知識就學什麼,前幾天有了必需要作的C#的Demo,關於檢測TCP通訊發過來的數據的.扯了這麼多...回調主要的應用是---若是你在一個線程裏操做像文本框,按鈕,Label等組件時,會報錯...緣由--C#它不讓這樣操做,,,,函數

看例子學習

窗體裏面就放一個textboxspa

先這樣寫線程

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            textBox1.Text = "123456";
        }
    }
}

結果3d

 

就這一句   textBox1.Text = "123456";往文本框中寫入123456     程序運行沒問題指針

 

如今加入線程調試

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        Thread writeThread;
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            writeThread = new Thread(write_Metod);
            writeThread.Start();
        }
        private void write_Metod()
        {
            textBox1.Text = "123456";
        }
    }
}

而後啓動code

蹦了orm

一塊兒動就退出

 說早了.......先說委託

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            write_Metod();
        }
        private void write_Metod()
        {
            textBox1.Text = "123456";                      
        }
    }
}

這個是沒有用委託的,簡單的例子,程序一運行就執行write_Metod();函數而後文本就打印123456了

如今就用委託來寫一下

 函數名字太長會讓人感受特複雜有木有....因此下面的名字.....

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        delegate void a();//定義一個委託111111111111111111
        
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            a b= new a(c);//實例化2222222222222222222
            b();//3333333333333333333333
        }
        void c()
        {
            textBox1.Text = "123456";                      
        }
    }
}

 

 

 delegate void a();//定義一個委託111111111111111111
這是定義一個委託 要用關鍵字 delegate
後面的

void a();
函數名字本身取,,不過要有本身的規範,,有本身的一套編程風格就好..
void a();
這是一個沒有返回值的,無參數的函數
由於學過函數指針都知道,只有類型同樣的函數才能把一個函數的指針(地址)傳給另外一個函數..
由於咱們但願把
void c()這個函數用另外一個函數代替
這個函數的類型是void 的 無參數的函數

因此就這樣定義了
delegate void a();//定義一個委託----再次說明delegate是關鍵字,意味着定義了一個委託------你說函數指針也行,,C#啊;;;淡化了指針的概念

而後
a b= new a(c);//實例化2222222222222222222

不知道有沒有不知道爲什麼實例化的
若是不知道實例化那麼知道對象嗎?是C#整的名詞對象
若是不知道對象那麼知道類嗎?,,,,,
上面所說的沒有什麼用的,只是用來嚇唬人的,記得第一次想學高級語言,,,一打開書,,,,,崩潰了,完全崩潰了,什麼對象,,,,一開頭介紹就是,,,什麼面向對象,,嚇的我趕忙把書方回去,,心有不甘再來一本,,沒想到一打開書...又來了,,,XXXXXX是面向對象的語言,,,,,那時候在想,我去過高深了,,面向對象,,面對誰呢!! 畢向東的JAVA講的不錯....學會一門高級語言,語言有不少共通的地方
又耽誤你們時間了....對了馬士兵的JAVA也挺好,,,都看看 都看看
關於實例化
定義一個A a;假設定義了一個a
若是你不實例化也就是 a = new A();
那麼系統不會爲a開闢任何空間,只有 a = new A();了 系統纔會在內存中爲 a 開闢一段空間,才真正有了a的存在


a b= new a(c);//實例化2222222222222222222
這一句的意思就是
b= c;
哦,原來就是函數指針賦值啊
那麼
b();   就至關於  c();
好再過一遍
先定義一個委託
delegate void a();//定義一個委託

而後實例化---a b = new a(c);
void c()
{

}
這樣 b 就==c了
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        delegate void a();//定義一個委託
        
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            a b = new a(c);//實例化
            b();
        }
        void c()
        {
            textBox1.Text = "123456";                      
        }
    }
}

 



 

咱接着

假如說

 void c(string str)
        {
            textBox1.Text = str;                      
        }

那麼我定義的委託(函數指針)也應該是

delegate void a(string str);//定義一個委託

怎樣把c傳過去呢

a b = new a(c);//實例化

而後

b("123456");

由於c是

void c(string str)
因此
必須寫個string進去嘛
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        delegate void a(string str);//定義一個委託
        
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            a b = new a(c);//實例化
            b("123456");
        }
        void c(string str)
        {
            textBox1.Text = str;                      
        }
    }
}


委託也就這樣吧
下面看 回調

窗體不變

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        Thread writeThread;
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            writeThread = new Thread(c);
            writeThread.Start();
        }
        private void c()
        {
            textBox1.Text = "123456";
        }
    }
}

而後啓動

程序---崩了

好如今

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        Thread writeThread;
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;////////**************************加了這一句
            writeThread = new Thread(c);
            writeThread.Start();
        }
        private void c()
        {
            textBox1.Text = "123456";
        }
    }
}

滿血復活了..

 

System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;
上一次程序崩掉是由於C#不讓跨線程調用窗體控件--不讓在一個新的線程裏
調用窗體控件---
textBox1.Text = "123456";就是在使用窗體控件textbox
加上這句System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls
Check  For  Illegal  Cross  Thread  Calls  == false;   不檢查
檢查 對 非法的 交叉 線程 調用
因此就經過了---當本身寫程序時調試可使用,,真正作項目嘛,,,,,因人而異吧

C#提供了幾種種方法來讓咱們在線程裏來操做窗體控件---其它高級語言也提供了相應的方法的
看 回調 來也

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        Thread writeThread;
        delegate void a();//定義回調
        a b;//聲明回調
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            b = new a(d);//實例化回調

            writeThread = new Thread(c);
            writeThread.Start();//啓動C線程
        }
        private void c()
        {
            textBox1.Invoke(b);
        }
        private void d()
        {
            textBox1.Text = "123456";
        }
    }
}
textBox1.Invoke(b);
其實就多了這一句
還能夠這樣
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        Thread writeThread;
        delegate void a(string str);
        a b;
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            b = new a(d);

            writeThread = new Thread(c);
            writeThread.Start();
        }
        private void c()
        {
            textBox1.Invoke(b,"123456");
        }
        private void d(string str)
        {
            textBox1.Text = str;
        }
    }
}

 

textBox1.Invoke(b,能夠帶參數);//這是固定形式,就是這樣用,,,人家規定的
如今放一個按鈕

一啓動

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;

namespace @delegate
{
    public partial class Form1 : Form
    {
        Thread writeThread;
        delegate void a(string str);
        a b;
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            b = new a(d);

            writeThread = new Thread(c);
            writeThread.Start();
        }
        private void c()
        {
            button1.Invoke(b, "123456");
        }
        private void d(string str)
        {
            button1.Text = str;
        }
    }
}

仍是這樣用

就寫這些吧!簡簡單單普普統統,,,如今博客園終於能複製粘貼圖片了........太方便了

相關文章
相關標籤/搜索