數據結構實現(順序表)


#include<stdio.h>
//數據結構之順序表
/*原理:地址連續的存儲單元依此存儲線性表的元素
    基本運算:插入 刪除 查找 排序
    要點:結構的定義很重要,所有的操作都是建立在數據結構之上的。
*/
/*順序表結構體定義:*/
//def數據最大長度
#define Maxsize 100
//定義數據元素類型,可以適應更多類型
typedef int Datatype;
//定義順序表類型,只是定義了一個類型,而不是變量
typedef struct SeqList//此處的SeqList可以不要 {
    Datatype data[Maxsize];
    int last;
}SeqList;

SeqList InitSlist()
{//初始化順序表
    SeqList L;//通過函數調用來申請空間,創建變量
    L.last = -1;//表示表空
    return L;
}

void Creat_Slist(SeqList &L)
{//創建順序表,給data初步賦值
    int flag;//標示數據規模
    scanf("%d",&flag);
    for(int i=0;i<flag;i++)
    {
        scanf("%d",&L.data[i]);
        L.last++;
    }
}
int Getlength(SeqList &L)
{
    return (L.last + 1);
}

int InsertSlist(SeqList &L,int i,Datatype x)
{//在表的第i個位置上插入一個值為 x 的新元素
    /*插入位置是否合理? 1 ≤ i ≤ n+1
    是否有空間? L->last == MAXSIZE-1
    插入流程:
            移動
            插入
            修改長度
*/
    int j;
    if(i<1||i>L.last+2) return -1;
    if(L.last==Maxsize-1)   return  -2;
    for(j=L.last;j>=(i-1);j--)
    {
        L.data[j+1]=L.data[j];
    }
    L.data[i-1]=x;
    L.last++;
    return 1;
}

int Deletelist(SeqList &L,int i)
{//刪除是指在表的第i個位置上刪除一個元素
    /*
    刪除位置是否合理? 1≤i≤n
        表是否為空? L->last==-1
            刪除操作
            刪除元素
            修改長度
    */
    int j;
    if(i<1||i>L.last+1) return -1;
    if(L.last==-1)  return -1;
    for(j=i;j<=L.last;j++)
    {
        L.data[j-1]=L.data[j];
    }
    L.last--;
    return 1;
}

int Findslist(SeqList &L,int i)
{//按序號查找
    if(i<1||i>L.last)   return -1;
    else return L.data[i-1];
}

int Findslist(SeqList &L,int X)
{//在表中找到與x相等的元素,則返回該元素在表中的序號
    if(i<1||i>L.last)   return -1;
    else for(int i=0;i<=L.last;i++)
    {
        if(L.data[i]==X)    return  i+1;
    }
    if(i==L.last+1) return -1;
}

void Findwrong(int num)
{
    if(num==1)
      printf("操作成功");
    else printf("操作失敗");

}
int main()
{
    SeqList L = InitSlist();
    Creat_Slist(L);
    switch (InsertSlist(L,i,x))
    {
        case 1:printf("插入數據成功");
        case -1:printf("位置不合理");
        case -2:printf("溢出,不可插");
        break;
    }
    printf("%d",Getlength(L));
    Findwrong(Deletelist(L,i));
    Findwrong(Findslist(L,i));
}

 

 

順序表的結構體創建:

         

 

 

 

typedef的查找資料獲得:

來源:http://zjf30366.blog.163.com/blog/static/411164582009817101543293/

有種很方便的寫法。

typedef int *p;

p pointer;

這時直接把pointer帶入原式中,取代p然后去掉typedef,得到的結果就是int * pointer;

哈哈,這樣直接替換就很直觀多了。

C語言語法簡單,但內涵卻博大精深;如果在學習時只是止步於表面,那么往往后期會遇到很多困難。typedef是C語言中一個很好用的工具,大量存在於已有代碼中,特別值得一提的是:C++標准庫實現中更是對typedef有着大量的使用。但很多初學者對其的理解僅局限於:typedef用來定義一個已有類型的"別名(alias)"。正是因為有了這樣的理解,才有了后來初學者在typedef int myint和typedef myint int之間的猶豫不決。很多國內大學的C語言課之授課老師也都是如是說的,或者老師講的不夠透徹,導致學生們都是如是理解的。我這里想結合C語言標准文檔以及一些代碼實例,也說說typedef。

int    *p;

這樣的代碼是C語言中最最基礎的一個語句了,大家都知道這個語句聲明了一個變量p,其類型是指向整型的指針(pointer to int);如果在這個聲明的前面加上一個typedef后,整個語義(semantics)又會是如何改變的呢?

typedef  int    *p;

我們先來看看C99標准中關於typedef是如何詮釋的?C99標准中這樣一小段精辟的描述:"In a declaration whose storage-class specifier is typedef, each declarator defines an identifier to be a typedef name that denotes the type specified for the identifier in the way described in xx"。

參照這段描述,並拿typedef  int    *p作為例子來理解:在一個聲明中,如果有存儲類說明符typedef的修飾,標識符p將被定義為了一個typedef name,這個typedef name表示(denotes)一個類型,什么類型呢?就是int *p這個聲明(declarator)中標識符(indentifier)p的類型(int*)。

再比對一下兩個聲明:

int    *p;

typedef  int    *p;

是不是有點"茅舍頓開"的感覺,int *p中, p是一個變量,其類型為pointer to int;在int *p前面增加一個typedef后,p變為一個typedef-name,這個typedef-name所表示的類型就是int *p聲明式中p的類型(int*)。說句白話,typedef讓p去除了普通變量的身份,搖身一變,變成了p的類型的一個typedef-name了。

為了鞏固上面的理解,我們再來看看"C語言參考手冊(C: A Reference Manual)"中的說法:任何declarator(如typedef int   *p)中的indentifier(如p)定義為typedef-name, 其(指代p)表示的類型是declarator為正常變量聲明(指代int  *p)的那個標識符(指代p)的類型(int*)。有些繞嘴,不過有例子支撐:

[例1]

typedef double MYDOUBLE;  

分析:

去掉typedef ,得到正常變量聲明=> double MYDOUBLE;

變量MYDOUBLE的類型為double;

=> "typedef double MYDOUBLE"中MYDOUBLE是類型double的一個typedef-name。

MYDOUBLE    d; <=> d是一個double類型的變量

[例2]

typedef double *Dp;  

分析:

去掉typedef  ,得到正常變量聲明=> double *Dp;

變量Dp的類型為double*,即pointer to double;

=> "typedef double *Dp"中Dp是類型double*的一個typedef-name。

Dp    dptr; <=> dptr是一個pointer to double的變量

[例3]

typedef int* Func(int);

分析:

去掉typedef  ,得到正常變量聲明=> int* Func(int);

變量Func的類型為一個函數標識符,該函數返回值類型為int*,參數類型為int;

=> "typedef int* Func(int)"中Func是函數類型(函數返回值類型為int*,參數類型為int)的一個typedef-name。

Func    *fptr; <=> fptr是一個pointer to function with one int parameter, returning a pointer to int

Func     f;   這樣的聲明意義就不大了。

[例4]

typedef int (*PFunc)(int);

分析:

去掉typedef  ,得到正常變量聲明=> int (*PFunc)(int);

變量PFunc的類型為一個函數指針,指向的返回值類型為int,參數類型為int的函數原型;

=> "typedef int (*PFunc)(int)"中PFunc是函數指針類型(該指針類型指向返回值類型為int,參數類型為int的函數)的一個typedef-name。

PFunc     fptr; <=> fptr是一個pointer to function with one int parameter, returning int

#include "iostream"

using namespace std;

int add(int a,int b){
return (a+b);
}

typedef int (* func)(int ,int ) ;

void main(){
func f = add;
int n = f(1,2);
cout << n << endl;
}

[例5]

typedef    int   A[5];

分析:

去掉typedef ,得到正常變量聲明 => int   A[5];

變量A的類型為一個含有5個元素的整型數組;

=> "typedef    int   A[5]"中A是含有5個元素的數組類型的一個typedef-name。

A   a = {3, 4, 5, 7, 8};

A   b = { 3, 4, 5, 7, 8, 9}; /* 會給出Warning: excess elements in array initializer */

[例6]

typedef    int   (*A)[5]; (注意與typedef    int*    A[5]; 區分)

分析:

去掉typedef ,得到正常變量聲明 => int   (*A)[5];

變量A的類型為pointer to an array with 5 int elements;

=> "typedef    int   (*A)[5]"中A是"pointer to an array with 5 int elements"的一個typedef-name。

int   c[5] = {3, 4, 5, 7, 8};  

A    a = &c;

printf("%d\n", (*a)[0]); /* output: 3 */

如果這樣賦值:

int   c[6] = {3, 4, 5, 7, 8, 9};  

A    a = &c; /* 會有Warning: initialization from incompatible pointer type */

[例7]

typedef struct _Foo_t Foo_t;

分析:

去掉typedef ,得到正常變量聲明 => struct _Foo_t Foo_t;

變量Foo_t的類型為struct _Foo_t;

=> "typedef struct _Foo_t Foo_t"中Foo_t是"struct _Foo_t"的一個typedef-name。

[例8]

typedef   struct { ... // }   Foo_t;

分析:

去掉typedef ,得到正常變量聲明 => struct { ... // }   Foo_t;

變量Foo_t的類型為struct { ... // } ;

=> "typedef   struct { ... // }   Foo_t "中Foo_t是"struct { ... // }"的一個typedef-name。這里struct {...//}是一個無"標志名稱(tag name)"的結構體聲明。

參考資料:

1、"ISOIEC-98991999(E)--Programming Languages--C"之Page 123;

2、C語言參考手冊(中文版) 之 Page 119

c中typedef函數指針的作用:

類型定義的語法可以歸結為一句話:只要在變量定義前面加上typedef,就成了類型定義。這兒的原本應該是變量的東西,就成為了類型。

int integer;     //整型變量
int *pointer;   //整型指針變量
int array [5]; //整型數組變量
int *p_array [5]; //整型指針的數組的變量
int (*array_pointer) [5];//整型數組的指針的變量
int function (int param);//函數定義,也可將函數名看作函數的變量
int *function (int param);//仍然是函數,但返回值是整型指針
int (*function) (int param);//現在就是指向函數的指針了

若要定義相應類型,即為類型來起名字,就是下面的形式:
typedef int integer_t;                      //整型類型
typedef int *pointer_t;     //整型指針類型
typedef int array_t [5]; //整型數組類型
typedef int *p_array_t [5];    //整型指針的數組的類型
typedef int (*array_pointer_t) [5]; //整型數組的指針的類型
typedef int function_t (int param);     //函數類型
typedef int *function_t (int param);    //函數類型
typedef int (*function_t) (int param); //指向函數的指針的類型
注意:上面的函數類型在C中可能會出錯,因為C中並沒有函數類型,它的函數變量會自動退化成函數指針;在C++中好像是可以的。在這里主要說明的是形式上的相似性.

typedef的一般形式為:
typedef   類型     定義名;
在編程中使用typedef目的一般有兩個,一個是給變量一個易記且意義明確的新名字,另一個是簡化一些比較復雜的類型聲明。
其實,在C語言中聲明變量的時候,有個存儲類型指示符(storage-class-specifier),它包括我們熟悉的extern、static、auto、register。在不指定存儲類型指示符的時候,編譯器會根據約定自動取缺省值。另外,存儲類型指示符的位置也是任意的(但要求在變量名和指針*之前),也就是說以下幾行代碼是等價的:
static const int i;
const static int i;
int const static i;
const int static i;
根據C語言規范,在進行句法分析的時候,typedef和存儲類型指示符是等價的!所以,我們把上述使用static的地方替換為typedef:
typedef const int i;
const typedef int i;
int const typedef i;
const int typedef i;
上述代碼的語義是:將i定義為一個類型名,其等價的類型為const int。以后如果我們有i   a代碼,就等價於const int a。對於有指針的地方也是一樣的,比如:
int const typedef *t;那么代碼t   p。就相當於int const *p。
另外,typedef不能和static等存儲類型指示符同時使用,因為每個變量只能有一種存儲類型,所以代碼:typedef static int i;是非法的。

typedef有兩種用法:
一、一般形式,定義已有類型的別名
  typedef   類型    定義名;
二、創建一個新的類型
     typedef   返回值類型   新類型名(參數列表);

1)typedef int NUM[10];//聲明整型數組類型

    NUM n;//定義n為整型數組變量,其中n[0]--n[9]可用

2)typedef char* STRING;//聲明STRING為字符指針類型

    STRING p,s[10];//p為字符指針變量,s為指針數組

3)typedef int (*POINTER)();//聲明POINTER為指向函數的指針類型,該函數返回整型值,沒有參數

    POINTER P1,P2;//p1,p2為POINTER類型的指針變量

 

說明:

      1)用typedef可以聲明各種類型名,但不能用來定義變量,用typedef可以聲明數組類型、字符串類型、使用比較方便。

例如:定義數組,原來是用:int a[10],b[10],c[10],d[10];由於都是一維數組,大小也相同,可以先將此數組類型聲明為一個名字:

typedef int ARR[10];

然后用ARR去定義數組變量:

ARR a,b,c,d;//ARR為數組類型,它包含10個元素。因此a,b,c,d都被定義為一維數組,含10個元素。可以看到,用typedef可以將 數組類型 和 數組變量 分離開來,利用數組類型可以定義多個數組變量。同樣可以定義字符串類型、指針類型等。

      2)用typedef只是對已經存在的類型增加一個類型名,而沒有創造新的類型。

      3)typedef與#define有相似之處,但事實上二者是不同的,#define是在 預編譯 時處理,它只能做簡單的字符串替換,而typedef是在 編譯時 處理的。它並不是做簡單的字符串替換,而是采用如同 定義變量 的方法那樣來 聲明 一個類型。

例如:typedef int COUNT;和#define COUNT int的作用都是用COUNT代表int,單事實上它們二者是不同的。

兩個陷阱:

陷阱一: 
記住,typedef是定義了一種類型的新別名,不同於宏,它不是簡單的字符串替換。比如:
先定義:
typedef char* PSTR;
然后:
int mystrcmp(const PSTR, const PSTR); 
const PSTR實際上相當於const char*嗎?不是的,它實際上相當於char* const。
原因在於const給予了整個指針本身以常量性,也就是形成了常量指針char* const。
簡單來說,記住當const和typedef一起出現時,typedef不會是簡單的字符串替換就行。 
陷阱二: 
typedef在語法上是一個存儲類的關鍵字(如auto、extern、mutable、static、register等一樣),雖然它並不真正影響對象的存儲特性,如:
typedef static int INT2; //不可行
編譯將失敗,會提示“指定了一個以上的存儲類”。

1.typedef  函數指針的使用方法

(1)typedef 首先是用來定義新的類型,i.e typedef struct {.....}mystruct; 在以后引用時,就可以用 mystruct 來定義自己的結構體,mystruct structname1,mystruct structname2.

(2)typedef 常用的地方,就在定義函數指針,行為和宏定義類似,用實際類型替換同義字,但是有區別: typedef 在編譯時被解釋,因此讓編譯器來應付超越預處理器能力的文本替換

案例一: 
通常講,typedef要比#define要好,特別是在有指針的場合。請看例子: 
typedef char *pStr1; 
#define pStr2 char *; 
pStr1 s1, s2; 
pStr2 s3, s4; 
在上述的變量定義中,s1、s2、s3都被定義為char *,而s4則定義成了char,不是我們所預期的指針變量,根本原因就在於#define只是簡單的字符串替換而typedef則是為一個類型起新名字。 
案例二: 
下面的代碼中編譯器會報一個錯誤,你知道是哪個語句錯了嗎? 
typedef char * pStr; 
char string[4] = "abc"; 
const char *p1 = string; 
const pStr p2 = string; 
p1++; 
p2++; 
是p2++出錯了。這個問題再一次提醒我們:typedef和#define不同,它不是簡單的文本替換。上述代碼中const pStr p2並不等於const char * p2。const pStr p2和const long x本質上沒有區別,都是對變量進行只讀限制,只不過此處變量p2的數據類型是我們自己定義的而不是系統固有類型而已。因此,const pStr p2的含義是:限定數據類型為char *的變量p2為只讀,因此p2++錯誤。

 

用法一:

 typedef  返回類型(*新類型)(參數表)

 typedef int ( * MYFUNCTION )( int,int ); 這種用法一般是在定義函數指針 MYFUNCTION 是一個函數指針類型 有兩個整型的參數,返回一個整型。

在對於這樣的形式,去掉typedef和別名 就剩下了的是原變量的類型 如:int (*)(int ,int); 在函數指針中,抽象得看待函數,函數名其實就是一個地址,函數名指向該函數的代碼在內存的首地址。

用法二: 復雜函數聲明類型

 下面是三個變量的聲明  用typedef 如何來做???

>1 int *(*a[5])(void *,void *);

>2 void (*b[5])(void (*)());

>3 float (*)()(*pa)[10];

分析如下:

>1 int *(*a[5])(void *,void *);

//pFUN是自己建立的類型別名 typedef int *(* pFUN)(void  *,void *); //等價於int *(*a[5])(void *,void *); 

pFUN a[5];  a是一個數組,包含五個元素,這些元素都是函數指針,該函數指針所指的函數的返回值是int的指針 輸入參數有兩個都是void *.

>2 void (*b[5])( void (*)() );

// first 為藍色的 聲明一個新的類型 typedef void (*pFUNParam)( );

//整體聲明一個新類型  typedef void (*pFUN)(FUNParam); 

//使用定義的新類型聲明對象 等價於void (*b[5])( void (*)() ); 

pFUN b[5]; b 是一個含有5個元素的數組,每個元素都是一個函數指針,該函數指針所指的函數的返回值是void.輸入參數是另一個函數指針,這個函數指針沒有參數,返回值為空。在這里套用了連續的函數指針。本身就是一個函數指針,而且參數也是一個函數指針。

>3 float (*)()(*pa)[10];

//first 為上面的藍色表達式聲明一個新類型 typedef float (*pFUN)(); 

//整體聲明一個新類型typedef pFUN (* pFunParam)[10];

//使用定義的新類型來聲明對象 等價與float (*)()(*pa)[10];

pa 是一個指針,指針指向一個含有10個元素的數組,數組的元素是函數指針,函數指針所指的函數沒有輸入參數,返回值為float.

**********************************************

使用typedef簡化復雜的變量聲明
1)、定義一個有10個指針的數組,該指針指向一個函數,該函數有一個整形參數,並返回一個整型?
第一種方法:int (*a[10])(int);
第二種方法:typedef int (*pfunc)(int);
             pfunc a[10];
2)、定義一個有10個指針的數組,該指針指向一個函數,該函數有一個函數指針(不帶參數,返回值為空)參數,並返回空。
第一種方法:void (*a[10])(void (*)(void));
第二種方法:typedef void (*pfuncParam)(void);
               typedef void (*pfunc)(pfuncParam);
pfunc a[10];
3)、一個指向有10個函數指針(不帶參數,返回值為double)數組的指針
第一種方法:double (*)(void) (*p)[10];
第二種方法:typedef double (*pfunc)(void);
             typedef pfunc (*pfuncParam)[10];
             pfuncParam p;

從變量名看起,先往右,再往左,碰到一個圓括號就調轉閱讀的方向;括號內分析完就跳出括號,還是按先右后左的順序,如此循環,直到整個聲明分析完。舉例:
int (*func)(int *p);
首先找到變量名func,外面有一對圓括號,而且左邊是一個*號,這說明func是一個指針;然后跳出這個圓括號,先看右邊,又遇到圓括號,這說明(*func)是一個函數,

所以func是一個指向這類函數的指針,即函數指針,這類函數具有int*類型的形參,返回值類型是int。
int (*func[5])(int *);
func右邊是一個[]運算符,說明func是具有5個元素的數組;func的左邊有一個*,說明func的元素是指針(注意這里的*不是修飾func,而是修飾func[5]的,

原因是[]運算符優先級比*高,func先跟[]結合)。跳出這個括號,看右邊,又遇到圓括號,說明func數組的元素是函數類型的指針,它指向的函數具有int*類型的形參,

返回值類型為int。 
也可以記住2個模式:
type (*)(....)函數指針 
type (*)[]數組指針 

**********************************************

finally 

typedef 使用最多的地方是創建易於記憶的類型名,用它來歸檔程序員的意圖。類型出現在所聲明的變量名字中,位於 ''typedef'' 關鍵字右邊。例如:
typedef int size;此聲明定義了一個 int 的同義字,名字為 size。注意 typedef 並不創建新的類型。它僅僅為現有類型添加一個同義字。你可以在任何需要 int 的上下文中使用 size:
void measure(size * psz); size array[4];size len = file.getlength();std::vector <size> vs; typedef 還可以掩飾符合類型,如指針和數組。例如,你不用象下面這樣重復定義有 81 個字符元素的數組:
char line[81];char text[81];定義一個 typedef,每當要用到相同類型和大小的數組時,可以這樣:
typedef char Line[81]; Line text, secondline;getline(text);同樣,可以象下面這樣隱藏指針語法:
typedef char * pstr;int mystrcmp(pstr, pstr);這里將帶我們到達第一個 typedef 陷阱。標准函數 strcmp()有兩個‘const char *'類型的參數。因此,它可能會誤導人們象下面這樣聲明 mystrcmp():
int mystrcmp(const pstr, const pstr); 這是錯誤的,按照順序,‘const pstr'被解釋為‘char * const'(一個指向 char 的常量指針),而不是‘const char *'(指向常量 char 的指針)。這個問題很容易解決:
typedef const char * cpstr; int mystrcmp(cpstr, cpstr); // 現在是正確的記住:不管什么時候,只要為指針聲明 typedef,那么都要在最終的 typedef 名稱中加一個 const,以使得該指針本身是常量,而不是對象。 
代碼簡化
上面討論的 typedef 行為有點像 #define 宏,用其實際類型替代同義字。不同點是 typedef 在編譯時被解釋,因此讓編譯器來應付超越預處理器能力的文本替換。例如:
typedef int (*PF) (const char *, const char *);這個聲明引入了 PF 類型作為函數指針的同義字,該函數有兩個 const char * 類型的參數以及一個 int 類型的返回值。如果要使用下列形式的函數聲明,那么上述這個 typedef 是不可或缺的:
PF Register(PF pf);Register() 的參數是一個 PF 類型的回調函數,返回某個函數的地址,其署名與先前注冊的名字相同。做一次深呼吸。下面我展示一下如果不用 typedef,我們是如何實現這個聲明的:
int (*Register (int (*pf)(const char *, const char *))) (const char *, const char *); 很少有程序員理解它是什么意思,更不用說這種費解的代碼所帶來的出錯風險了。顯然,這里使用 typedef 不是一種特權,而是一種必需。持懷疑態度的人可能會問:“OK,有人還會寫這樣的代碼嗎?”,快速瀏覽一下揭示 signal()函數的頭文件 < csinal>,一個有同樣接口的函數。 
typedef 和存儲類關鍵字(storage class specifier)
這種說法是不是有點令人驚訝,typedef 就像 auto,extern,mutable,static,和 register 一樣,是一個存儲類關鍵字。這並是說 typedef 會真正影響對象的存儲特性;它只是說在語句構成上,typedef 聲明看起來象 static,extern 等類型的變量聲明。下面將帶到第二個陷阱:
typedef register int FAST_COUNTER; // 錯誤編譯通不過。問題出在你不能在聲明中有多個存儲類關鍵字。因為符號 typedef 已經占據了存儲類關鍵字的位置,在 typedef 聲明中不能用 register(或任何其它存儲類關鍵字)。 
促進跨平台開發
typedef 有另外一個重要的用途,那就是定義機器無關的類型,例如,你可以定義一個叫 REAL 的浮點類型,在目標機器上它可以i獲得最高的精度:
typedef long double REAL; 在不支持 long double 的機器上,該 typedef 看起來會是下面這樣:
typedef double REAL; 並且,在連 double 都不支持的機器上,該 typedef 看起來會是這樣: 、
typedef float REAL; 你不用對源代碼做任何修改,便可以在每一種平台上編譯這個使用 REAL 類型的應用程序。唯一要改的是 typedef 本身。在大多數情況下,甚至這個微小的變動完全都可以通過奇妙的條件編譯來自動實現。不是嗎? 標准庫廣泛地使用 typedef 來創建這樣的平台無關類型:size_t,ptrdiff 和 fpos_t 就是其中的例子。此外,象 std::string 和 std::ofstream 這樣的 typedef 還隱藏了長長的,難以理解的模板特化語法,例如:basic_string<char, char_traits<char>,allocator<char>> 和 basic_ofstream<char, char_traits<char>>。

 


免責聲明!

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



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