C語言程序設計精髓(MOOC第9周 )題

第9周編程題在線測試

NOTE:

  1. 要輸出一個變量的地址值時,要用格式轉換說明符 %p
  2. 空指針 ,即值爲NULL的值,是無效指針。
  3. p = 0 和p = NULL 的區別:p = NULL可以明確說明p是指針變量,而不是一個數值變量,而且空指針並不一定就是指向地址爲0的存儲單元的指針,並非所有編譯器都是用0地址。
  4. warning: 每次用指針時都要對其進行初始化!!!
  5. 判斷素數的時候注意:對於double類型變量x(不是整型)不能執行求餘運算,即不能用 x % i == 0來判斷x是否能被i整除,可以使用 x / i == (int)(x/i)來判斷x是否能被i整除。

1. 重複數字檢查

題目內容:
從鍵盤輸入一個數,檢查這個數中是否有重複出現的數字。如果這個數中有重複出現的數字,則顯示“Repeated digit!”;否則顯示“No repeated digit!”。

函數原型: int CountRepeatNum(int count[], int n);

注: 若有重複數字,則該函數返回重複出現的數字;否則返回-1.

#include<stdio.h>
#define N 10
int CountRepeatNum(int count[], int n);
int main()
{
    int count[N] = {0};
    int n;
    printf("Input n:\n");
    scanf("%d",&n);
    if(CountRepeatNum(count,n) == -1)
        printf("No repeated digit!\n");//返回-1,沒有重複數字

    else
        printf("%d is Repeated digit!\n",CountRepeatNum(count,n));//返回重複數字

    return 0;
}

int CountRepeatNum(int count[], int n)
{
    int i = 0,digit = 0;
    int cnt = 0;
    while(n != 0)
    {
        count[i] = n%10;
        n = n/10;
        i++;
    }
    int a[10] = {0};
    for(int j = 0;j < i; j++)
    {
        a[count[j]]++;
    }
    for(int k = 0;k < 10; k++)
    {
        if(a[k] != 0)
            cnt++;
    }

    if(cnt == i)
        return -1;
    else
    {
        for(int p = 0;p < 10; p++)
        {
            if(a[p] > 1)
            return p;
        }
    }
}

2. 教授的課

題目內容:
教授正在爲一個有N個學生的班級講授離散數學課。他對某些學生缺乏紀律性很不滿意,於是決定:如果課程開始後上課的人數小於K,就取消這門課程。從鍵盤輸入每個學生的到達時間,請編程確定該課程是否被取消。如果該門課程被取消,則輸出“Yes”,否則輸出“No”。假設教授在時刻0開始上課。如果一個學生的到達時間是非正整數,則表示該學生在上課前進入教室。如果一個學生的到達時間是正整數,則表示該學生在上課後進入教室。如果一個學生在時刻0進入教室,也被認爲是在上課前進入教室。假設到達時間的絕對值不超過100,學生數N不超過1000。要求在輸入學生的到達時間之前,先輸入N和K。

函數原型: int IsCancel(int a[], int n, int k);

注: 函數功能:根據數組a中記錄的學生到達時間確定課程是否被取消,取消則返回1,否則返回0

#include<stdio.h>
#define N 1000
int IsCancel(int a[], int n, int k);
int main()
{
    int n,k;
    int a[N] = {0};
    printf("Input n,k:\n");
    scanf("%d,%d",&n,&k);
    getchar();
    for(int i = 0;i < n; i++)
        scanf("%d",&a[i]);
    if(IsCancel(a,n,k))
        printf("YES\n");
    else
        printf("NO\n");
    return 0;
}

int IsCancel(int a[], int n, int k)
{
    int count = 0;
    for(int i = 0;i < n; i++)
    {
        if(a[i] <= 0)
            count++;
    }
    if(count < k)return 1;
    else return 0;
}

3. 尋找鞍點

題目內容:
請編程找出一個M*N矩陣中的鞍點,即該位置上的元素是該行上的最大值,是該列上的最小值。如果矩陣中沒有鞍點,則輸出“No saddle point!”

函數原型: void FindSaddlePoint(int a[][N], int m, int n);

注: 在該函數中輸出有無鞍點的信息。

#include<stdio.h>
#define N 100
void FindSaddlePoint(int a[][N], int m, int n);
int main()
{
    int m,n;
    int a[N][N];
    printf("Input m,n:\n");
    scanf("%d,%d",&m,&n);
    printf("Input matrix:\n");
    for(int i = 0;i < m; i++)
        for(int j = 0;j < n; j++)
            scanf("%d",&a[i][j]);
    FindSaddlePoint(a,m,n);
    return 0;
}

void FindSaddlePoint(int a[][N], int m, int n)
{
    int i,j;
    for(i = 0;i < m; i++)
    {
        int maxpos = 0;
        int flag = 0;
        for(int k = 0;k < n; k++)
        {
            if(a[i][k] > a[i][maxpos])
                maxpos = k;
        }
        for(int p = 0;p < m; p++)
        {
            if(a[p][maxpos] < a[i][maxpos])
                flag = 1;
        }
        if(flag == 1&& i != m - 1)continue;
        else if(flag == 1 && i == m - 1)
            printf("No saddle point!\n");
        else
            {
                printf("a[%d][%d] is %d\n",i,maxpos,a[i][maxpos]);
                break;
            }
    }

}


4. 計算三位階乘和數

題目內容:
試求出所有三位階乘和數:m=a!+b!+c!(其中a爲百位數字,b爲十位數字,c爲個位數字。約定0!=1,並輸出所有的m)

#include<stdio.h>
int fact(int n)
{
    if(n == 0)return 1;
    else
        return fact(n - 1)*n;
}
int main()
{
    int a, b, c , i;
    for(i = 100; i <= 999; i++)
    {
        a = i/100;
        b = (i/10)%10;
        c = i%10;
        if(i == fact(a) + fact(b) + fact(c))
        printf("%d\n",i);
    }
    return 0;
}

第9周練兵區編程題

1. 二分法求根

題目內容:
用二分法求下面方程的根

x^3-x-1=0

用二分法求方程的根的基本原理是:若函數有實根,則函數曲線應當在根x*這一點上與x軸有一個交點,並且由於函數是單調的,在根附近的左右區間內,函數值的符號應當相反。利用這一特點,可以通過不斷將求根區間二分的方法,每次將求根區間縮小爲原來的一半,在新的折半後的區間內繼續搜索方程的根,對根所在區間繼續二分,直到求出方程的根爲止。

#include<stdio.h>
#include<math.h>
float fun(float x)
{
    return x*x*x - x - 1;
}
int main()
{
    float x0,x1,mid;
    scanf("%f,%f",&x0,&x1);
    mid = (x0 + x1)/2;
    if(fun(mid) == 0)
        printf("x=%6.2f\n",mid);
    else
    {
        do
        {
           if(fun(x0)*fun(mid) < 0)
            {
                x1 = mid;
                mid = (x0 + x1)/2;
            }
            else
            {
                x0 = mid;
                mid = (x0 + x1)/2;
            }
        }while(fabs(fun(mid)) > 1e-6);
        printf("x=%6.2f\n",mid);
    }
    return 0;
}

2. 矩陣轉置

題目內容:
某二維數組存放的數據構成一個nn的方陣,其中n<=5。寫程序,從鍵盤輸入n的值(n<=5),該nn矩陣中各元素的值按下面的公式計算:

a[i][j] = i * n + j + 1

其中,a[i][j]表示第i行第j列的元素。要求分別輸出該矩陣和它的轉置矩陣。

注意:定義數字大小N時,請用 #define N 10

#include<stdio.h>
#define N 10
int main()
{
    int a[N][N] = {0};
    int n;
    scanf("%d",&n);
    for(int i = 0; i < n; i++)
    {
        for(int  j = 0; j < n; j++)
        {
            a[i][j] = i * n + j + 1;
        }
    }
    printf("The original matrix is:\n");
    for(int i = 0; i < n; i++)
    {
        for(int  j = 0; j < n; j++)
        {
            printf("%3d",a[i][j]);
        }
        printf("\n");
    }
    printf("The changed matrix is:\n");
    for(int i = 0; i < n; i++)
    {
        for(int  j = 0; j < n; j++)
        {
            printf("%3d",a[j][i]);
        }
        printf("\n");
    }
    return 0;
}

3. 程序改錯

題目內容:
下面程序的功能是從鍵盤任意輸入n個數,然後找出其中的最大數與最小數,並將其位置對換。目前程序中存在錯誤,請修改正確。並按照給出的程序運行結果示例檢查修改後的程序。

#include  <stdio.h>
#define ARR_SIZE 10
void  MaxMinExchang(int *a, int n);
int main()
{
     int a[ARR_SIZE], i, n;
     printf("Input n(n<=10):\n");
     scanf("%d", &n);
     printf("Input %d Numbers:\n", n);
     for (i=0; i<n; i++)
     {
        scanf("%d", &a[i]);
     }
     MaxMinExchang(a, n);
    printf("After MaxMinExchange:\n");
     for (i=0; i<n; i++)
     {
        printf("%d ", a[i]);
     }
    printf("\n");
    return 0;
}
void  MaxMinExchang(int *a, int n)
{
     int  maxValue = a[0], minValue = a[0], maxPos = 0, minPos = 0;
     int  i, temp;
     for (i=0; i<n; i++)
{
    if (a[i] > maxValue)
    {
        maxValue = a[i];
        maxPos = i;
    }
    if (a[i] < minValue)
    {
        minValue = a[i];
        minPos = i;
    }
}
     temp = a[maxPos];
     a[maxPos] = a[minPos];
     a[minPos] = temp;
}

4. 蛇形矩陣

題目內容:
從鍵盤任意輸入一個自然數n(n表示矩陣的大小,假設不超過100),請編程輸出一個n*n的蛇形矩陣。如果輸入的n不是自然數或者輸入了不合法的數字,則輸出"Input error!"。

函數原型: void ZigzagMatrix(int a[][N], int n);

函數功能: 計算n*n的蛇形矩陣

提示: 用兩個雙重循環分別計算n*n矩陣的左上三角和右下三角,設置一個計數器從1開始記錄當前要寫入矩陣的元素值,每次寫完一個計數器加1,在計算左上角和右下角矩陣元素時,分奇數和偶數兩種情況考慮待寫入的元素在矩陣中的行列下標位置。

#include<stdio.h>
#define N 100
int main()
{
    int a[N][N]={0};
    int n,ret,i,j,count = 1;
    printf("Input n:\n");
    ret = scanf("%d",&n);
    if(ret != 1 || n < 0 || n > 100)
        printf("Input error!\n");
    for(i = 0;i < n; i++) //左上三角,包括對角線,用i來控制行和列的和
    {
        if(i%2 == 0)  	  //分奇偶,走向會不同
        {
            for(j = i; j >= 0; j--)
            {
                a[j][i - j] = count;
                count++;
            }
        }
        else
        {
            for(j = 0; j <= i; j++)
            {
                a[j][i - j] = count;
                 count++;
            }
        }
    }
    int end = 1;
    for(;i <= (n - 1)*2; i++) //右下三角,用i控制行和列的和
    {
        if(i%2 == 1)
        {
            for(j = n - 1;j >= end; j--)
            {
                a[i - j][j] = count;
                count++;
            }
        }
        else
        {
            for(j = n - 1;j >= end; j--)
            {
                a[j][i - j] = count;
                count++;
            }
        }
        end++;
    }
    for(int k = 0; k < n; k++)
    {
        for(int m = 0; m < n; m++)
        {
            printf("%4d",a[k][m]);
        }
        printf("\n");
    }
}

5. 親密數_1

題目內容:
相親數,也稱爲親密數,如果整數A的全部因子(包括1,不包括A本身)之和等於B,且整數B的全部因子(包括1,不包括B本身)之和等於A,則將整數A和B稱爲親密數。

從鍵盤任意輸入兩個整數m和n,編程判斷m和n是否是親密數。若是親密數,則輸出“Yes!”,否則輸出“No!”

#include<stdio.h>
int SumFactor(int n);
int main()
{
    int m,n;
    printf("Input m, n:\n");
    scanf("%d,%d",&m,&n);
    if(m == SumFactor(n) && n == SumFactor(m))
        printf("Yes!\n");
    else
        printf("No!\n");
    return 0;
}

int SumFactor(int n)
{
    int sum = 0;
    for(int i = 1;i < n; i++)\
    {
        if(n%i == 0)
            sum += i;
    }
    return sum;
}

6. 親密數_2

題目內容:
從鍵盤任意輸入一個整數n,編程計算並輸出n以內的全部親密數。

#include<stdio.h>
int SumFactor(int n);
int main()
{
    int n,i,j;
    printf("Input n:\n");
    scanf("%d",&n);
    for(i = 200;i < n; i++)
    {
        j = SumFactor(i);
        if(i == SumFactor(j)&& i < j)
            printf("(%d,%d)\n",i,j);
    }
    return 0;
}

int SumFactor(int n)
{
    int sum = 0;
    for(int i = 1;i < n; i++)\
    {
        if(n%i == 0)
            sum += i;
    }
    return sum;
}

7. 完全數

題目內容:
完全數(Perfect Number),又稱完美數或完數,它是指這樣的一些特殊的自然數。它所有的真因子(即除了自身以外的約數)的和,恰好等於它本身,即m的所有小於m的不同因子(包括1)加起來恰好等於m本身。注意:1沒有真因子,所以1不是完全數。計算機已經證實在10300以下,沒有奇數的完全數。例如,因爲6 = 1 + 2 + 3,所以6是一個完全數。

從鍵盤任意輸入一個整數m,編程判斷m是否是完全數。若m是完全數,則輸出“Yes!”,並同時打印出每一個完美數的全部因子,以驗證這個數確實是一個完美數。若m不是完全數,則輸出“No!”

#include<stdio.h>
int IsPerfect(int m);
int main()
{
    int m;
    printf("Input m:\n");
    scanf("%d",&m);
    if(!IsPerfect(m))
        printf("No!\n");
    else
    {
        printf("Yes!\n");
        printf("1");
        for(int j = 2;j < m; j++)
        {
            if(m%j == 0)
                printf(",%d",j);
        }
    }
    return 0;
}

int IsPerfect(int m)
{
    int sum = 0;
    for(int i = 1;i < m; i++)
    {
        if(m%i == 0)
            sum += i;
    }
    if(sum != m)
        return 0;
    else
        return 1;

}

8. 迴文素數

題目內容:
所謂迴文素數是指對一個素數n,從左到右和從右到左讀是相同的,這樣的數就稱爲迴文素數,例如11,101,313等。編程計算並輸出不超過n(100<=n<1000)的迴文素數,並統計這些迴文素數的個數,其中n的值從鍵盤輸入。

#include<stdio.h>
#include<math.h>
#define N 100
int IsBackPrime(int n);
int IsPrime(int n);
int main()
{
    int n,count = 0;
    printf("Input n:\n");
    scanf("%d",&n);
    for(int  i = 10; i <= n; i++)
    {
        if(IsBackPrime(i))
        {
            printf("%4d",i);
            count++;
        }
    }
    printf("\ncount=%d\n",count);
    return 0;
}
int IsPrime(int n)
{
    for(int i = 2;i <= sqrt(n); i++)
    {
        if(n%i == 0)
            return 0;
    }
    return 1;
}
int IsBackPrime(int n)
{
    int a[N] = {0};
    int j = 0;
    int low = 0,high;
    if(IsPrime(n) == 0)
        return 0;
    while(n)
    {
            a[j] = n%10;
            n = n/10;
            j++;
    }
    high = j - 1;
    while(low < high)
    {
        if(a[low] != a[high])
            return 0;
        low++;
        high--;
    }
    return 1;
}

9. 梅森尼數

題目內容:
形如 2^i-1 的素數,稱爲梅森尼數。編程計算並輸出指數i在[2,n]中的所有梅森尼數,並統計這些梅森尼數的個數,其中n的值由鍵盤輸入,並且n的值不能大於50。其中,2^i 表示2的i次方,請不要使用pow(2,i)編程計算,應採用循環累乘求積的方式計算2^i。
提示: 當i 超過30以後,2^i-1的值會很大,不能用long型變量來存儲,必須使用double類型來存儲。對於double類型變量x(不是整型)不能執行求餘運算,即不能用 x % i == 0來判斷x是否能被i整除,可以使用 x / i == (int)(x/i)來判斷x是否能被i整除。

#include<stdio.h>
#include<math.h>
#define N 100
int IsMersenne(int i,double *sum);
int IsPrime(double n);
int main()
{
    int count = 0,n;
    double sum[1] = {0};
    printf("Input n:\n");
    scanf("%d",&n);
    for(int i = 1; i < n; i++)
    {
        if(IsMersenne(i,&sum))
        {
            printf("2^%d-1=%.0lf\n",i,sum[0] - 1);
            count++;
        }
    }
    printf("count=%d\n",count);
    return 0;
}
int IsPrime(double n)
{
    if(n == 1)
        return 0;
    for(int i = 2;i <= sqrt(n); i++)
    {
        if(n/i == (int)(n/i))
            return 0;
    }
    return 1;
}

int IsMersenne(int i,double *sum)
{
    double n;
    sum[0] = 1;
    for(int j = 1; j <= i; j++)
    {
        sum[0] = sum[0] * 2;
    }
    if(IsPrime(sum[0] - 1))
        return 1;
    else
        return 0;
}

10. 工資統計

題目內容:
某公司有職員(最多50人),試編寫程序打印最高工資、最低工資和平均工資。公司人數在主函數給出,職工工資輸入請調用Input函數,計算最高工資、最低工資和平均工資調用Compute函數,打印最高工資、最低工資和平均工資在主函數。請在給定的框架下寫出完整程序。

#include<stdio.h>
void Input(float wage[], int n);
float Compute(float wage[], int n, float *pmaxwage, float *pminwage);
int main()
{
    float wage[50],maxwage,minwage,avewage;
    int n;
    printf("Please input n:\n");
    scanf("%d",&n);
    Input(wage,n);
    avewage=Compute(wage,n,&maxwage,&minwage);
    printf("maxwage=%.2f, minwage=%.2f, avewage=%.2f\n",maxwage,minwage,avewage);
    return 0;
}

void Input(float wage[], int n)
{   int i;
    for(i = 0;i < n; i++)
    {
        scanf("%f",&wage[i]);
    }
}

float Compute(float wage[], int n, float *pmaxwage, float *pminwage)
{
    int  i;
    *pmaxwage = wage[0];
    *pminwage = wage[0];
    float sumwage = 0;
    for(i = 0;i < n; i++)
    {
        if(wage[i] > *pmaxwage)
			*pmaxwage = wage[i];
        if(wage[i] < *pminwage)
            *pminwage = wage[i];
        sumwage += wage[i];
    }
    return sumwage/n;
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章