c# contains part of string

2020-07-06 03:02发布

问题:

So I have a list with Materiel-objects. In Materiel I have 15 get and set methods. I want to construct a search-method that loops all the objects in the list, and all of the variables in each Materiel-object. The looping part is easy enough, but I'm struggling with the string-contains-part. The search term could for instance be "acto", and I should get a hit for "Tractor". I have tried using the string-Contains class, but as far as I can figure out, it only checks the string beginning in position 0. So "Tra" gets a hit, but not "acto".

Is there any build in classes, or should I program one myself?

Sorry for the bad explanation.

My code. I see now that I get hits for the substring, but also other results :)

    protected void Button_search_Click(object sender, EventArgs e)
    {
        string searchTerm = TextBox1.Text.ToString().ToLower();

        TableRow row;
        TableCell cell;

        int rowNumber = 1;

        foreach (Materiell mat in allItems)
        {
            if (searchTerm.Contains(mat.itemID.ToString().ToLower()) ||
                searchTerm.Contains(mat.manufacturer.ToLower()) ||
                searchTerm.Contains(mat.model.ToLower()) ||
                searchTerm.Contains(mat.serialNo.ToLower()) ||
                searchTerm.Contains(mat.dateProd.ToString().ToLower()) ||
                searchTerm.Contains(mat.location.ToLower()) ||
                searchTerm.Contains(mat.mainCategory.ToLower()) ||
                searchTerm.Contains(mat.subCategory.ToLower()) ||
                searchTerm.Contains(mat.dateAcquired.ToString().ToLower()) ||
                searchTerm.Contains(mat.price.ToString().ToLower()) ||
                searchTerm.Contains(mat.ownerID.ToString().ToLower()) ||
                searchTerm.Contains(mat.extra.ToString().ToLower()) ||
                searchTerm.Contains(mat.textComment.ToLower()) ||
                searchTerm.Contains(mat.active.ToString().ToLower()))
            {
                row = new TableRow();
                row.ID = "row" + rowNumber.ToString();
                rowNumber++;

                cell = new TableCell();
                cell.Text = "<a href=\"#\" class=\"opendiv\">" + mat.itemID.ToString() + "</a>";
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.manufacturer.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.model.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.serialNo.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.dateProd.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.location.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.mainCategory.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.subCategory.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.dateAcquired.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.price.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.ownerID.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.extra.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.ownDefData.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.textComment.ToString();
                row.Cells.Add(cell);

                cell = new TableCell();
                cell.Text = mat.active.ToString();
                row.Cells.Add(cell);

                Table1.Rows.Add(row);
            }
        }
    }

回答1:

"some string".Contains("str") will return true, are you having problems with case sesitivity?

If so you could use this:

public static bool Contains(this string source, string toCheck, StringComparison comp) {
  return source.IndexOf(toCheck, comp) >= 0;
}

string title = "STRING";
bool contains = title.Contains("string", StringComparison.OrdinalIgnoreCase);

(Taken from Case insensitive 'Contains(string)')



回答2:

Use IndexOf

string searchWithinThis = "ABCDEFGHIJKLMNOP";
string searchForThis = "DEF";
int firstCharacter = searchWithinThis.IndexOf(searchForThis);

Console.WriteLine("First occurrence: {0}", firstCharacter);

if substring is not found, it returns -1. Its very useful for also knowing where abouts the string is found.



回答3:

For shits and giggles I thought it was a nice lunch-break project to come up with a simple, but 'elegant' solution to the Question (as I understood it :)):

E.g.

// I made up a Material class for testing:
public class Materiel
{
    public string A { get; set; }
    public int B { get; set; }
    public DateTime? C { get; set; }
    public string D { get; set; }
    public Nested E { get; set; }
}     

// [...] usage:

foreach (var pattern in new[]{ "World" , "dick", "Dick", "ick", "2012", "Attach" })
    Console.WriteLine("{0} records match '{1}'", Database.Search(pattern).Count(), pattern);

Outputs:

2 records match 'World'
1 records match 'dick'
1 records match 'Dick'
2 records match 'ick'
1 records match '2012'
2 records match 'Attach'

The code also supports

  • regex matches
  • any property types (e.g. nullable DateTimes or nested classes)
  • showing which property matched the pattern/substring

Enjoy:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Text.RegularExpressions;

namespace AClient
{
    public class Materiel
    {
        public string A { get; set; }
        public int B { get; set; }
        public DateTime? C { get; set; }
        public string D { get; set; }
        public Nested E { get; set; }
    }

    public struct Nested
    {
        public string Data { get; set; }
        public override string ToString() { return string.Format("Extra: {0}", Data); }
    }


    public static class FullText
    {
        public class PropMatched<T> { public PropertyInfo Property; public T Item; }

        public static IEnumerable<PropMatched<T> > ByProperty<T>(this IEnumerable<T> items, string substr)
        {
            return items.ByProperty(new Regex(Regex.Escape(substr), RegexOptions.IgnoreCase));
        }

        public static IEnumerable<PropMatched<T> > ByProperty<T>(this IEnumerable<T> items, Regex pattern)
        {
            return items.Select(i => i.MatchingProperties(pattern)).Where(m => null != m);
        }

        public static IEnumerable<T> Search<T>(this IEnumerable<T> items, string substr)
        {
            return items.Search(new Regex(Regex.Escape(substr), RegexOptions.IgnoreCase));
        }

        public static IEnumerable<T> Search<T>(this IEnumerable<T> items, Regex pattern)
        {
            return items.Where(i => null != i.MatchingProperties(pattern));
        }

        public static PropMatched<T> MatchingProperties<T>(this T item, Regex pattern)
        {
            if (null == pattern || null == item) return null;

            var properties = item.GetType().GetProperties(BindingFlags.Public | BindingFlags.FlattenHierarchy | BindingFlags.Instance);
            var matches = from   prop in properties 
                          let    val = prop.GetGetMethod(true).Invoke(item, new object[]{}) 
                          where  pattern.IsMatch((val??"").ToString()) 
                          select prop;

            var found = matches.FirstOrDefault();
            return found == null ? null : new PropMatched<T> {Item = item, Property = found};
        }
    }

    class Client
    {
        private static readonly IEnumerable<Materiel> Database = new List<Materiel>
            {
                new Materiel {
                        A = "Hello", B = 1, C = null, D = "World",
                        E = new Nested {Data = "Attachment"}
                    },
                new Materiel {
                        A = "Transfigured", B = 2, C = null, D = "Nights",
                        E = new Nested {Data = "Schoenberg"}
                    },
                new Materiel {
                        A = "Moby", B = 3, C = null, D = "Dick",
                        E = new Nested {Data = "Biographic"}
                    },
                new Materiel {
                        A = "Prick", B = 4, C = DateTime.Today, D = "World",
                        E = new Nested {Data = "Attachment"}
                    },
                new Materiel {
                        A = "Oh Noes", B = 2, C = null, D = "Nights",
                        E = new Nested {Data = "Schoenberg"}
                    },
            };


        static void Main()
        {
            foreach (var pattern in new[]{ "World" , "dick", "Dick", "ick", "2012", "Attach" })
                Console.WriteLine("{0} records match '{1}'", Database.Search(pattern).Count(), pattern);

            // regex sample:
            var regex = new Regex(@"N\w+s", RegexOptions.IgnoreCase);

            Console.WriteLine(@"{0} records match regular expression 'N\w+s'", Database.Search(regex).Count());

            // with context info:
            foreach (var contextMatch in Database.ByProperty(regex))
            {
                Console.WriteLine("1 match of regex in propery {0} with value '{1}'",
                    contextMatch.Property.Name, contextMatch.Property.GetGetMethod().Invoke(contextMatch.Item, new object[0]));

            }
        }
    }
}


回答4:

class SearchInString
{
    static void Main()
    {
        string strn= "A great things are happen with great humans.";
        System.Console.WriteLine("'{0}'",strn);

        bool case1= strn.StartsWith("A great");
        System.Console.WriteLine("starts with 'A great'? {0}", case1);

        bool case2= strn.StartsWith("A great", System.StringComparison.OrdinalIgnoreCase);
        System.Console.WriteLine("starts with 'A great'? {0} (ignoring case)", case2);

        bool case3= strn.EndsWith(".");
        System.Console.WriteLine("ends with '.'? {0}", case3);

        int start= strn.IndexOf("great");
        int end= strn.LastIndexOf("great");
        string strn2 = strn.Substring(start, end- start);
        System.Console.WriteLine("between two 'great' words: '{0}'", strn2);
    }
}


回答5:

 Bool doesContain = "Tractor".Contains("acto");

doesContain will be true.

Contains works on collections, and in this instance I think the string is treated as a collection of chars.

http://msdn.microsoft.com/en-us/library/dy85x1sa.aspx



回答6:

The string.Contains method does look for the substring anywhere in the string.

"asdf".Contains("as") --> True
"asdf".Contains("sd") --> True
"asdf".Contains("df") --> True
"asdf".Contains("xy") --> False

The comparison is however case sensetive, so you might need to convert case if you want to make an case insesetive search:

"Asdf".Contains("as") --> False
"Asdf".Contains("As") --> True

"Asdf".ToUpper().Contains("as".ToUpper()) --> True