目錄
讓我們來看一個例子
我們該如何解決這個問題?
在C語言中實作介面
一個真實的用例
結論
首頁 Java java教程 C語言中的物件導向?從頭開始實作介面。

C語言中的物件導向?從頭開始實作介面。

Jan 21, 2025 am 10:05 AM

Orientação a Objetos em C? Implementando uma interface do zero.

我一直對電腦充滿好奇,總是會想:「好吧,我知道怎麼用,但它到底是怎麼運作的?」在這個過程中,我常常會做個思想實驗:如果讓我從零開始實現它,我會怎麼做?在本文中,我們將探討介面在物件導向程式設計中的工作原理(使用Java),然後在C語言中實作一個簡陋的介面版本。

讓我們來看一個例子

我們的例子很簡單:計算車輛的價格。如果是汽車,價格將根據其最高速度計算;如果是摩托車,價格將根據其排氣量計算。我們先用介面定義車輛的行為:

public class Main {
    public interface Vehicle {
        Integer price();
    }
}
登入後複製
登入後複製

這裡沒什麼特別的,只是一個傳回整數的方法。現在讓我們實作汽車類別:

public class Main {
    // ...

    public static class Car implements Vehicle {
        private final Integer speed;

        public Car(Integer speed) {
            this.speed = speed;
        }

        @Override
        public Integer price() {
            return speed * 60;
        }
    }
}
登入後複製
登入後複製

很經典:一個建構函式和price方法的實現,將速度乘以60。現在,讓我們實作摩托車類別:

public class Main {
    // ...

    public static class Motorcycle implements Vehicle {
        private final Integer cc;

        public Motorcycle(Integer cc) {
            this.cc = cc;
        }

        @Override
        public Integer price() {
            return cc * 10;
        }
    }
}
登入後複製
登入後複製

幾乎一樣,唯一的差別是現在我們將排氣量乘以10。然後,我們實現一個列印車輛價格的方法:

public class Main {
    // ...

    public static void printVehiclePrice(Vehicle vehicle) {
        System.out.println("$" + vehicle.price() + ".00");
    }
}
登入後複製
登入後複製

沒什麼秘密。最後,我們的main方法:

public class Main {
    // ...

    public static void main(String[] args) {
        Car car = new Car(120);
        Motorcycle motorcycle = new Motorcycle(1000);

        printVehiclePrice(car);
        printVehiclePrice(motorcycle);
    }
}
登入後複製
登入後複製
<code>$ java Main.java
00.00
000.00</code>
登入後複製
登入後複製

這就是我們想要達到的模型,但是現在要在C語言中從零開始實作。

我們該如何解決這個問題?

當我想到物件時,首先想到的是一組表示狀態的資料和操作和管理該狀態的方法。在C語言中表示資料集合最直接的方法是結構體。對於方法,最接近的方法是接收狀態作為參數的函數。此狀態將對應於類別中的this,例如:

typedef struct {
    int height_in_cm;
    int weight_in_kg;
} Person;

float person_bmi(Person *person) {
    float height_in_meters = (float)person->height_in_cm / 100;
    float bmi =
        (float)person->weight_in_kg / (height_in_meters * height_in_meters);

    return bmi;
}
登入後複製
登入後複製

在這裡,我們在Person結構體中定義了一個人的數據,並使用這些數據進行簡單的計算。這是我們在C語言中可以擁有的最接近類別的結構。也許在結構體中使用函數指標也是一個好主意?好吧,這留到下一篇文章再討論。

好的,我們有了一種類似類別的結構。現在,我們如何在C語言中定義介面?如果仔細想想,編譯器/解釋器不會做魔法來猜測哪些類別實作了介面。它可以在編譯時確定這一點,並將我們使用介面的所有部分替換為特定的類型。在編譯後的程式中,介面甚至不存在。

由於C語言編譯器沒有提供這種可能性,我們必須自己實作這個方案。我們需要知道所有實作我們介面的類型,並想辦法使用這些實作的函數。

在C語言中實作介面

首先,讓我們定義我們簡陋介面的框架。我們將創建一個枚舉,其中包含不同的實作和我們函數的簽名。

public class Main {
    public interface Vehicle {
        Integer price();
    }
}
登入後複製
登入後複製

在這裡,我們定義了我們的枚舉,其中包含我們稍後將要實現的實作。這看起來可能不像,但這部分非常重要。接下來,我們聲明了vehicle_free函數(稍後將解釋)和vehicle_price函數,我們希望在我們的「類別」中實作這些函數。現在讓我們來看看汽車的實現:

public class Main {
    // ...

    public static class Car implements Vehicle {
        private final Integer speed;

        public Car(Integer speed) {
            this.speed = speed;
        }

        @Override
        public Integer price() {
            return speed * 60;
        }
    }
}
登入後複製
登入後複製

car_init函數在記憶體中初始化一個新的「物件」Car。在Java中,這將透過new自動完成。在這裡,我們需要手動完成。 vehicle_free函數將用於釋放先前初始化的任何「物件」分配的內存,使用car_free等實作。摩托車的實現非常相似:

public class Main {
    // ...

    public static class Motorcycle implements Vehicle {
        private final Integer cc;

        public Motorcycle(Integer cc) {
            this.cc = cc;
        }

        @Override
        public Integer price() {
            return cc * 10;
        }
    }
}
登入後複製
登入後複製

幾乎一樣,只是現在我們用VEHICLE_MOTORCYCLE初始化,並乘以10。現在讓我們來看看列印車輛價格的函數:

public class Main {
    // ...

    public static void printVehiclePrice(Vehicle vehicle) {
        System.out.println("$" + vehicle.price() + ".00");
    }
}
登入後複製
登入後複製

如此簡單……這樣看來,我們似乎沒有做太多工作。現在,最後也是最重要的一點,我們必須實現我們在上面介面定義中聲明的函數,還記得嗎?幸運的是,我們甚至不需要考慮這個實作。我們總會有一個簡單的窮舉switch/case,僅此而已。

public class Main {
    // ...

    public static void main(String[] args) {
        Car car = new Car(120);
        Motorcycle motorcycle = new Motorcycle(1000);

        printVehiclePrice(car);
        printVehiclePrice(motorcycle);
    }
}
登入後複製
登入後複製

現在我們可以使用我們所做的一切:

<code>$ java Main.java
00.00
000.00</code>
登入後複製
登入後複製
typedef struct {
    int height_in_cm;
    int weight_in_kg;
} Person;

float person_bmi(Person *person) {
    float height_in_meters = (float)person->height_in_cm / 100;
    float bmi =
        (float)person->weight_in_kg / (height_in_meters * height_in_meters);

    return bmi;
}
登入後複製
登入後複製

成功了!但你可能會想:「好吧,這有什麼用?」

一個真實的用例

我最喜歡的專案類型之一是解析器,從解釋器到簡單的數學表達式解析器。通常,當您實作這些解析器時,會遇到稱為AST(抽象語法樹)的東西。顧名思義,它是一棵樹,它將表示您正在處理的語法,例如,變數聲明int foo = 10; 是AST的一個節點,它包含另外三個節點,一個類型節點,用於int,一個標識符節點,用於foo,以及一個表達式節點,用於10,該節點包含另一個值為10的整數節點。看到它有多複雜了嗎?

當我們在C語言中這樣做時,我們必須在包含多個字段的巨大結構體之間進行選擇,以表示任何可能的AST節點,或者使用多個小型結構體實現抽象定義,每個結構體表示不同的節點,就像我們在這裡用我們的「介面」所做的那樣。如果您想查看一個簡單的範例,在這個數學表達式解析器中,我實作了第二種方法。

結論

編譯器或解釋器所做的任何事情都不是魔法。嘗試自己實現一些東西總是一個有趣的練習。希望這是一篇有益的閱讀。謝謝!

以上是C語言中的物件導向?從頭開始實作介面。的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1654
14
CakePHP 教程
1413
52
Laravel 教程
1306
25
PHP教程
1252
29
C# 教程
1225
24
公司安全軟件導致應用無法運行?如何排查和解決? 公司安全軟件導致應用無法運行?如何排查和解決? Apr 19, 2025 pm 04:51 PM

公司安全軟件導致部分應用無法正常運行的排查與解決方法許多公司為了保障內部網絡安全,會部署安全軟件。 ...

如何將姓名轉換為數字以實現排序並保持群組中的一致性? 如何將姓名轉換為數字以實現排序並保持群組中的一致性? Apr 19, 2025 pm 11:30 PM

將姓名轉換為數字以實現排序的解決方案在許多應用場景中,用戶可能需要在群組中進行排序,尤其是在一個用...

如何優雅地獲取實體類變量名構建數據庫查詢條件? 如何優雅地獲取實體類變量名構建數據庫查詢條件? Apr 19, 2025 pm 11:42 PM

在使用MyBatis-Plus或其他ORM框架進行數據庫操作時,經常需要根據實體類的屬性名構造查詢條件。如果每次都手動...

如何使用MapStruct簡化系統對接中的字段映射問題? 如何使用MapStruct簡化系統對接中的字段映射問題? Apr 19, 2025 pm 06:21 PM

系統對接中的字段映射處理在進行系統對接時,常常會遇到一個棘手的問題:如何將A系統的接口字段有效地映�...

Java對像如何安全地轉換為數組? Java對像如何安全地轉換為數組? Apr 19, 2025 pm 11:33 PM

Java對象與數組的轉換:深入探討強制類型轉換的風險與正確方法很多Java初學者會遇到將一個對象轉換成數組的�...

IntelliJ IDEA是如何在不輸出日誌的情況下識別Spring Boot項目的端口號的? IntelliJ IDEA是如何在不輸出日誌的情況下識別Spring Boot項目的端口號的? Apr 19, 2025 pm 11:45 PM

在使用IntelliJIDEAUltimate版本啟動Spring...

電商平台SKU和SPU數據庫設計:如何兼顧用戶自定義屬性和無屬性商品? 電商平台SKU和SPU數據庫設計:如何兼顧用戶自定義屬性和無屬性商品? Apr 19, 2025 pm 11:27 PM

電商平台SKU和SPU表設計詳解本文將探討電商平台中SKU和SPU的數據庫設計問題,特別是如何處理用戶自定義銷售屬...

如何利用Redis緩存方案高效實現產品排行榜列表的需求? 如何利用Redis緩存方案高效實現產品排行榜列表的需求? Apr 19, 2025 pm 11:36 PM

Redis緩存方案如何實現產品排行榜列表的需求?在開發過程中,我們常常需要處理排行榜的需求,例如展示一個�...

See all articles