Answers:
如果可以操纵数组之一,则可以在执行复制之前调整其大小:
T[] array1 = getOneArray();
T[] array2 = getAnotherArray();
int array1OriginalLength = array1.Length;
Array.Resize<T>(ref array1, array1OriginalLength + array2.Length);
Array.Copy(array2, 0, array1, array1OriginalLength, array2.Length);
否则,您可以制作一个新数组
T[] array1 = getOneArray();
T[] array2 = getAnotherArray();
T[] newArray = new T[array1.Length + array2.Length];
Array.Copy(array1, newArray, array1.Length);
Array.Copy(array2, 0, newArray, array1.Length, array2.Length);
Array.Resize
实际上并不会调整数组的大小,而是会对其进行复制。这就是第一个参数是by-ref的原因(这意味着您的第一个代码可能不会编译)。
在C#3.0中,可以使用LINQ的Concat方法轻松完成此操作:
int[] front = { 1, 2, 3, 4 };
int[] back = { 5, 6, 7, 8 };
int[] combined = front.Concat(back).ToArray();
在C#2.0中,您没有直接的方法,但是Array.Copy可能是最好的解决方案:
int[] front = { 1, 2, 3, 4 };
int[] back = { 5, 6, 7, 8 };
int[] combined = new int[front.Length + back.Length];
Array.Copy(front, combined, front.Length);
Array.Copy(back, 0, combined, front.Length, back.Length);
这可以轻松地用于实现您自己的Concat
。
使用LINQ:
var arr1 = new[] { 1, 2, 3, 4, 5 };
var arr2 = new[] { 6, 7, 8, 9, 0 };
var arr = arr1.Union(arr2).ToArray();
请记住,这将删除重复项。如果要保留重复项,请使用Concat。
首先,请确保您问自己一个问题:“我真的应该在这里使用数组吗?”
除非您要构建对速度至关重要的东西,否则最好使用类型化的List List<int>
。通过网络发送内容时,我唯一一次使用数组是用于字节数组。除此之外,我从未碰过它们。
List<T>
在公共API中公开:blogs.msdn.com/b/kcwalina/archive/2005/09/26/474010.aspx
使用LINQ更容易:
var array = new string[] { "test" }.ToList();
var array1 = new string[] { "test" }.ToList();
array.AddRange(array1);
var result = array.ToArray();
首先将数组转换为列表并将其合并...之后,只需将列表转换回数组即可:)
我认为您可以为此使用Array.Copy。它需要一个源索引和一个目标索引,因此您应该能够将一个数组追加到另一个数组。如果您需要的不只是将一个附加到另一个上而变得更加复杂,那么这可能不是适合您的工具。
假设目标阵列有足够的空间,Array.Copy()
将可以工作。您也可以尝试使用List<T>
和及其.AddRange()
方法。
就个人而言,我更喜欢自己的语言扩展,可以随意添加或删除这些语言扩展以快速进行原型制作。
以下是字符串的示例。
//resides in IEnumerableStringExtensions.cs
public static class IEnumerableStringExtensions
{
public static IEnumerable<string> Append(this string[] arrayInitial, string[] arrayToAppend)
{
string[] ret = new string[arrayInitial.Length + arrayToAppend.Length];
arrayInitial.CopyTo(ret, 0);
arrayToAppend.CopyTo(ret, arrayInitial.Length);
return ret;
}
}
它比LINQ和Concat快得多。更快速的是,使用了一个自定义的IEnumerable
Type-wrapper,它存储传递的数组的引用/指针,并允许遍历整个集合,就好像它是普通数组一样。(在HPC,图形处理,图形渲染中很有用...)
您的代码:
var someStringArray = new[]{"a", "b", "c"};
var someStringArray2 = new[]{"d", "e", "f"};
someStringArray.Append(someStringArray2 ); //contains a,b,c,d,e,f
有关整个代码和泛型版本,请参见:https : //gist.github.com/lsauer/7919764
注意:这将返回未扩展的IEnumerable对象。返回扩展对象要慢一些。
自2002年以来,我编译了此类扩展,其中许多功劳归功于CodeProject和“ Stackoverflow”上的有用人员。我将在短期内发布这些链接,并将链接放在此处。
这就是我想出的。适用于可变数量的数组。
public static T[] ConcatArrays<T>(params T[][] args)
{
if (args == null)
throw new ArgumentNullException();
var offset = 0;
var newLength = args.Sum(arr => arr.Length);
var newArray = new T[newLength];
foreach (var arr in args)
{
Buffer.BlockCopy(arr, 0, newArray, offset, arr.Length);
offset += arr.Length;
}
return newArray;
}
...
var header = new byte[] { 0, 1, 2};
var data = new byte[] { 3, 4, 5, 6 };
var checksum = new byte[] {7, 0};
var newArray = ConcatArrays(header, data, checksum);
//output byte[9] { 0, 1, 2, 3, 4, 5, 6, 7, 0 }
只是将其记录为一个选项:如果您使用的数组是原始类型-布尔型(布尔型),Char,SByte,字节,Int16(短型),UInt16,Int32(int),UInt32,Int64(长型) ),UInt64,IntPtr,UIntPtr,Single或Double-那么您可以(或应该?)尝试使用Buffer.BlockCopy。根据Buffer类的MSDN页面:
与System.Array类中的类似方法相比,此类提供了更好的操作原始类型的性能。
以@OwenP的答案中的C#2.0示例作为起点,它的工作方式如下:
int[] front = { 1, 2, 3, 4 };
int[] back = { 5, 6, 7, 8 };
int[] combined = new int[front.Length + back.Length];
Buffer.BlockCopy(front, 0, combined, 0, front.Length);
Buffer.BlockCopy(back, 0, combined, front.Length, back.Length);
Buffer.BlockCopy
和Array.Copy
@OwenP所使用的语法之间几乎没有语法差异,但这应该更快(即使只有一点点)。
如果其他人正在寻找如何合并两个图像字节数组:
private void LoadImage()
{
string src = string.empty;
byte[] mergedImageData = new byte[0];
mergedImageData = MergeTwoImageByteArrays(watermarkByteArray, backgroundImageByteArray);
src = "data:image/png;base64," + Convert.ToBase64String(mergedImageData);
MyImage.ImageUrl = src;
}
private byte[] MergeTwoImageByteArrays(byte[] imageBytes, byte[] imageBaseBytes)
{
byte[] mergedImageData = new byte[0];
using (var msBase = new MemoryStream(imageBaseBytes))
{
System.Drawing.Image imgBase = System.Drawing.Image.FromStream(msBase);
Graphics gBase = Graphics.FromImage(imgBase);
using (var msInfo = new MemoryStream(imageBytes))
{
System.Drawing.Image imgInfo = System.Drawing.Image.FromStream(msInfo);
Graphics gInfo = Graphics.FromImage(imgInfo);
gBase.DrawImage(imgInfo, new Point(0, 0));
//imgBase.Save(Server.MapPath("_____testImg.png"), ImageFormat.Png);
MemoryStream mergedImageStream = new MemoryStream();
imgBase.Save(mergedImageStream, ImageFormat.Png);
mergedImageData = mergedImageStream.ToArray();
mergedImageStream.Close();
}
}
return mergedImageData;
}
这是一个使用Array.CopyTo的简单示例。我认为它回答了您的问题,并提供了一个CopyTo用法的示例-当我需要使用此函数时,我总是很困惑,因为帮助有点不清楚-索引是目标数组在其中发生插入的位置。
int[] xSrc1 = new int[3] { 0, 1, 2 };
int[] xSrc2 = new int[5] { 3, 4, 5, 6 , 7 };
int[] xAll = new int[xSrc1.Length + xSrc2.Length];
xSrc1.CopyTo(xAll, 0);
xSrc2.CopyTo(xAll, xSrc1.Length);
我想你再简单不过了。
我需要一个解决方案来合并数量未知的数组。
惊讶没有人使用提供了解决方案SelectMany
有params
。
private static T[] Combine<T>(params IEnumerable<T>[] items) =>
items.SelectMany(i => i).Distinct().ToArray();
如果您不希望有不同的项目,只需删除不同的项目即可。
public string[] Reds = new [] { "Red", "Crimson", "TrafficLightRed" };
public string[] Greens = new [] { "Green", "LimeGreen" };
public string[] Blues = new [] { "Blue", "SkyBlue", "Navy" };
public string[] Colors = Combine(Reds, Greens, Blues);
注意:使用distinct时,绝对不能保证订购。
我假设您使用的是自己的数组类型,而不是内置的.NET数组:
public string[] merge(input1, input2)
{
string[] output = new string[input1.length + input2.length];
for(int i = 0; i < output.length; i++)
{
if (i >= input1.length)
output[i] = input2[i-input1.length];
else
output[i] = input1[i];
}
return output;
}
完成此操作的另一种方法是使用内置的ArrayList类。
public ArrayList merge(input1, input2)
{
Arraylist output = new ArrayList();
foreach(string val in input1)
output.add(val);
foreach(string val in input2)
output.add(val);
return output;
}
两个示例都是C#。
int [] SouceArray1 = new int[] {2,1,3};
int [] SourceArray2 = new int[] {4,5,6};
int [] targetArray = new int [SouceArray1.Length + SourceArray2.Length];
SouceArray1.CopyTo(targetArray,0);
SourceArray2.CopyTo(targetArray,SouceArray1.Length) ;
foreach (int i in targetArray) Console.WriteLine(i + " ");
使用上面的代码,可以很容易地合并两个数组。
创建和扩展方法来处理null
public static class IEnumerableExtenions
{
public static IEnumerable<T> UnionIfNotNull<T>(this IEnumerable<T> list1, IEnumerable<T> list2)
{
if (list1 != null && list2 != null)
return list1.Union(list2);
else if (list1 != null)
return list1;
else if (list2 != null)
return list2;
else return null;
}
}
如果您在数组本身中包含源数组,则可以使用SelectMany:
var arrays = new[]{new[]{1, 2, 3}, new[]{4, 5, 6}};
var combined = arrays.SelectMany(a => a).ToArray();
foreach (var v in combined) Console.WriteLine(v);
给
1
2
3
4
5
6
可能这不是最快的方法,但可能取决于用例。
此代码适用于所有情况:
int[] a1 ={3,4,5,6};
int[] a2 = {4,7,9};
int i = a1.Length-1;
int j = a2.Length-1;
int resultIndex= i+j+1;
Array.Resize(ref a2, a1.Length +a2.Length);
while(resultIndex >=0)
{
if(i != 0 && j !=0)
{
if(a1[i] > a2[j])
{
a2[resultIndex--] = a[i--];
}
else
{
a2[resultIndex--] = a[j--];
}
}
else if(i>=0 && j<=0)
{
a2[resultIndex--] = a[i--];
}
else if(j>=0 && i <=0)
{
a2[resultIndex--] = a[j--];
}
}