天天看點

嵌入式面試題整理(一)

自己整理了一些來自各個公司嵌入式行業的面試的題目。答案在後面(是自己寫的答案,不敢保證100%正确,哈哈,能力有限)。題目都是近兩年的公司面試題

  1. 關鍵字volatile有什麼含義?并列舉一個例子。
  2. 嵌入式系統總是要使用者變量或寄存器進行位操作。給定一個整變量a,寫兩段代碼,第一個設定a的bit 3,第二個清零a的bit3。在以上兩個操作中,要保持其它位不變。
  3. 嵌入式系統經常具有要求程式員去通路某特定的記憶體位置的特點。在某工程中,要求設定一絕對位址位0x67a9整型變量的值為0xaa66。編譯器是一個純粹的ANSI編譯器。寫代碼去完成這一任務。
  4. 中斷是嵌入式系統中重要的組成部分,這導緻了很多編譯開發商提供一種擴充—讓标準C支援中斷。 具代表事實是,産生一個新的關鍵字__interrupt。下面的代碼就使用了__interrupt關鍵字去定義了一個中斷服務子程式(ISR),請評論一下這段代碼的。

__interrupt double compute _area(double radius)

{

double area = PI*radius *radius;

printf(” Area = %f”, area);

return area;

}

  1. 請問運作test函數會有什麼結果?為什麼?

(1)void getmemory( char *p )

{

P=(char*)malloc( 100 );

}

void test( void )

{

Char *str = null;

getmemory(str);

strcpy( str, “hello world” );

Printf( str );

}

(2)char *getmemory( void )

{

char p[] = “hello world”;

return p;

}

void test( void)

{

char *str = null;

str = getmemory();

printf( str );

}

6、用一句語句實作x是否是2的若幹次幂的判斷?

解答:

Volatile關鍵字修飾變量,表示該變量是一個會意想不到改變的量(在某個特定的時刻就會改變),是以該變量是不能被編譯器優化的(有時編譯器為了操作友善,會把變量儲存在一個寄存器中,使用的時候會直接從寄存器去取該變量的值),要去讀取資料的時候要從儲存變量的記憶體單元去取。避免讀到的資料是有誤的。

例子:在使用多線程程式設計時,同時通路同一個變量就可能造成資料不符的情況

小知識:C語言的原子性:在執行程式的過程中,如果一條語句不能被拆分,則該語句具有原子性。但在C語言中大部分語句是不具備原子性的。比如接下來的一條語句:i++; 剛剛接觸的可能會想,不就是i自加1嗎,這中間還會有其他的操作。答案是,會的,在彙編的時候該語句,包含了三個動作:1.讀取i變量到寄存器,2.對寄存器的值加1,3.把結果儲存到i變量所對應的記憶體中。是以當多段代碼同時要使用i變量的時候就會造成資料的錯誤讀寫。

2.

在做這道前要知道以下的知識點

1&0=0; 1&1=1; 0&0=0;

1|0=1; 1|1=1; 0|0=0;

1^0=1 1^1=0; 0^0=1;

了解這些這道題就比較簡單了;

将a的第三位置1,其他位不變 将a的第三位清0,其他位不變

a=a|(0x1<<2); a=a&(~(0x1<<2))

總結:(假設m=4)

一個整數n,把它的第m個二進制位變成0,其他的二進制位不允許改變

                    n = n&11111111 11111111 11111111 11110111

                    n = n&(~(1<<(m-1)))

一個整數n,把它的第m個二進制位變成1,其他的二進制位不允許改變

                    n = n|00000000 00000000 00000000 00001000

                    n = n|(1<<(m-1))

一個整數n,把它的第m個二進制位反轉(原本1--》0原本0--》1),其他的二進制位不允許改變

                    n = n^00000000 00000000 00000000 00001000

                    n = n^(1<<(m-1)

3.

   置一絕對位址位0x67a9整型變量的值為0xaa66。編譯器是一個純粹的ANSI編譯器。

方法一:

int *p;

p = (int *)0x67a9;

*p =(int) 0xaa55;

通過該方式可以比較明确的完成目标。

方法二:

一些大神也會這樣寫:

*(int * const)(0x67a9) = 0xaa55;

其實意思一樣的。

注:在你用unix來測試該端代碼的時候會發生當段錯誤,可能在開發闆上才能用(來沒有開始接觸開發闆)。

4.

還沒有接觸到底層開發。。。........

5.

(1)void getmemory( char *p )

{

P=(char*)malloc( 100 );

}

void test( void )

{

char *str = null;

getmemory(str);

strcpy( str, “hello world” );

Printf( str );

}

首先要明白的一點是,char *p 這是str的形參,而不是儲存的str指針的位址,是以p和str是兩個”人 ”,p的一切行為和str沒有任何關系。

是以在主函數執行完getmemory(str);後str還是一個指向NULL的一個指針。沒有被配置設定空間。是以執行strcpy( str, “hello world” );會發生錯誤

(2)char *getmemory( void )

{

char p[] = “hello world”;

return p;

}

void test( void)

{

char *str = null;

str = getmemory();

printf( str );

}

這個問題就比較明顯了,p是一個數組,是存放在是一個局部變量,存儲在棧空間中,當getmemory()函數調用完後p的記憶體空間會被釋放掉,str得到的将會是一個野指針。自然不能列印想要的字元串了。

6.

用一句語句實作x是否是2的若幹次幂的判斷?

首先要了解的是如果一個數是2的若幹次幂,那麼按照位來看,就是隻有滿足有且隻有一位是1,其他全0;是以隻要通過以下語句就可以實作了。

(x&(x-1))?(false):(ture)

繼續閱讀