C#委托+回調詳解


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

學習C#有一段時間了,不過C#的委托+回調才這兩天才會用,以前只是知道怎么用.前面的一篇文章,函數指針,其實是為這個做鋪墊的,說白了委托就相當於C語言中的函數指針,具體說用委托有什么好處,我也不知道,但是因為你只要學C#那么回調就一定要會,回調是委托的一種.回調多用於線程間....還是用代碼一點點的說明吧!現在認為自己以前不會用委托是因為手太懶,再者沒有太多的必須用C#做的Demo,自己學東西都是用到什么學什么,想做什么東西需要什么知識就學什么,前幾天有了必須要做的C#的Demo,關於檢測TCP通信發過來的數據的.扯了這么多...回調主要的應用是---如果你在一個線程里操作像文本框,按鈕,Label等組件時,會報錯...原因--C#它不讓這樣操作,,,,

看例子

窗體里面就放一個textbox

先這樣寫

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";
        }
    }
}

結果

 

就這一句   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";
        }
    }
}

然后啟動

蹦了

一起動就退出

 說早了.......先說委托

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;
        }
    }
}

還是這樣用

就寫這些吧!簡簡單單普普通通,,,現在博客園終於能復制粘貼圖片了........太方便了


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM