博客專欄

EEPW首頁 > 博客 > 嵌入式軟件工程師測試題21道與簡析

嵌入式軟件工程師測試題21道與簡析

發(fā)布人:創(chuàng)客物聯(lián)網(wǎng) 時間:2019-05-09 來源:工程師 發(fā)布文章

1. C語言中,修飾符volatile含義是什么?其應(yīng)用場合有哪些?

答:volatile關(guān)鍵字的作用

volatile提醒編譯器它后面所定義的變量隨時都有可能改變,因此編譯后的程序每次需要存儲或讀取這個變量的時候,都會直接從變量地址中讀取數(shù)據(jù)。如果沒有volatile關(guān)鍵字,則編譯器可能優(yōu)化讀取和存儲,可能暫時使用寄存器中的值,如果這個變量由別的程序更新了的話,將出現(xiàn)不一致的現(xiàn)象。下面舉例說明。在DSP開發(fā)中,經(jīng)常需要等待某個事件的觸發(fā),所以經(jīng)常會寫出這樣的程序:

short flag;
void test()
{
do1();
while(flag==0);
do2();

}

這段程序等待內(nèi)存變量flag的值變?yōu)?(懷疑此處是0,有點(diǎn)疑問,)之后才運(yùn)行do2()。變量flag的值由別的程序更改,這個程序可能是某個硬件中斷服務(wù)程序。例如:如果某個按鈕按下的話,就會對DSP產(chǎn)生中斷,在按鍵中斷程序中修改flag為1,這樣上面的程序就能夠得以繼續(xù)運(yùn)行。但是,編譯器并不知道flag的值會被別的程序修改,因此在它進(jìn)行優(yōu)化的時候,可能會把flag的值先讀入某個寄存器,然后等待那個寄存器變?yōu)?。如果不幸進(jìn)行了這樣的優(yōu)化,那么while循環(huán)就變成了死循環(huán),因為寄存器的內(nèi)容不可能被中斷服務(wù)程序修改。為了讓程序每次都讀取真正flag變量的值,就需要定義為如下形式:

volatile short flag;

需要注意的是,沒有volatile也可能能正常運(yùn)行,但是可能修改了編譯器的優(yōu)化級別之后就又不能正常運(yùn)行了。因此經(jīng)常會出現(xiàn)debug版本正常,但是release版本卻不能正常的問題。所以為了安全起見,只要是等待別的程序修改某個變量的話,就加上volatile關(guān)鍵字。

2. 請問TCP/IP協(xié)議分為哪幾層?FTP協(xié)議在哪一層?

答:


TCP/IP整體構(gòu)架概述


TCP/IP協(xié)議并不完全符合OSI的七層參考模型。傳統(tǒng)的開放式系統(tǒng)互連參考模型,是一種通信協(xié)議的7層抽象的參考模型,其中每一層執(zhí)行某一特定任務(wù)。該模型的目的是使各種硬件在相同的層次上相互通信。這7層是:物理層、數(shù)據(jù)鏈路層、網(wǎng)絡(luò)層、傳輸層、會話層、表示層和應(yīng)用層。而TCP/IP通訊協(xié)議采用了4層的層級結(jié)構(gòu),每一層都呼叫它的下一層所提供的網(wǎng)絡(luò)來完成自己的需求。這4層分別為:


應(yīng)用層:應(yīng)用程序間溝通的層,如簡單電子郵件傳輸(SMTP)、文件傳輸協(xié)議(FTP)、網(wǎng)絡(luò)遠(yuǎn)程訪問協(xié)議(Telnet)等。


傳輸層:在此層中,它提供了節(jié)點(diǎn)間的數(shù)據(jù)傳送服務(wù),如傳輸控制協(xié)議(TCP)、用戶數(shù)據(jù)報協(xié)議(UDP)等,TCP和UDP給數(shù)據(jù)包加入傳輸數(shù)據(jù)并把它傳輸?shù)较乱粚又?,這一層負(fù)責(zé)傳送數(shù)據(jù),并且確定數(shù)據(jù)已被送達(dá)并接收。


互連網(wǎng)絡(luò)層:負(fù)責(zé)提供基本的數(shù)據(jù)封包傳送功能,讓每一塊數(shù)據(jù)包都能夠到達(dá)目的主機(jī)(但不檢查是否被正確接收),如網(wǎng)際協(xié)議(IP)。


網(wǎng)絡(luò)接口層:對實(shí)際的網(wǎng)絡(luò)媒體的管理,定義如何使用實(shí)際網(wǎng)絡(luò)(如Ethernet、SerialLine等)來傳送數(shù)據(jù)。

3. 在網(wǎng)絡(luò)應(yīng)用中,函數(shù)htons,htonl,ntohs,ntohl的作用是什么?


答:

htons


htons函數(shù)用來轉(zhuǎn)換u_short來自主機(jī)的TCP / IP網(wǎng)絡(luò)字節(jié)順序(即big-endian )的. u_short htons ( u_short hostshort ) ; 參數(shù)hostshort [ ] 16位元數(shù)的主機(jī)字節(jié)順序. 返回值的htons函數(shù)返回值的TCP /IP網(wǎng)絡(luò)字節(jié)順序. 須知htons函數(shù)有一個16位號碼主機(jī)字節(jié)順序并返回一個16位數(shù)字網(wǎng)絡(luò)字節(jié)命令中使用的TCP /IP網(wǎng)絡(luò).


htonl


將主機(jī)的無符號長整形數(shù)轉(zhuǎn)換成網(wǎng)絡(luò)字節(jié)順序。

#include <winsock.h>

u_long PASCAL FAR htonl( u_long hostlong);

hostlong:主機(jī)字節(jié)順序表達(dá)的32位數(shù)。

注釋:

  本函數(shù)將一個32位數(shù)從主機(jī)字節(jié)順序轉(zhuǎn)換成網(wǎng)絡(luò)字節(jié)順序。

  返回值:htonl()返回一個網(wǎng)絡(luò)字節(jié)順序的值。

  參見:htons(), ntohl(), ntohs().


ntohs


將一個無符號短整形數(shù)從網(wǎng)絡(luò)字節(jié)順序轉(zhuǎn)換為主機(jī)字節(jié)順序。

#include<winsock.h>

u_short PASCAL FAR ntohs(u_short netshort);

netshort:一個以網(wǎng)絡(luò)字節(jié)順序表達(dá)的16位數(shù)。

  注釋:

  本函數(shù)將一個16位數(shù)由網(wǎng)絡(luò)字節(jié)順序轉(zhuǎn)換為主機(jī)字節(jié)順序。

  返回值:ntohs()返回一個以主機(jī)字節(jié)順序表達(dá)的數(shù)。

  參見:htonl(), htons(),ntohl().


ntohl


將一個無符號長整形數(shù)從網(wǎng)絡(luò)字節(jié)順序轉(zhuǎn)換為主機(jī)字節(jié)順序。

 ?。nclude <winsock.h>

u_long PASCAL FAR ntohl(u_long netlong);

netlong:一個以網(wǎng)絡(luò)字節(jié)順序表達(dá)的32位數(shù)。

  注釋:

  本函數(shù)將一個32位數(shù)由網(wǎng)絡(luò)字節(jié)順序轉(zhuǎn)換為主機(jī)字節(jié)順序。

  返回值:ntohl()返回一個以主機(jī)字節(jié)順序表達(dá)的數(shù)。

  參見:htonl(), htons(),ntohs().

4. C語言中static函數(shù)與普通函數(shù)的區(qū)別是什么?


答:

1.static有什么用途?(請至少說明兩種)


1)在函數(shù)體,一個被聲明為靜態(tài)的變量在這一函數(shù)被調(diào)用過程中維持其值不變。


2) 在模塊內(nèi)(但在函數(shù)體外),一個被聲明為靜態(tài)的變量可以被模塊內(nèi)所有函數(shù)訪問,但不能被模塊外其它函數(shù)訪問。它是一個本地的全局變量。


3) 在模塊內(nèi),一個被聲明為靜態(tài)的函數(shù)只可被這一模塊內(nèi)的其它函數(shù)調(diào)用。那就是,這個函數(shù)被限制在聲明它的模塊的本地范圍內(nèi)使用

static全局變量與普通的全局變量有什么區(qū)別?static局部變量和普通局部變量有什么區(qū)別?static函數(shù)與普通函數(shù)有什么區(qū)別?


全局變量(外部變量)的說明之前再冠以static 就構(gòu)成了靜態(tài)的全局變量。全局變量本身就是靜態(tài)存儲方式,靜態(tài)全局變量當(dāng)然也是靜態(tài)存儲方式。這兩者在存儲方式上并無不同。這兩者的區(qū)別雖在于非靜態(tài)全局變量的作用域是整個源程序,當(dāng)一個源程序由多個源文件組成時,非靜態(tài)的全局變量在各個源文件中都是有效的。而靜態(tài)全局變量則限制了其作用域,即只在定義該變量的源文件內(nèi)有效,在同一源程序的其它源文件中不能使用它。由于靜態(tài)全局變量的作用域局限于一個源文件內(nèi),只能為該源文件內(nèi)的函數(shù)公用,因此可以避免在其它源文件中引起錯誤。從以上分析可以看出,把局部變量改變?yōu)殪o態(tài)變量后是改變了它的存儲方式即改變了它的生存期。把全局變量改變?yōu)殪o態(tài)變量后是改變了它的作用域,限制了它的使用范圍。


static函數(shù)與普通函數(shù)作用域不同。僅在本文件。只在當(dāng)前源文件中使用的函數(shù)應(yīng)該說明為內(nèi)部函數(shù)(static),內(nèi)部函數(shù)應(yīng)該在當(dāng)前源文件中說明和定義。對于可在當(dāng)前源文件以外使用的函數(shù),應(yīng)該在一個頭文件中說明,要使用這些函數(shù)的源文件要包含這個頭文件static全局變量與普通的全局變量有什么區(qū)別:static全局變量只初使化一次,防止在其他文件單元中被引用; static局部變量和普通局部變量有什么區(qū)別:static局部變量只被初始化一次,下一次依據(jù)上一次結(jié)果值;static函數(shù)與普通函數(shù)有什么區(qū)別:static函數(shù)在內(nèi)存中只有一份,普通函數(shù)在每個被調(diào)用中維持一份拷貝

5. 請實(shí)現(xiàn)內(nèi)存復(fù)制函數(shù)voidmemcpy(void *dst const void *src,int size)

原型:extern void*memcpy(void *dest, void *src, unsigned int count);
用法:#include<string.h>
功能:由src所指內(nèi)存區(qū)域復(fù)制count個字節(jié)到dest所指內(nèi)存區(qū)域。
說明:src和dest所指內(nèi)存區(qū)域不能重疊,函數(shù)返回指向dest的指針。

舉例:
// memcpy.c

#include <syslib.h>
#include <string.h>
int main(int argc, char* argv[])
{
char *s="Golden Global View";
char d[20];
clrscr();
memcpy(d,s,strlen(s));
d[strlen(s)]='\0';
printf("%s",d);
getchar();
return 0;
}
截取view
#include <string.h>
int main(int argc, char* argv[])
{
char *s="Golden Global View";
char d[20];
memcpy(d,s+14,4);
//memcpy(d,s+14*sizeof(char),4*sizeof(char));也可
d[5]='\0';
printf("%s",d);
getchar();
return 0;
}
輸出結(jié)果:
View

初始化數(shù)組
char msg[10];
memcpy(msg,0,sizeof(memcpy));

方法2:

寫一個內(nèi)存拷貝函數(shù)

// 考慮重疊的狀況
void* _memcpy(void* dest, void* src, int len)
{
if(!dest || !src || !len || dest == src)
return dest;

char* pdest = static_cast<char*>(dest);
char* psrc = static_cast<char*>(src);

// dest 在 src + len 范圍內(nèi)
if(pdest > psrc && pdest < (psrc + len))
{
// 先備份被覆蓋部分
int need = psrc + len - pdest;
int offset = pdest - psrc;
char* pcache = new char[need];

int i = 0;
for (i = 0; i < need; ++i)
pcache[i] = psrc[offset + i]; 

// 拷貝起始部分
for (i = 0; i < offset; ++i)
pdest[i] = psrc[i];

// 拷貝剩余部分
for (i = 0; i < need; ++i)
pdest[offset + i] = pcache[i]; 

delete[] pcache;
}
else
{
for (int i = 0; i < len; ++i)
pdest[i] = psrc[i];
}

return dest;
}


6.32位機(jī)器上,假設(shè)有一個32位數(shù)字0x1234abcd保存在0x00000000開始的內(nèi)存中,那么在little endian和big endian的機(jī)器上,按字節(jié)該整數(shù)在內(nèi)存中存放的順序是怎么樣的?



答:大端模式:數(shù)據(jù)的高字節(jié)存儲在內(nèi)存地址的低字節(jié),(正常存儲)小端模式:數(shù)據(jù)的高字節(jié)存儲在內(nèi)存地址的高字節(jié).

littleendian:0x00000000-0x000000003h:cd,ab,34,12

big endian: 0x00000000-0x000000003h: 12,34,ab,cd


7.ISO七層模型是什么,tcp/udp屬于哪一層?


答:物理層-數(shù)據(jù)鏈路層-網(wǎng)絡(luò)層-傳輸層-會話層-表示層-應(yīng)用層,tcp/udp工作在傳輸層。



8.下面是一個中斷服務(wù)程序的代碼,請指出有那些問題?


_interrupt double compute_area(double radius)

{

double area=PI*radius*radius;

return area;

}


答:中斷子程序不能有返回值,去掉return area; compute_area之前的double關(guān)鍵字。這個函數(shù)有太多的錯誤了,以至讓人不知從何說起了: 

1)ISR 不能返回一個值。如果你不懂這個,那么你不會被雇用的。


2) ISR 不能傳遞參數(shù)。如果你沒有看到這一點(diǎn),你被雇用的機(jī)會等同第一項。


3) 在許多的處理器/編譯器中,浮點(diǎn)一般都是不可重入的。有些處理器/編譯器需要讓額處的寄存器入棧,有些處理器/編譯器就是不允許在ISR中做浮點(diǎn)運(yùn)算。此外,ISR應(yīng)該是短而有效率的,在ISR中做浮點(diǎn)運(yùn)算是不明智的。


4) 與第三點(diǎn)一脈相承,printf()經(jīng)常有重入和性能上的問題。如果你丟掉了第三和第四點(diǎn),我不會太為難你的。不用說,如果你能得到后兩點(diǎn),那么你的被雇用前景越來越光明了。


9.多任務(wù)系統(tǒng)中,常見的任務(wù)通訊機(jī)制有哪些?


答:操作系統(tǒng)還提供進(jìn)程間的通訊機(jī)制來幫助完成這樣的任務(wù)。Linux中常見的進(jìn)程間通訊機(jī)制有:信號、管道、共享內(nèi)存、信號量和套接字等。


10.請實(shí)現(xiàn)內(nèi)存復(fù)制函數(shù)memcpy(void *dst,const void*src,int size).


答:


舉例:

// memcpy.c
#include <syslib.h>
#include <string.h>
int main(int argc, char* argv[])
{
char *s="Golden Global View";
char d[20];
clrscr();

memcpy(d,s,strlen(s));
d[strlen(s)]='\0';
printf("%s",d);
getchar();
return 0;
}

截取view
#include <string.h>
int main(int argc, char* argv[])
{
char *s="Golden Global View";
char d[20];
memcpy(d,s+14,4);
//memcpy(d,s+14*sizeof(char),4*sizeof(char));也可
d[5]='\0';
printf("%s",d);
getchar();
return 0;
}
輸出結(jié)果:
View

初始化數(shù)組

char msg[10];

memcpy(msg,0,sizeof(memcpy));

11.請列舉主流linux的發(fā)布版本(四個以上)。


答:


(1)Redhat有兩大Linux產(chǎn)品系列,其一是免費(fèi)的Fedora Core系列主要用于桌面版本,提供了較多新特性的支持。另外一個產(chǎn)品系列是收費(fèi)的Enterprise系列,這個系列分成:AS/ES/WS等分支。


(2)Advanced Server,縮寫即AS。AS在標(biāo)準(zhǔn)Linux內(nèi)核的基礎(chǔ)上,做了性能上的增強(qiáng),并提高了可靠性,集成了眾多常見服務(wù)器的驅(qū)動程序。可輕松識別IBM/DELL/HP等常見機(jī)架式服務(wù)器的磁盤陣列卡等設(shè)備。


12.當(dāng)前l(fā)inux最主流的兩大桌面環(huán)境是什么,兩者區(qū)別是什么?


答:KDE與GNOME是目前Linux/UNIX系統(tǒng)最流行的圖形操作環(huán)境


13.linux系統(tǒng)下主要三類設(shè)備文件類型是什么?


答:塊設(shè)備、字符設(shè)備、網(wǎng)絡(luò)設(shè)備。


Linux 中的設(shè)備有2種類型:字符設(shè)備(無緩沖且只能順序存取)、塊設(shè)備(有緩沖且可以隨機(jī)存取)。每個字符設(shè)備和塊設(shè)備都必須有主、次設(shè)備號,主設(shè)備號相同的設(shè)備是同類設(shè)備(使用同一個驅(qū)動程序)。這些設(shè)備中,有些設(shè)備是對實(shí)際存在的物理硬件的抽象,而有些設(shè)備則是內(nèi)核自身提供的功能(不依賴于特定的物理硬件,又稱為"虛擬設(shè)備")。


14.以太網(wǎng)的MTU是多大?


答:通常意義上的以太網(wǎng)MTU是指沒有以太網(wǎng)header和FCS的以太網(wǎng)payload部分,IEEE802規(guī)定了大小為0~1500字節(jié);所以,二層以太網(wǎng)幀長應(yīng)該為這個長度加上18B(6B的DA、6B的SA和2B的Length/Etype以及4B的FCS),這樣大小應(yīng)該<1518;在Dot1Q情況下,應(yīng)該在加上4B的802.1Q的Tag,即應(yīng)該<1522B;在MPLS VPN環(huán)境中,IGP標(biāo)簽和VPN標(biāo)簽各是4B,所以,作為中間環(huán)節(jié)的交換機(jī)如果不支持Jumbo幀的話,就需要手工配置MTU=1500+N*4(N為MPLS Tag層數(shù))。


15. ARP協(xié)議作用


答:IP數(shù)據(jù)包常通過以太網(wǎng)發(fā)送。以太網(wǎng)設(shè)備并不識別32位IP地址:它們是以48位以太網(wǎng)地址傳輸以太網(wǎng)數(shù)據(jù)包的。因此,IP驅(qū)動器必須把IP目的地址轉(zhuǎn)換成以太網(wǎng)網(wǎng)目的地址。在這兩種地址之間存在著某種靜態(tài)的或算法的映射,常常需要查看一張表。地址解析協(xié)議(Address Resolution Protocol,ARP)就是用來確定這些映象的協(xié)議。


ARP工作時,送出一個含有所希望的IP地址的以太網(wǎng)廣播數(shù)據(jù)包。目的地主機(jī),或另一個代表該主機(jī)的系統(tǒng),以一個含有IP和以太網(wǎng)地址對的數(shù)據(jù)包作為應(yīng)答。發(fā)送者將這個地址對高速緩存起來,以節(jié)約不必要的ARP通信。


如果有一個不被信任的節(jié)點(diǎn)對本地網(wǎng)絡(luò)具有寫訪問許可權(quán),那么也會有某種風(fēng)險。這樣一臺機(jī)器可以發(fā)布虛假的ARP報文并將所有通信都轉(zhuǎn)向它自己,然后它就可以扮演某些機(jī)器,或者順便對數(shù)據(jù)流進(jìn)行簡單的修改。ARP機(jī)制常常是自動起作用的。在特別安全的網(wǎng)絡(luò)上, ARP映射可以用固件,并且具有自動抑制協(xié)議達(dá)到防止干擾的目的。

16. 編寫一個簡單的ECHO服務(wù)器


答:如下代碼是一個簡單的echo服務(wù)器,它示例的一些Winsock函數(shù)的用法。

#include <iostream>
using namespace std;

#ifdef WIN32

# include <WinSock2.h>
/// 也可在setting/liker/input/Additional dependencies加入導(dǎo)入庫引用
# pragma comment(lib, "ws2_32.lib")

/// 包含AcceptEx,TransmitFile,DisconnectEx等函數(shù)
# include <mswsock.h>
# pragma comment(lib,"Mswsock.lib")

# define PRINT_SOCKET_ERROR(FUNC) 
do


cout << "<" << #FUNC << "> Call Error!" << "" 
<< "Error Number : " << WSAGetLastError() << endl; 
} while(0)

#endif // WIN32

#define PORT 5432

int main(int argc, char* argv[])
{
// = 初始化Lib庫
int nRet;
WSAData wsaData;
if ((nRet = WSAStartup(0x0202, &wsaData)) != 0)

PRINT_SOCKET_ERROR(WSAStartup);
return -1;
}

// = 設(shè)置監(jiān)聽socket句柄
SOCKET hListen;

/// #1 
if ((hListen = socket(AF_INET, SOCK_STREAM, 0)) == INVALID_SOCKET)
{
PRINT_SOCKET_ERROR(socket);
return -1;
}

/// #2
SOCKADDR_IN hostAddr; 
{
u_long nIp = htonl(INADDR_ANY);
{
/// 相關(guān)函數(shù)示例
char szHostName[256] = {0};
gethostname(szHostName, 256);
hostent* thisHost = gethostbyname(szHostName);
nIp = inet_addr(inet_ntoa(*((in_addr*)thisHost->h_addr_list[0])));
}
hostAddr.sin_family = AF_INET;
hostAddr.sin_port = htons(PORT);
hostAddr.sin_addr.s_addr = nIp;


if ((nRet = bind(hListen, (PSOCKADDR)&hostAddr, sizeof(SOCKADDR_IN))) != 0)
{
PRINT_SOCKET_ERROR(bind);
return -1;
}

/// #3
if ((nRet = listen(hListen, 5)) != 0)
{
PRINT_SOCKET_ERROR(listen);
return -1;
}
else
{
cout << "Listening at " << inet_ntoa(hostAddr.sin_addr) 
<< ":" << ntohs(hostAddr.sin_port) << endl;
}

// = 處理請求
SOCKET hConnected;
SOCKADDR_IN remoteAddr;
int remote_addr_size = sizeof(SOCKADDR_IN);
if ((hConnected = accept(hListen, (PSOCKADDR)&remoteAddr, &remote_addr_size)) == INVALID_SOCKET)
{
PRINT_SOCKET_ERROR(accept);
return -1;
}
else
{
cout << "Connect from " << inet_ntoa(remoteAddr.sin_addr) 
<< ":" << ntohs(remoteAddr.sin_port) << endl; 
}

// = 數(shù)據(jù)收發(fā)
do
{
char buffer[1024] = {0};
int recv_bytes = recv(hConnected, buffer, 1024, 0);
if (recv_bytes == -1)
{
PRINT_SOCKET_ERROR(recv);
break;
}
else if (recv_bytes == 0)

/// 客戶端關(guān)閉了鏈接
cout << "Connection closed from " << inet_ntoa(remoteAddr.sin_addr)
<< ":" << ntohs(remoteAddr.sin_port) << endl;
closesocket(hConnected); 
break;
}
else
{
buffer[recv_bytes] = 0;
cout << buffer; cout.flush();
int send_bytes = send(hConnected, buffer, recv_bytes, 0);
if (send_bytes != recv_bytes)
{
cout << "Not send out all data!" << endl;
}
}

} while (1);

closesocket(hListen); 

// = 清理Lib庫
if (WSACleanup() != 0)
{
PRINT_SOCKET_ERROR(WSACleanup);
return -1;
}

return 0;
};


17. 什么是DSP?請簡述DSP與通用CPU的差別。


答:以微處理器解度看,兩者似乎沒什么差別. 但二者所專注的鄰域卻完全不一樣. 通用的CPU專注的事務(wù)處理,對實(shí)時性的支持相對DSP差了很多.雖然一些通用CPU也在加入一些高性能的運(yùn)算支持,但它和DSP相比還是天差地遠(yuǎn).

比如現(xiàn)在的通用DSP可以在幾個us內(nèi)完成1024點(diǎn)的復(fù)數(shù)FFT,通用處理器是咋都達(dá)不到這水平(現(xiàn)在的情況),雖然其系統(tǒng)時鐘有可能比DSP高出一個數(shù)量級.但一個周期完成多個復(fù)雜操作和多個周期完成一個操作的差別是很大的.

DSP完成的都是算法密集性的事務(wù),可能工作比較單一而簡單,但對實(shí)時要求很高很高.必需要某個確定的時間內(nèi)(有可能就那么幾u(yù)s)完成所需要的所有操作.


通用CPU對事務(wù)管理很突出, 在這方面的能力比DSP支持得要好得多.


所以大多數(shù)的高性能系統(tǒng)會是DSP和通用CPU的結(jié)合.充分利用各自的優(yōu)點(diǎn).

由于DSP特注重高性能,其結(jié)構(gòu)在同等條件比通用CPU要復(fù)雜得多,設(shè)計上也更困難.因此其價格相比通用CPU要貴那么一些(只是通用DSP和通用CPU 相比),一些很單一功能的DSP可是很便宜的,比通用CPU還便宜. 通用DSP因為要考慮通用,所以結(jié)構(gòu)和專用的也是不一樣的.從表面上來看,DSP與標(biāo)準(zhǔn)微處理器有許多共同的地方:一個以ALU為核心的處理器、地址和數(shù)據(jù)總線、RAM、ROM以及I/O端口,從廣義上講,DSP、微處理器和微控制器(單片機(jī))等都屬于處理器,可以說DSP是一種CPU。但DSP和一般的CPU又不同:

首先是體系結(jié)構(gòu):CPU是馮.諾伊曼結(jié)構(gòu)的,而DSP有分開的代碼和數(shù)據(jù)總線即“哈佛結(jié)構(gòu)”,這樣在同一個時鐘周期內(nèi)可以進(jìn)行多次存儲器訪問——這是因為 數(shù)據(jù)總線也往往有好幾組。有了這種體系結(jié)構(gòu),DSP就可以在單個時鐘周期內(nèi)取出一條指令和一個或者兩個(或者更多)的操作數(shù)。

標(biāo)準(zhǔn)化和通用性:CPU的標(biāo)準(zhǔn)化和通用性做得很好,支持操作系統(tǒng),所以以CPU為核心的系統(tǒng)方便人機(jī)交互以及和標(biāo)準(zhǔn)接口設(shè)備通信,非常方便而且不需要硬件 開發(fā)了;但這也使得CPU外設(shè)接口電路比較復(fù)雜,DSP主要還是用來開發(fā)嵌入式的信號處理系統(tǒng)了,不強(qiáng)調(diào)人機(jī)交互,一般不需要很多通信接口,因此結(jié)構(gòu)也較為簡單,便于開發(fā)。如果只是著眼于嵌入式應(yīng)用的話,嵌入式CPU和DSP的區(qū)別應(yīng)該只在于一個偏重控制一個偏重運(yùn)算了。

流水線結(jié)構(gòu):大多數(shù)DSP都擁有流水結(jié)構(gòu),即每條指令都由片內(nèi)多個功能單元分別完成取指、譯碼、取數(shù)、執(zhí)行等步驟,這樣可以大大提高系統(tǒng)的執(zhí)行效率。但流水線的采用也增加了軟件設(shè)計的難度,要求設(shè)計者在程序設(shè)計中考慮流水的需要。

快速乘法器:信號處理算法往往大量用到乘加(multiply-accumulate,MAC)運(yùn)算。DSP有專用的硬件乘法器,它可以在一個時鐘周期內(nèi) 完成MAC運(yùn)算。硬件乘法器占用了DSP芯片面積的很大一部分。(與之相反,通用CPU采用一種較慢的、迭代的乘法技術(shù),它可以在多個時鐘周期內(nèi)完成一次 乘法運(yùn)算,但是占用了較少了硅片資源)。

地址發(fā)生器:DSP有專用的硬件地址發(fā)生單元,這樣它可以支持許多信號處理算法所要求的特定數(shù)據(jù)地址模式。這包括前(后)增(減)、環(huán)狀數(shù)據(jù)緩沖的模地址以及FFT的比特倒置地址。地址發(fā)生器單元與主ALU和乘法器并行工作,這就進(jìn)一步增加了DSP可以在一個時鐘周期內(nèi)可以完成的工作量。

硬件輔助循環(huán):信號處理算法常常需要執(zhí)行緊密的指令循環(huán)。對硬件輔助循環(huán)的支持,可以讓DSP高效的循環(huán)執(zhí)行代碼塊而無需讓流水線停轉(zhuǎn)或者讓軟件來測試循環(huán)終止條件。

低功耗:DSP的功耗較小,通常在0.5W到4W,采用低功耗的DSP甚至只有0.05W,可用電池供電,很適合嵌入式系統(tǒng);而CPU的功耗通常在20W以上。


18. 靜態(tài)局部變量與普通局部變量的區(qū)別是什么?


答:全局變量(外部變量)的說明之前再冠以static就構(gòu)成了靜態(tài)的全局變量。全局變量本身就是靜態(tài)存儲方式, 靜態(tài)全局變量當(dāng)然也是靜態(tài)存儲方式。 這兩者在存儲方式上并無不同。這兩者的區(qū)別雖在于非靜態(tài)全局變量的作用域是整個源程序,當(dāng)一個源程序由多個源文件組成時,非靜態(tài)的全局變量在各個源文件中都是有效的。 而靜態(tài)全局變量則限制了其作用域, 即只在定義該變量的源文件內(nèi)有效, 在同一源程序的其它源文件中不能使用它。由于靜態(tài)全局變量的作用域局限于一個源文件內(nèi),只能為該源文件內(nèi)的函數(shù)公用,因此可以避免在其它源文件中引起錯誤。


從以上分析可以看出,把局部變量改變?yōu)殪o態(tài)變量后是改變了它的存儲方式即改變了它的生存期。把全局變量改變?yōu)殪o態(tài)變量后是改變了它的作用域, 限制了它的使用范圍。


static函數(shù)與普通函數(shù)作用域不同。僅在本文件。只在當(dāng)前源文件中使用的函數(shù)應(yīng)該說明為內(nèi)部函數(shù)(static),內(nèi)部函數(shù)應(yīng)該在當(dāng)前源文件中說明和定義。對于可在當(dāng)前源文件以外使用的函數(shù),應(yīng)該在一個頭文件中說明,要使用這些函數(shù)的源文件要包含這個頭文件

static全局變量與普通的全局變量有什么區(qū)別:static全局變量只初使化一次,防止在其他文件單元中被引用;static局部變量和普通局部變量有什么區(qū)別:static局部變量只被初始化一次,下一次依據(jù)上一次結(jié)果值;static函數(shù)與普通函數(shù)有什么區(qū)別:static函數(shù)在內(nèi)存中只有一份,普通函數(shù)在每個被調(diào)用中維持一份拷貝。程序的局部變量存在于(堆棧)中,全局變量存在于(靜態(tài)區(qū) )中,動態(tài)申請數(shù)據(jù)存在于( 堆)中。

19.請說明DSP中定點(diǎn)、浮點(diǎn)的概念


答:定點(diǎn)數(shù)指小數(shù)點(diǎn)在數(shù)中的位置是固定不變的,通常有定點(diǎn)整數(shù)和定點(diǎn)小數(shù)。在對小數(shù)點(diǎn)位置作出選擇之后,運(yùn)算中的所有數(shù)均應(yīng)統(tǒng)一為定點(diǎn)整數(shù)或定點(diǎn)小數(shù),在運(yùn)算中不再考慮小數(shù)問題。


(1)定義:數(shù)據(jù)中小數(shù)點(diǎn)位置固定不變的數(shù) (2)種類:定點(diǎn)整數(shù)(3)小數(shù)點(diǎn)在符號位與有效位之間。


注:定點(diǎn)數(shù)受字長的限制,超出范圍會有溢出。


20.定點(diǎn)數(shù)與浮點(diǎn)數(shù)區(qū)別


答:定點(diǎn)表示法運(yùn)算直觀,但數(shù)的表示范圍較小,不同的數(shù)運(yùn)算時要考慮比例因子的選取,以防止溢出。浮點(diǎn)表示法運(yùn)算時可以不考慮溢出,但浮點(diǎn)運(yùn)算,編程較難。要掌握定、浮點(diǎn)數(shù)的轉(zhuǎn)換方法及浮點(diǎn)數(shù)規(guī)格化方法。

21.什么是cache?cache有哪些操作?


答:cache n. 高速緩沖存儲器 一種特殊的存儲器子系統(tǒng),其中復(fù)制了頻繁使用的數(shù)據(jù)以利于快速訪問。存儲器的高速緩沖存儲器存儲了頻繁訪問的 RAM 位置的內(nèi)容及這些數(shù)據(jù)項的存儲地址。當(dāng)處理器引用存儲器中的某地址時,高速緩沖存儲器便檢查是否存有該地址。如果存有該地址,則將數(shù)據(jù)返回處理器;如果沒有保存該地址,則進(jìn)行常規(guī)的存儲器訪問。因為高速緩沖存儲器總是比主RAM 存儲器速度快,所以當(dāng) RAM 的訪問速度低于微處理器的速度時,常使用高速緩沖存儲器。

*博客內(nèi)容為網(wǎng)友個人發(fā)布,僅代表博主個人觀點(diǎn),如有侵權(quán)請聯(lián)系工作人員刪除。



關(guān)鍵詞:

相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉