函數(shù)指針編譯時出錯怎么辦?
2) 分步解析
有的C語言基礎不是很好的朋友,可能無法一眼看出來這個定義,為了讓新手更容易看懂,我們來看一下下面一個遞進式的定義:
int fun;
這是一個整型變量fun;
int fun();
這是一個函數(shù)fun,參數(shù) :空返回值:int型
int fun(struct touch_message *);
這是一個函數(shù)fun,參數(shù) :struct touch_message *的一個指針返回值:int型
上述的變化都好理解,下面我們將fun做如下修改:
int (*fun)(struct touch_message *);
括號的優(yōu)先級最高,(fun)一旦如此定義,那么fun就要先和結合,所以fun變成了一個指針,
那么該指針指向什么呢?就需要看外面是如何定義的,右邊是(struct touch_message * ),左邊是int,所以說明指針指向的是一個函數(shù),
參數(shù) :struct touch_message *的一個指針返回值:int型
舉例:將函數(shù)my_fun賦值給函數(shù)指針fun。int my_fun(struct touch_message *){}int (*fun)(struct touch_message *);fun = my_fun;
這里有一個隱藏的知識點,函數(shù)名其實也是一個地址,而且賦值的時候函數(shù)類型必須和函數(shù)指針類型一致。
typedef int (*fun)(struct touch_message *);
如果左邊再加上typedef呢?相當于是設置fun為新的類型,我們可以用fun來定義一個函數(shù)指針,該函數(shù)類型同上。
舉例用新的類型定義一個函數(shù)指針變量,并給他賦值。typedef int (*fun)(struct touch_message *);int my_fun(struct touch_message *){}fun fun_ptr;fun_ptr = my_fun;
然后將參數(shù)修改為,touch_message_t,就得到了粉絲的源碼中的樣子,
typedef int (*fun)(touch_message_t);
但是粉絲的源碼中定義的函數(shù)類型缺少了對函數(shù)返回值的描述,所以左側增加一個int或者其他類型即可即可。
3. 函數(shù)指針
函數(shù)指針在linux內核中使用非常頻繁,
比如字符設備,內核給多有的字符設備提供了一個統(tǒng)一的接口,我們對設備的所有操作被抽象成read、write、open、close等,并封裝到結構體struct file_operations 中:
struct file_operations {
struct module *owner;
loff_t (*llseek) (struct file *, loff_t, int);
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
ssize_t (*aio_read) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
ssize_t (*aio_write) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
int (*iterate) (struct file *, struct dir_context *);
unsigned int (*poll) (struct file *, struct poll_table_struct *);
long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
long (*compat_ioctl) (struct file *, unsigned int, unsigned long);
int (*mmap) (struct file *, struct vm_area_struct *);
int (*open) (struct inode *, struct file *);
int (*flush) (struct file *, fl_owner_t id);
int (*release) (struct inode *, struct file *);
int (*fsync) (struct file *, loff_t, loff_t, int datasync);
int (*aio_fsync) (struct kiocb *, int datasync);
int (*fasync) (int, struct file *, int);
int (*lock) (struct file *, int, struct file_lock *);
ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
int (*check_flags)(int);
int (*flock) (struct file *, int, struct file_lock *);
ssize_t (*splice_write)(struct pipe_inode_info *, struct file *, loff_t *, size_t, unsigned int);
ssize_t (*splice_read)(struct file *, loff_t *, struct pipe_inode_info *, size_t, unsigned int);
int (*setlease)(struct file *, long, struct file_lock **);
long (*fallocate)(struct file *file, int mode, loff_t offset,
loff_t len);
int (*show_fdinfo)(struct seq_file *m, struct file *f);
};
那么我們應該如何定義該結構體變量并初始化呢?
static struct file_operations hello_ops =
{
.open = hello_open,
.release = hello_release,
.read = hello_read,
.write = hello_write,
};
函數(shù)定義如下:
static int hello_open (struct inode *inode, struct file *filep)
{
return 0;
}
static int hello_release (struct inode *inode, struct file *filep)
{
return 0;
}
static ssize_t hello_read (struct file *filep, char __user *buf, size_t size, loff_t *pos)
{
return size;
}
static ssize_t hello_write (struct file *filep, const char __user *buf, size_t size, loff_t *pos)
{
return size;
}
注意,函數(shù)的參數(shù)和返回值,必須嚴格按照結構體struct file_operations中的類型定義。

請輸入評論內容...
請輸入評論/評論長度6~500個字
最新活動更多
推薦專題
- 1 UALink規(guī)范發(fā)布:挑戰(zhàn)英偉達AI統(tǒng)治的開始
- 2 北電數(shù)智主辦酒仙橋論壇,探索AI產(chǎn)業(yè)發(fā)展新路徑
- 3 降薪、加班、裁員三重暴擊,“AI四小龍”已折戟兩家
- 4 “AI寒武紀”爆發(fā)至今,五類新物種登上歷史舞臺
- 5 國產(chǎn)智駕迎戰(zhàn)特斯拉FSD,AI含量差幾何?
- 6 光計算迎來商業(yè)化突破,但落地仍需時間
- 7 東陽光:2024年扭虧、一季度凈利大增,液冷疊加具身智能打開成長空間
- 8 地平線自動駕駛方案解讀
- 9 封殺AI“照騙”,“淘寶們”終于不忍了?
- 10 優(yōu)必選:營收大增主靠小件,虧損繼續(xù)又逢關稅,能否乘機器人東風翻身?