18
13

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

続 C++ はどれくらい速いのか?:バブルソートの演算で比較してみる

Last updated at Posted at 2023-08-22

続 C++ はどれくらい速いのか?:バブルソートの演算で比較してみる

こんにちは、@studio_meowtoon です。今回は、WSL の Ubuntu 環境で C++ プログラムを実行する方法などを紹介します。
cpp_on_ubuntu.png

目的

Windows 11 の Linux でクラウド開発します。

こちらから記事の一覧がご覧いただけます。

実現すること

Ubuntu 22.04 上で、C++ を含む複数のプログラミング言語を使用して、バブルソートの演算の実行速度を比較します。

バブルソートの計算時間だけでは、言語の性能を評価するのは難しいことをあらかじめご承知ください。また各言語での実装内容は、シンプルさを優先していることと、コンパイラや実行環境で異なる結果になることを予めご理解ください。

比較する言語

No 言語/環境 実行形式 コンパイラ/実行環境
1 C++ ネイティブ g++ 11.3.0
2 C ネイティブ gcc 11.4.0
3 Java JITコンパイル openjdk version "17.0.7"
4 Java ネイティブ GraalVM 22.3.1 Java 17 CE
5 C#.NET JITコンパイル dotnet 7.0.202
6 Python インタプリタ python 3.10.6
7 Node.js インタプリタ node v19.8.1
8 Go ネイティブ go1.18.1 linux/amd64
9 Rust ネイティブ rustc 1.71.1
おまけ シェルスクリプト インタプリタ GNU bash version 5.1.16(1)-release

バブルソートの仕様

以下のようなバブルソートを実行する関数を各言語で実装します。

C++ 版の関数
// バブルソートアルゴリズムを実行する関数
void bubblesort(int array[], int size) {
    for (int i = 0; i < size - 1; ++i) {
        for (int j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                std::swap(array[j], array[j + 1]);
            }
        }
    }
}

この記事では、バブルソートアルゴリズムを実行する関数の性能には触れません。ご了承ください。

演算時間を取得する手順

プロジェクトの作成

プロジェクトフォルダを作成します。
※ ~/tmp/bub-comp をプロジェクトフォルダとします。

$ mkdir -p ~/tmp/bub-comp
$ cd ~/tmp/bub-comp

C++ の処理時間を測定

環境構築手順を開きます。

g++ をインストールします。

$ sudo apt update
$ sudo apt install g++

確認します。

$  g++ --version
g++ (Ubuntu 11.3.0-1ubuntu1~22.04.1) 11.3.0

cpp ファイルを作成します。

$ vim bubblesort.cpp

ファイルの内容

bubblesort.cpp ※抜粋
// バブルソートアルゴリズムを実行する関数
void bubblesort(int array[], int size) {
    for (int i = 0; i < size - 1; ++i) {
        for (int j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                swap(array[j], array[j + 1]);
            }
        }
    }
}
コード全体を表示します。
bubblesort.cpp
#include <chrono>
#include <cstdlib>
#include <ctime>
#include <filesystem>
#include <fstream>
#include <iostream>

using namespace std;
using namespace std::chrono;
using namespace std::filesystem;

const long ARRAY_SIZE = 30000; // ソートする配列のサイズ
const string FILE_PATH = "result/bubblesort_cpp.csv"; // ソート結果を出力するパス

// バブルソートアルゴリズムを実行する関数
void bubblesort(int array[], int size) {
    for (int i = 0; i < size - 1; ++i) {
        for (int j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                swap(array[j], array[j + 1]);
            }
        }
    }
}

// 配列のデータをCSVファイルに出力する関数
void save_array_to_csv(const string& filepath, int array[], int array_size) {
    create_directories(path(filepath).parent_path());
    ofstream outfile(filepath);
    if (outfile.is_open()) {
        outfile << "index,value" << endl;
        for (int i = 0; i < array_size; ++i) {
            outfile << i << "," << array[i] << endl;
        }
        outfile.close();
        cout << "A CSV file was saved successfully." << endl;
    } else {
        cout << "Error: opening CSV file." << endl;
    }
}

int main() {
    // 乱数で配列を初期化
    srand(static_cast<unsigned int>(time(nullptr)));
    int array[ARRAY_SIZE];
    for (int i = 0; i < ARRAY_SIZE; ++i) {
        array[i] = rand() % RAND_MAX;
    }

    // 関数の実行時間を測定
    auto start_time = high_resolution_clock::now();
    bubblesort(array, ARRAY_SIZE);
    auto end_time = high_resolution_clock::now();

    // 実行時間を秒に変換して表示
    auto duration_microseconds = duration_cast<microseconds>(end_time - start_time).count();
    double duration_seconds = static_cast<double>(duration_microseconds) / 1e6;
    cout << "Sorted " << ARRAY_SIZE << " elements in " << fixed << setprecision(5) << setfill('0') << duration_seconds << " seconds." << endl;
    
    // ソート結果をCSVファイルに保存
    save_array_to_csv(FILE_PATH, array, ARRAY_SIZE);

    return 0;
}
ビルドして実行する手順を開きます。

ビルドします。

$ g++ -o bubblesort_cpp bubblesort.cpp -O2

実行します。

$ ./bubblesort_cpp

出力結果

Sorted 30000 elements in 0.96081 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 0.96264
2 0.95842 MIN
3 0.98127 MAX
4 0.95899

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

C の処理時間を測定

環境構築手順を開きます。

gcc をインストールします。

$ sudo apt update
$ sudo apt install gcc

確認します。

$ gcc --version
gcc (Ubuntu 11.4.0-1ubuntu1~22.04) 11.4.0

c ファイルを作成します。

$ vim bubblesort.c

ファイルの内容

bubblesort.c ※抜粋
// バブルソートアルゴリズムを実行する関数
void bubblesort(int array[], int size) {
    for (int i = 0; i < size - 1; ++i) {
        for (int j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                int temp = array[j];
                array[j] = array[j + 1];
                array[j + 1] = temp;
            }
        }
    }
}
コード全体を表示します。
bubblesort.c
#include <stdio.h>
#include <stdlib.h>
#include <time.h>

#define ARRAY_SIZE 30000 // ソートする配列のサイズ
#define FILE_PATH "result/bubblesort_c.csv" // ソート結果を出力するパス

// バブルソートアルゴリズムを実行する関数
void bubblesort(int array[], int size) {
    for (int i = 0; i < size - 1; ++i) {
        for (int j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                int temp = array[j];
                array[j] = array[j + 1];
                array[j + 1] = temp;
            }
        }
    }
}

// 配列のデータをCSVファイルに出力する関数
void save_array_to_csv(const char* filepath, int array[], int array_size) {
    FILE* outfile = fopen(filepath, "w");
    if (outfile != NULL) {
        fprintf(outfile, "index,value\n");
        for (int i = 0; i < array_size; ++i) {
            fprintf(outfile, "%d,%d\n", i, array[i]);
        }
        fclose(outfile);
        printf("A CSV file was saved successfully.\n");
    } else {
        printf("Error: opening CSV file.\n");
    }
}

int main() {
    // 乱数で配列を初期化
    srand((unsigned int)time(NULL));
    int array[ARRAY_SIZE];
    for (int i = 0; i < ARRAY_SIZE; ++i) {
        array[i] = rand() % RAND_MAX;
    }

    // 関数の実行時間を測定
    clock_t start_time = clock();
    bubblesort(array, ARRAY_SIZE);
    clock_t end_time = clock();

    // 実行時間を秒に変換して表示
    double duration_seconds = (double)(end_time - start_time) / CLOCKS_PER_SEC;
    printf("Sorted %d elements in %.5lf seconds.\n", ARRAY_SIZE, duration_seconds);

    // ソート結果をCSVファイルに保存
    save_array_to_csv(FILE_PATH, array, ARRAY_SIZE);

    return 0;
}
ビルドして実行する手順を開きます。

ビルドします。

$ gcc -o bubblesort_c bubblesort.c -O2

実行します。

$ ./bubblesort_c

出力結果

Sorted 30000 elements in 0.94438 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 0.94655
2 0.94221
3 0.94856 MAX
4 0.93655 MIN

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。

Java の処理時間を測定

環境構築手順を開きます。

jdk をインストールします。

$ sudo apt update
$ sudo apt install openjdk-17-jdk

確認します。

$ java -version
openjdk version "17.0.7" 2023-04-18
OpenJDK Runtime Environment (build 17.0.7+7-Ubuntu-0ubuntu122.04.2)
OpenJDK 64-Bit Server VM (build 17.0.7+7-Ubuntu-0ubuntu122.04.2, mixed mode, sharing)

java ファイルを作成します。

$ vim Bubblesort.java

ファイルの内容

Bubblesort.java ※抜粋
// バブルソートアルゴリズムを実行する関数
public static void bubblesort(int[] array) {
    int size = array.length;
    for (int i = 0; i < size - 1; ++i) {
        for (int j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                int temp = array[j];
                array[j] = array[j + 1];
                array[j + 1] = temp;
            }
        }
    }
}
コード全体を表示します。
Bubblesort.java
import java.io.FileWriter;
import java.io.IOException;
import java.util.Random;
import static java.lang.System.*;

public class Bubblesort {
    static final int ARRAY_SIZE = 30000; // ソートする配列のサイズ
    static final String FILE_PATH = "result/bubblesort_java.csv"; // ソート結果を出力するパス

    // バブルソートアルゴリズムを実行する関数
    public static void bubblesort(int[] array) {
        int size = array.length;
        for (int i = 0; i < size - 1; ++i) {
            for (int j = 0; j < size - i - 1; ++j) {
                if (array[j] > array[j + 1]) {
                    int temp = array[j];
                    array[j] = array[j + 1];
                    array[j + 1] = temp;
                }
            }
        }
    }

    // 配列のデータをCSVファイルに出力する関数
    public static void saveArrayToCSV(String filepath, int[] array) {
        try {
            FileWriter writer = new FileWriter(filepath);
            writer.write("index,value\n");
            for (int i = 0; i < array.length; ++i) {
                writer.write(i + "," + array[i] + "\n");
            }
            writer.close();
            out.println("A CSV file was saved successfully.");
        } catch (IOException e) {
            out.println("Error: opening CSV file.");
        }
    }

    public static void main(String[] args) {
        // 乱数で配列を初期化
        int[] array = new int[ARRAY_SIZE];
        Random rand = new Random();
        for (int i = 0; i < ARRAY_SIZE; ++i) {
            array[i] = rand.nextInt(Integer.MAX_VALUE);
        }

        // 関数の実行時間を測定
        long start_time = nanoTime();
        bubblesort(array);
        long end_time = nanoTime();

        // 実行時間を秒に変換して表示
        double duration_seconds = (double) (end_time - start_time) / 1e9;
        out.printf("Sorted %d elements in %.5f seconds.%n", ARRAY_SIZE, duration_seconds);

        // ソート結果をCSVファイルに保存
        saveArrayToCSV(FILE_PATH, array);
    }
}
ビルドして実行する手順を開きます。

ビルドします。

$ javac Bubblesort.java

実行します。

$ java Bubblesort

出力結果

Sorted 30000 elements in 1.09717 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 1.09765
2 1.10883 MAX
3 1.09669
4 1.08630 MIN

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

Java (GraalVM ネイティブイメージ) の処理時間を測定

環境構築については以下の記事でご確認いただけます。

ビルドして実行する手順を開きます。

ネイティブイメージビルドします。

$ native-image -o Bubblesort_java Bubblesort -O2

実行します。

$ ./Bubblesort_java

出力結果

Sorted 30000 elements in 0.95360 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 0.95887
2 0.94834
3 1.03931 MAX
4 0.94098 MIN

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

C#.NET の処理時間を測定

環境構築手順を開きます。

dotnet-sdk をインストールします。

$ sudo apt update
$ sudo apt install dotnet-sdk-7.0

確認します。

$ dotnet --list-sdks
7.0.202 [/usr/share/dotnet/sdk]
$ dotnet --version
7.0.202

cs ファイルを作成します。

$ vim Bubblesort.cs

ファイルの内容

Bubblesort.cs ※抜粋
// バブルソートアルゴリズムを実行する関数
static void bubblesort(int[] array) {
    int size = array.Length;
    for (int i = 0; i < size - 1; ++i) {
        for (int j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                int temp = array[j];
                array[j] = array[j + 1];
                array[j + 1] = temp;
            }
        }
    }
}
コード全体を表示します。
Bubblesort.cs
using System;
using System.IO;
using static System.Console;
using static System.DateTime;

public class Program {
    const int ARRAY_SIZE = 30000; // ソートする配列のサイズ
    const string FILE_PATH = "result/bubblesort_cs.csv"; // ソート結果を出力するパス

    // バブルソートアルゴリズムを実行する関数
    static void bubblesort(int[] array) {
        int size = array.Length;
        for (int i = 0; i < size - 1; ++i) {
            for (int j = 0; j < size - i - 1; ++j) {
                if (array[j] > array[j + 1]) {
                    int temp = array[j];
                    array[j] = array[j + 1];
                    array[j + 1] = temp;
                }
            }
        }
    }

    // 配列のデータをCSVファイルに出力する関数
    static void saveArrayToCSV(string filepath, int[] array) {
        try {
            using (StreamWriter writer = new(filepath)) {
                writer.WriteLine("index,value");
                for (int i = 0; i < array.Length; ++i) {
                    writer.WriteLine($"{i},{array[i]}");
                }
            }
            WriteLine("A CSV file was saved successfully.");
        } catch (IOException) {
            WriteLine("Error: opening CSV file.");
        }
    }

    static void Main(string[] args) {
        // 乱数で配列を初期化
        int[] array = new int[ARRAY_SIZE];
        Random rand = new();
        for (int i = 0; i < ARRAY_SIZE; ++i) {
            array[i] = rand.Next(int.MaxValue);
        }

        // 関数の実行時間を測定
        long start_time = Now.Ticks;
        bubblesort(array);
        long end_time = Now.Ticks;

        // 実行時間を秒に変換して表示
        double duration_seconds = (end_time - start_time) / 1e7;
        WriteLine($"Sorted {ARRAY_SIZE} elements in {duration_seconds:f5} seconds.");

        // ソート結果をCSVファイルに保存
        saveArrayToCSV(FILE_PATH, array);
    }
}
ビルドして実行する手順を開きます。

Bubblesort.csproj ファイルを作成します。

$ vim Bubblesort.csproj

ファイルの内容

Bubblesort.csproj
<Project Sdk="Microsoft.NET.Sdk">
  <PropertyGroup>
    <OutputType>Exe</OutputType>
    <TargetFramework>net7.0</TargetFramework>
  </PropertyGroup>
</Project>

ビルドします。

$ dotnet build Bubblesort.csproj -c Release -o out

実行します。

$ dotnet out/Bubblesort.dll

出力結果

Sorted 30000 elements in 1.15536 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 1.16973 MAX
2 1.15081
3 1.15992
4 1.14980 MIN

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

Python の処理時間を測定

環境構築手順を開きます。

通常 Ubuntu 22.04 には、Python はプリインストールされています。

確認します。

$ python3 --version
Python 3.10.6

py ファイルを作成します。

$ vim bubblesort.py

ファイルの内容 ※抜粋

bubblesort.py
# バブルソートアルゴリズムを実行する関数
def bubblesort(array):
    size = len(array)
    for i in range(size - 1):
        for j in range(size - i - 1):
            if array[j] > array[j + 1]:
                array[j], array[j + 1] = array[j + 1], array[j]
コード全体を表示します。
bubblesort.py
import csv
import random
import time

ARRAY_SIZE = 30000  # ソートする配列のサイズ
FILE_PATH = "result/bubblesort_py.csv"  # ソート結果を出力するパス

# バブルソートアルゴリズムを実行する関数
def bubblesort(array):
    size = len(array)
    for i in range(size - 1):
        for j in range(size - i - 1):
            if array[j] > array[j + 1]:
                array[j], array[j + 1] = array[j + 1], array[j]

# 配列のデータをCSVファイルに出力する関数
def save_array_to_csv(filepath, array):
    try:
        with open(filepath, mode='w', newline='') as csvfile:
            writer = csv.writer(csvfile)
            writer.writerow(['index', 'value'])
            for i in range(len(array)):
                writer.writerow([i, array[i]])
        print("A CSV file was saved successfully.")
    except IOError:
        print("Error: opening CSV file.")

def main():
    # 乱数で配列を初期化
    array = []
    for _ in range(ARRAY_SIZE):
        random_value = random.randint(0, 0x7FFFFFFF)
        array.append(random_value)

    # 関数の実行時間を測定
    start_time = time.time()
    bubblesort(array)
    end_time = time.time()

    # 実行時間を秒に変換して表示
    duration_seconds = end_time - start_time
    print(f"Sorted {ARRAY_SIZE} elements in {duration_seconds:.5f} seconds.")

    # ソート結果をCSVファイルに保存
    save_array_to_csv(FILE_PATH, array)

if __name__ == "__main__":
    main()
実行する手順を開きます。

実行します。

$ python3 bubblesort.py

出力結果

Sorted 30000 elements in 43.00358 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 42.96953 MIN
2 42.97733
3 43.02983
4 43.16352 MAX

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

Node.js の処理時間を測定

環境構築手順を開きます。

Node.js の公式パッケージリポジトリを追加します。

$ curl -fsSL https://deb.nodesource.com/setup_19.x | sudo -E bash -

Node.js をインストールします。

$ sudo apt update
$ sudo apt install nodejs

確認します。

$ node -v
v19.8.1
$ npm -v
9.5.1

js ファイルを作成します。

$ vim bubblesort.js

ファイルの内容

bubblesort.js ※抜粋
// バブルソートアルゴリズムを実行する関数
function bubblesort(array) {
    const size = array.length;
    for (let i = 0; i < size - 1; ++i) {
        for (let j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                const temp = array[j];
                array[j] = array[j + 1];
                array[j + 1] = temp;
            }
        }
    }
}
コード全体を表示します。
bubblesort.js
const fs = require('fs');

const ARRAY_SIZE = 30000; // ソートする配列のサイズ
const FILE_PATH = 'result/bubblesort_js.csv'; // ソート結果を出力するパス

// バブルソートアルゴリズムを実行する関数
function bubblesort(array) {
    const size = array.length;
    for (let i = 0; i < size - 1; ++i) {
        for (let j = 0; j < size - i - 1; ++j) {
            if (array[j] > array[j + 1]) {
                const temp = array[j];
                array[j] = array[j + 1];
                array[j + 1] = temp;
            }
        }
    }
}

// 配列のデータをCSVファイルに出力する関数
function save_array_to_csv(filepath, array) {
    try {
        const writer = fs.createWriteStream(filepath);
        writer.write('index,value\n');
        for (let i = 0; i < array.length; ++i) {
            writer.write(`${i},${array[i]}\n`);
        }
        writer.end();
        console.log('A CSV file was saved successfully.');
    } catch (error) {
        console.error('Error: opening CSV file.');
    }
}

function main() {
    // 乱数で配列を初期化
    const array = new Array(ARRAY_SIZE);
    for (let i = 0; i < ARRAY_SIZE; ++i) {
        array[i] = Math.floor(Math.random() * 0x7FFFFFFF);
    }

    // 関数の実行時間を測定
    const start_time = process.hrtime.bigint();
    bubblesort(array);
    const end_time = process.hrtime.bigint();

    // 実行時間を秒に変換して表示
    const duration_seconds = Number(end_time - start_time) / 1e9;
    console.log(`Sorted ${ARRAY_SIZE} elements in ${duration_seconds.toFixed(5)} seconds.`);

    // ソート結果をCSVファイルに保存
    save_array_to_csv(FILE_PATH, array);
}

main();
実行する手順を開きます。

実行します。

$ node bubblesort.js

出力結果

Sorted 30000 elements in 1.39350 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 1.40223 MAX
2 1.38343 MIN
3 1.39101
4 1.39599

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

Go の処理時間を測定

環境構築手順を開きます。

Go をインストールします。

$ sudo apt update
$ sudo apt install golang

確認します。

$ go version
go version go1.18.1 linux/amd64

go ファイルを作成します。

$ bubblesort.go

ファイルの内容

bubblesort.go ※抜粋
// バブルソートアルゴリズムを実行する関数
func bubblesort(array []int) {
	size := len(array)
	for i := 0; i < size-1; i++ {
		for j := 0; j < size-i-1; j++ {
			if array[j] > array[j+1] {
				array[j], array[j+1] = array[j+1], array[j]
			}
		}
	}
}
コード全体を表示します。
bubblesort.go
package main

import (
	"fmt"
	"math"
	"math/rand"
	"os"
	"time"
)

const ARRAY_SIZE = 30000                     // ソートする配列のサイズ
const FILE_PATH = "result/bubblesort_go.csv" // ソート結果を出力するパス

// バブルソートアルゴリズムを実行する関数
func bubblesort(array []int) {
	size := len(array)
	for i := 0; i < size-1; i++ {
		for j := 0; j < size-i-1; j++ {
			if array[j] > array[j+1] {
				array[j], array[j+1] = array[j+1], array[j]
			}
		}
	}
}

// 配列のデータをCSVファイルに出力する関数
func save_array_to_csv(filepath string, array []int) {
	file, err := os.Create(filepath)
	if err != nil {
		fmt.Println("Error: opening CSV file.")
		return
	}
	defer file.Close()
	file.WriteString("index,value\n")
	for i, value := range array {
		file.WriteString(fmt.Sprintf("%d,%d\n", i, value))
	}
	fmt.Println("A CSV file was saved successfully.")
}

func main() {
	// 乱数で配列を初期化
	rand.Seed(time.Now().UnixNano())
	array := make([]int, ARRAY_SIZE)
	for i := 0; i < ARRAY_SIZE; i++ {
		array[i] = rand.Intn(math.MaxInt32)
	}

	// 関数の実行時間を測定
	start_time := time.Now()
	bubblesort(array)
	end_time := time.Now()

	// 実行時間を秒に変換して表示
	duration_seconds := end_time.Sub(start_time).Seconds()
	fmt.Printf("Sorted %d elements in %.5f seconds.\n", ARRAY_SIZE, duration_seconds)

	// ソート結果をCSVファイルに保存
	save_array_to_csv(FILE_PATH, array)
}
ビルドして実行する手順を開きます。

ビルドします。

$ go build -o bubblesort_go bubblesort.go

実行します。

$ ./bubblesort_go

出力結果

Sorted 30000 elements in 0.93943 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 0.93528 MIN
2 0.93876
3 0.94010
4 0.94512 MAX

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

Rust の処理時間を測定

環境構築手順を開きます。

Rust をインストールします。

$ curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
1) Proceed with installation (default)
2) Customize installation
3) Cancel installation
> 1

Rust の環境設定を .bashrc ファイルに追記します。

$ vim ~/.bashrc

一番下へ次の内容を追加します。

# Rust Environment
source "$HOME/.cargo/env"

設定を反映します。

$ source ~/.bashrc

確認します。

$ rustc --version
rustc 1.71.1 (eb26296b5 2023-08-03)

rs ファイルを作成します。

$ vim bubblesort.rs

ファイルの内容

bubblesort.rs ※抜粋
// バブルソートアルゴリズムを実行する関数
fn bubblesort(array: &mut [i32]) {
    let size = array.len();
    for i in 0..size - 1 {
        for j in 0..size - i - 1 {
            if array[j] > array[j + 1] {
                array.swap(j, j + 1);
            }
        }
    }
}
コード全体を表示します。
bubblesort.rs
use std::fs::File;
use std::io::Write;
use std::time::Instant;
use rand::Rng;

const ARRAY_SIZE: usize = 30000; // ソートする配列のサイズ
const FILE_PATH: &str = "result/bubblesort_rs.csv"; // ソート結果を出力するパス

// バブルソートアルゴリズムを実行する関数
fn bubblesort(array: &mut [i32]) {
    let size = array.len();
    for i in 0..size - 1 {
        for j in 0..size - i - 1 {
            if array[j] > array[j + 1] {
                array.swap(j, j + 1);
            }
        }
    }
}

// 配列のデータをCSVファイルに出力する関数
fn save_array_to_csv(filepath: &str, array: &[i32]) {
    let mut file = File::create(filepath).expect("Error: creating CSV file.");
    file.write_all(b"index,value\n").expect("Error: writing CSV header.");
    for (i, &value) in array.iter().enumerate() {
        writeln!(file, "{},{}", i, value).expect("Error: writing CSV data.");
    }
    println!("A CSV file was saved successfully.");
}

fn main() {
    // 乱数で配列を初期化
    let mut array = Vec::with_capacity(ARRAY_SIZE);
    let mut rng = rand::thread_rng();
    for _ in 0..ARRAY_SIZE {
        array.push(rng.gen_range(0..i32::MAX));
    }

    // 関数の実行時間を測定
    let start_time = Instant::now();
    bubblesort(&mut array);
    let end_time = Instant::now();

    // 実行時間を秒に変換して表示
    let duration_seconds = end_time.duration_since(start_time).as_secs_f64();
    println!("Sorted {} elements in {:.5} seconds.", ARRAY_SIZE, duration_seconds);

    // ソート結果をCSVファイルに保存
    save_array_to_csv(FILE_PATH, &array);
}
ビルドして実行する手順を開きます。

Cargo.toml ファイルを作成します。

$ vim Cargo.toml

ファイルの内容

Cargo.toml
[package]
name = "bubblesort_rs"
version = "0.1.0"
edition = "2021"

[dependencies]
rand = "0.8"

[[bin]]
name = "bubblesort"
path = "bubblesort.rs"

ビルドします。

$ cargo build --release --bin bubblesort

実行します。

$ ./target/release/bubblesort_rs

出力結果

Sorted 30000 elements in 1.00214 seconds.
A CSV file was saved successfully.
実行結果の詳細をみます。
回数 時間(秒) 備考
1 1.00304
2 1.00125
3 1.00098 MIN
4 1.01357 MAX

4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

シェルスクリプト の処理時間を測定

sh ファイルを作成します。

$ vim bubblesort.sh

ファイルの内容

bubblesort.sh ※抜粋
# バブルソートアルゴリズムを実行する関数
bubblesort() {
    local array=("$@")
    local size=${#array[@]}
    for ((i = 0; i < size - 1; i++)); do
        for ((j = 0; j < size - i - 1; j++)); do
            if ((array[j] > array[j + 1])); then
                temp=${array[j]}
                array[j]=${array[j + 1]}
                array[j + 1]=$temp
            fi
        done
    done
    echo "${array[@]}"
}
コード全体を表示します。
bubblesort.sh
#!/bin/bash

ARRAY_SIZE=30000 # ソートする配列のサイズ
FILE_PATH="result/bubblesort_sh.csv" # ソート結果を出力するパス

# バブルソートアルゴリズムを実行する関数
bubblesort() {
    local array=("$@")
    local size=${#array[@]}
    for ((i = 0; i < size - 1; i++)); do
        for ((j = 0; j < size - i - 1; j++)); do
            if ((array[j] > array[j + 1])); then
                temp=${array[j]}
                array[j]=${array[j + 1]}
                array[j + 1]=$temp
            fi
        done
    done
    echo "${array[@]}"
}

# 配列のデータをCSVファイルに出力する関数
save_array_to_csv() {
    local filepath=$1
    shift
    local array=("$@")
    echo "index,value" > "$filepath"
    for ((i = 0; i < ${#array[@]}; i++)); do
        echo "$i,${array[i]}" >> "$filepath"
    done
    echo "A CSV file was saved successfully."
}

# 乱数で配列を初期化
array=()
for ((i = 0; i < ARRAY_SIZE; i++)); do
    array[i]=$((RANDOM % 0x7FFFFFFF))
done

# 関数の実行時間を測定
start_time=$(date +%s.%N)
sorted_array=($(bubblesort "${array[@]}"))
end_time=$(date +%s.%N)

# 実行時間を計算して表示
duration_seconds=$(echo "$end_time - $start_time" | bc)
duration_milliseconds=$(printf "%.5f" "$duration_seconds")
echo "Sorted $ARRAY_SIZE elements in $duration_milliseconds seconds."

# ソート結果をCSVファイルに保存
save_array_to_csv "$FILE_PATH" "${sorted_array[@]}"
実行する手順を開きます。

実行します。

$ chmod +x bubblesort.sh
$ ./bubblesort.sh

出力結果

※ 計測断念: 遅すぎる😭

シェルスクリプトでは、一般的なプログラム言語と同じような書き方をすると効率的でない場合があります。

計測した結果を比較

実行環境

項目 内容
プロセッサ Intel(R) Core(TM) i7-10750H CPU @ 2.60GHz 2.59 GHz
実装 RAM 16.0 GB
システムの種類 64 ビット オペレーティング システム、x64 ベース プロセッサ
OS Ubuntu 22.04.1 LTS (Windows 11 WSL)

計測結果

順位 言語/コンパイラ/環境 実行形式 処理時間(秒) Java JITコンパイルを 1.0 とした割合
1 Go (go1.18.1 linux/amd64) ネイティブ 0.93943 0.856
2 C (gcc 11.4.0) ネイティブ 0.94438 0.861
3 Java (GraalVM 22.3.1 Java 17 CE) ネイティブ 0.95360 0.869
4 C++ (g++ 11.3.0) ネイティブ 0.96081 0.876
5 Rust (rustc 1.71.1) ネイティブ 1.00214 0.913
6 Java (openjdk version "17.0.7") JITコンパイル 1.09717 1.000
7 C#.NET (dotnet 7.0.202) JITコンパイル 1.15536 1.054
8 Node.js (node v19.8.1) インタプリタ 1.39350 1.270
9 Python (python 3.10.6) インタプリタ 43.00358 39.175

各言語で4回の実行結果から最大値と最小値を取り除いて、残りの2回の結果の平均値を求めています。※小数点5桁以下は切り捨て

結果からわかったこと

キーワード 考察
ネイティブコンパイルの優位性 C や C++、Go、Rust などのネイティブコンパイル言語が、JIT コンパイルやインタプリタを使用する言語に比べて処理速度が速いことが示されています。
インタプリタの遅さ インタプリタを使用する Python や Node.js では処理時間が比較的長くなっており、実行速度が他の言語より遅いことが示されています。

Java では、GraalVM ネイティブイメージビルドにより高速化を期待しました。結果、バブルソートの演算に関しては今回の計測結果においては、ネイティブイメージ実行の方が JIT コンパイル最適化よりも若干高速でした。

まとめ

  • バブルソートの演算に関しては、ネイティブコードを生成する言語がほぼ同じようなパフォーマンスで動作しました。
  • Java、C# の JITコンパイル方式の言語においても、バブルソートの演算に関してはかなり優秀な性能を示しています。
  • また、各言語でのファイル出力の方法について学ぶことができました。

本記事は、Ubuntu 22.04 上でのプログラム実行速度比較に焦点を当て、バブルソート演算を通じて異なるプログラミング言語のパフォーマンスを調査しました。クラウド環境におけるアプリケーション開発者にとって、言語選定の際の参考となれば幸いです。

どうでしたか? Windows 11 の Linux で、クラウド向けの開発環境を手軽に構築することができます。ぜひお試しください。今後もさまざまな言語や開発環境の情報などを紹介していきますので、ぜひお楽しみにしてください。

関連記事

18
13
4

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
18
13

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?