Unsafe code trong C#

C# cho phép sử dụng các biến con trỏ trong một hàm của khối code khi nó được đánh dấu bởi unsafe modifier. Khái niệm unsafe code hoặc unmanaged code trong C# là một khối code mà sử dụng một biến con trỏ.

Cách compile unsafe code trong Visual Studio 2010

Để biên dịch và chạy các chương trình trong chế độ /unsafe, bạn chỉ cần click chuột phải vào Project, sau đó chọn Properties --> Build --> Allow unsafe code rồi nhấn tổ hợp phím Ctrl + S để lưu các thay đổi như trong hình:

Con trỏ (Pointer) trong C#

Một con trỏ là một biến mà có giá trị là địa chỉ của biến khác, ví dụ: địa chỉ trực tiếp của vị trí bộ nhớ. Tương tự như bất kỳ biến hoặc hằng khác trong C#, bạn phải khai báo một con trỏ trước khi bạn có thể sử dụng nó để lưu giữ bất kỳ địa chỉ biến nào.

Form chung của một khai báo con trỏ trong C# là:

type *var-name;

Dưới đây là các khai báo con trỏ hợp lệ trong C#:

int    *ip;    /* con trỏ tới một số nguyên */
double *dp;    /* con trỏ tới một số double */
float  *fp;    /* con trỏ tới một số float */
char   *ch     /* con trỏ tới một ký tự */

Ví dụ sau minh họa sự sử dụng của con trỏ, sử dụng unsafe modifier trong C#:

using System;


namespace HoclaptrinhCsharp
{
    class TestCsharp
    {
        static unsafe void Main(string[] args)
        {
            Console.WriteLine("Con tro trong C#");
            Console.WriteLine("--------------------------------");
            int var = 20;
            int* p = &var;
            Console.WriteLine("Du lieu: {0} ", var);
            Console.WriteLine("Dia chi: {0}", (int)p);
            Console.ReadKey();
        }
    }
}

Nếu bạn không sử dụng lệnh Console.ReadKey(); thì chương trình sẽ chạy và kết thúc luôn (nhanh quá đến nỗi bạn không kịp nhìn kết quả). Lệnh này cho phép chúng ta nhìn kết quả một cách rõ ràng hơn.

Biên dịch và chạy chương trình C# trên sẽ cho kết quả sau:

Thay cho việc khai báo toàn bộ phương thức ở dạng unsafe, bạn cũng có thể khai báo một phần code dạng unsafe.

Thu hồi giá trị dữ liệu bởi sử dụng con trỏ trong C#

Bạn có thể thu hồi dữ liệu đã được lưu giữ tại vị trí được tham chiếu bởi biến con trỏ, sử dụng phương thức ToString() trong C#. Đây là ví dụ minh họa:

using System;


namespace HoclaptrinhCsharp
{
    class TestCsharp
    {
        static void Main(string[] args)
        {
            Console.WriteLine("Con tro trong C#");
            Console.WriteLine("Unsafe code trong C#");
            Console.WriteLine("--------------------------------");
            unsafe
            {
                int var = 20;
                int* p = &var;
                Console.WriteLine("Du lieu: {0} ", var);
                Console.WriteLine("Du lieu: {0} ", p->ToString());
                Console.WriteLine("Dia chi: {0} ", (int)p);
            }


            Console.ReadKey();
        }
    }
}

Biên dịch và chạy chương trình C# trên sẽ cho kết quả sau:

Truyền Con trỏ như là các Tham số tới phương thức trong C#

Bạn có thể truyền một biến con trỏ tới một phương thức dưới dạng các tham số. Đây là ví dụ minh họa:

using System;


namespace HoclaptrinhCsharp
{
    class TestCsharp
    {
        public unsafe void swap(int* p, int* q)
        {
            int temp = *p;
            *p = *q;
            *q = temp;
        }


        public unsafe static void Main()
        {
            Console.WriteLine("Unsafe code trong C#");
            Console.WriteLine("Truyen con tro nhu la tham so");
            Console.WriteLine("-----------------------------------");
            TestCsharp p = new TestCsharp();
            int var1 = 10;
            int var2 = 20;
            int* x = &var1;
            int* y = &var2;


            Console.WriteLine("Truoc khi trao doi: var1 = {0}, var2 = {1}", var1, var2);
            p.swap(x, y);


            Console.WriteLine("Sau khi trao doi: var1 = {0}, var2 = {1}", var1, var2);
            Console.ReadKey();
        }
    }
}

Biên dịch và chạy chương trình C# trên sẽ cho kết quả sau:

Truy cập các phần tử mảng bởi sử dụng một con trỏ trong C#

Trong C#, một tên mảng và một con trỏ tới cùng kiểu dữ liệu ở dạng dữ liệu mảng, là không cùng kiểu biến. Ví dụ, int *p và int[] p, là không cùng kiểu. Bạn có thể lượng gia biến con trỏ p bởi vì nó không là cố định trong bộ nhớ, nhưng một địa chỉ mảng là cố định trong bộ nhớ, và bạn không thể lượng gia nó.

Vì thế, nếu bạn cần truy cập một dữ liệu mảng bởi sử dụng một biến con trỏ, như khi chúng ta đã làm trong C hoặc C++ bạn cần cố định con trỏ đó bởi sử dụng từ khóa fixed trong C#.

Sau đây là ví dụ minh họa:

using System;


namespace HoclaptrinhCsharp
{
    class TestCsharp
    {
        public unsafe static void Main()
        {
            Console.WriteLine("Unsafe code trong C#");
            Console.WriteLine("Truy cap cac phan tu mang boi su dung con tro");
            Console.WriteLine("-----------------------------------");
            int[]  list = {10, 100, 200};
            fixed (int* ptr = list)


                /* mang dia chi trong con tro */
                for (int i = 0; i < 3; i++)
                {
                    Console.WriteLine("Dia chi cua list[{0}]={1}", i, (int)(ptr + i));
                    Console.WriteLine("Gia tri cua list[{0}]={1}", i, *(ptr + i));
                }


            Console.ReadKey();
        }
    }
}

Biên dịch và chạy chương trình C# trên sẽ cho kết quả sau:

Bình luận