5

我在 C# 中创建了一个窗口应用程序。现在我想为这个应用程序设置 CPU 亲和性。我可能有 2 个处理器、4 个处理器、8 个处理器或可能超过 8 个处理器。

我想使用来自接口的输入来设置 cpu 亲和性。

我怎样才能做到这一点?如何使用 Environment.ProcessorCount 设置亲和力?

4

4 回答 4

13

试试这个:

Process.GetCurrentProcess().ProcessorAffinity = (System.IntPtr)2;

这里有更多关于它的信息。

ProcessorAffinity将每个处理器表示为一个位。位 0 代表处理器 1,位 1 代表处理器 2,依此类推。下表显示了四处理器系统可能的ProcessorAffinity的一个子集。

Property value (in hexadecimal)  Valid processors

0x0001                           1
0x0002                           2
0x0003                           1 or 2
0x0004                           3
0x0005                           1 or 3
0x0007                           1, 2, or 3
0x000F                           1, 2, 3, or 4

这是一个小示例程序:

//TODO: manage exceptions
class Program
{
    static void Main(string[] args)
    {
        Console.WriteLine("Total # of processors: {0}", Environment.ProcessorCount);
        Console.WriteLine("Current processor affinity: {0}", Process.GetCurrentProcess().ProcessorAffinity);
        Console.WriteLine("*********************************");
        Console.WriteLine("Insert your selected processors, separated by comma (first CPU index is 1):");
        var input = Console.ReadLine();
        Console.WriteLine("*********************************");
        var usedProcessors = input.Split(',');

        //TODO: validate input
        int newAffinity = 0;
        foreach (var item in usedProcessors)
        {
            newAffinity = newAffinity | int.Parse(item);
            Console.WriteLine("Processor #{0} was selected for affinity.", item);
        }
        Process.GetCurrentProcess().ProcessorAffinity = (System.IntPtr)newAffinity;
        Console.WriteLine("*********************************");
        Console.WriteLine("Current processor affinity is {0}", Process.GetCurrentProcess().ProcessorAffinity);
    }
}
于 2012-12-12T07:10:10.580 回答
5

Alex Filipovivi 提供的示例程序似乎不正确,因为它将处理器编号 OR 到 newAffinity 中,而没有首先将它们转换为设置位。所以如果你在这个程序中输入 3,4,你会得到一个 7 的关联掩码,即核心 1、2 和 3!掩码应设置为 12(十六进制 0xC,二进制 1100,如果位 0 是最低有效位,则设置位 2 和位 3)。

更换

newAffinity = NewAffinity | int.Parse(item);

newAffinity = newAffinity | (1 << int.Parse(item)-1);

是一种合理的方式来做到这一点。

于 2013-07-25T23:34:01.457 回答
2

对于寻找线程亲和力的人。

public class CpuAffinity
{
    [DllImport("kernel32.dll")]
    static extern IntPtr GetCurrentThread();

    [DllImport("kernel32.dll")]
    static extern IntPtr SetThreadAffinityMask(IntPtr hThread, IntPtr dwThreadAffinityMask);

    /// <summary>
    /// Sets the current Thread to have affinity to the specified cpu/processor if the system has more than one.
    /// 
    /// Supports most systems as we use a signed int; Anything more than 31 CPU's will not be supported.
    /// </summary>
    /// <param name="cpu">The index of CPU to set.</param>
    public static void SetCurrentThreadToHaveCpuAffinityFor(int cpu)
    {
        if (cpu < 0)
        {
            throw new ArgumentOutOfRangeException("cpu");
        }

        if (Environment.ProcessorCount > 1)
        {
            var ptr = GetCurrentThread();
            SetThreadAffinityMask(ptr, new IntPtr(1 << cpu));

            Debug.WriteLine("Current Thread Of OS Id '{0}' Affinity Set for CPU #{1}.", ptr, cpu);
        }else
        {
            Debug.WriteLine("The System only has one Processor.  It is impossible to set CPU affinity for other CPU's that do not exist.");
        }
    }
}
于 2015-03-25T01:10:46.957 回答
1

System.Diagnostics.Process.ProcessorAffinity

你想用来做什么Environment.ProcessorCount?用户输入验证?无论如何,如果您想选择特定的处理器(#1 或 #2 或 #3...),请创建一个像这样的位掩码:

if (userSelection <= 0 || userSelection > Environment.ProcessorCount)
{
    throw new ArgumentOutOfRangeException();
}

int bitMask = 1 << (userSelection - 1);
Process.GetCurrentProcess().ProcessorAffinity = (IntPtr)bitMask;

其中 userSelection - 是选定处理器的数量。

如果您想选择多个处理器,请执行

bitMask |= 1 << (anotherUserSelection - 1);

对于每个用户选择

于 2012-12-12T07:37:32.193 回答