รับซับสตริงระหว่างอักขระสองตัวโดยใช้ javascript


195

ฉันพยายามที่จะดึงสตริงจากภายในสตริงที่มีขนาดใหญ่ขึ้นซึ่งมันจะทำให้ทุกอย่างอยู่ระหว่าง ':' และ a ';'

ปัจจุบัน

Str = 'MyLongString:StringIWant;'

ผลลัพธ์ที่ต้องการ

newStr = 'StringIWant'

คำตอบ:


426

คุณสามารถลองสิ่งนี้

var mySubString = str.substring(
    str.lastIndexOf(":") + 1, 
    str.lastIndexOf(";")
);

4
วิธีแก้ปัญหาเล็กน้อย แต่มีประโยชน์โดยเฉพาะถ้าคุณต้องการหลีกเลี่ยงการแสดงออกปกติ
Nikolay Frick

8
ไม่มีใครรู้ว่าฉันจะทำเช่นนี้สำหรับการเกิดสตริงย่อยระหว่างสตริงเริ่มต้นและสิ้นสุดของฉันทุกครั้งหรือไม่
MarksCode

6
@VirtualTroll 8 วินาที? ฉันต้องการเห็น "ทางออก" ของคุณ: D
tom

@tom แม้ว่า "8 วินาที" ที่ไม่มีบริบทจะไม่แสดงข้อมูลที่มีความหมาย แต่ฉันค่อนข้างแน่ใจว่ามันไม่ต่างจากการทำงานครั้งเดียว)
อัตตา

7
ฉันแน่ใจว่าเขาหมายความว่าคำตอบนี้โพสต์ 8 วินาทีก่อนเขา ในขณะที่เขากำลังเขียนมัน
MaxSantos

111

คุณสามารถลองสิ่งนี้:

var str = 'one:two;three';    
str.split(':').pop().split(';')[0]; // returns 'two'

13
ไม่มี regex รักมัน
flimflam57

str.split(':').pop().split(';')[0]อาจเร็วกว่าการใช้.shift()
MysteryPancake

48

ใช้ split()

var s = 'MyLongString:StringIWant;';
var arrStr = s.split(/[:;]/);
alert(arrStr);

arrStrจะมีสตริงทั้งหมดที่คั่นด้วย:หรือ;
ดังนั้นเข้าถึงทุกสตริงผ่านfor-loop

for(var i=0; i<arrStr.length; i++)
    alert(arrStr[i]);

สตริงที่ฉันต้องการอยู่ระหว่าง [] และสิ่งนี้ไม่ทำงาน ... เช่น: 'MyLongString [StringIWant]'. split (/ [[]] /);
Philippe

1
@Philippe สำหรับกรณีการใช้งานของคุณให้ใช้ regex นี้\[(.*?)\] --- ในระยะสั้นคุณต้องหลีกเลี่ยงวงเล็บเหลี่ยมเนื่องจาก [] หมายถึงคลาสอักขระใน regex
asifsid88

33

@Basasaheb Gosavi คำตอบนั้นสมบูรณ์แบบถ้าคุณมีวัสดุย่อย (หนึ่ง: "และ"; ") แต่เมื่อคุณมีเหตุการณ์หลายครั้งมันอาจจะยุ่งยากเล็กน้อย


ทางออกที่ดีที่สุดที่ฉันได้รับในการทำงานกับหลายโครงการคือการใช้สี่วิธีภายในวัตถุ

  • วิธีแรก:คือการรับซับสตริงจากระหว่างสองสตริงจริง ๆ (อย่างไรก็ตามมันจะค้นหาผลลัพธ์เดียวเท่านั้น)
  • วิธีที่สอง:จะลบผลลัพธ์ที่จะได้รับล่าสุดที่มีสตริงย่อยหลังและก่อนหน้า
  • วิธีที่สาม:จะทำสองวิธีข้างต้นซ้ำในสตริง
  • วิธีที่สี่:จะใช้วิธีที่สามและส่งคืนผลลัพธ์

รหัส

พอคุยกันแล้วเราจะเห็นรหัส:

var getFromBetween = {
    results:[],
    string:"",
    getFromBetween:function (sub1,sub2) {
        if(this.string.indexOf(sub1) < 0 || this.string.indexOf(sub2) < 0) return false;
        var SP = this.string.indexOf(sub1)+sub1.length;
        var string1 = this.string.substr(0,SP);
        var string2 = this.string.substr(SP);
        var TP = string1.length + string2.indexOf(sub2);
        return this.string.substring(SP,TP);
    },
    removeFromBetween:function (sub1,sub2) {
        if(this.string.indexOf(sub1) < 0 || this.string.indexOf(sub2) < 0) return false;
        var removal = sub1+this.getFromBetween(sub1,sub2)+sub2;
        this.string = this.string.replace(removal,"");
    },
    getAllResults:function (sub1,sub2) {
        // first check to see if we do have both substrings
        if(this.string.indexOf(sub1) < 0 || this.string.indexOf(sub2) < 0) return;

        // find one result
        var result = this.getFromBetween(sub1,sub2);
        // push it to the results array
        this.results.push(result);
        // remove the most recently found one from the string
        this.removeFromBetween(sub1,sub2);

        // if there's more substrings
        if(this.string.indexOf(sub1) > -1 && this.string.indexOf(sub2) > -1) {
            this.getAllResults(sub1,sub2);
        }
        else return;
    },
    get:function (string,sub1,sub2) {
        this.results = [];
        this.string = string;
        this.getAllResults(sub1,sub2);
        return this.results;
    }
};

วิธีใช้?

ตัวอย่าง:

var str = 'this is the haystack {{{0}}} {{{1}}} {{{2}}} {{{3}}} {{{4}}} some text {{{5}}} end of haystack';
var result = getFromBetween.get(str,"{{{","}}}");
console.log(result);
// returns: [0,1,2,3,4,5]

รับRangeError: Maximum call stack size exceededข้อยกเว้น
Alejandro Cotilla

1
คำตอบที่ดี นี่คือสิ่งที่ฉันต้องการ
Andres Felipe

20
var s = 'MyLongString:StringIWant;';
/:([^;]+);/.exec(s)[1]; // StringIWant

1
วัตถุประสงค์คืออะไร; ใน [^;]
Jaakko Karhu

2
การแปลคือ: "/" เริ่มต้นรูปแบบ ตรงกับ ":" กับ "[]" ทุกอย่างของ "^;" ไม่ใช่เซมิโคลอน "+" ซ้ำ ๆ แล้วค้นหา "; เซมิโคลอนและ "/" สิ้นสุดรูปแบบ
DeveloperWeeks

15

ฉันชอบวิธีนี้:

var str = 'MyLongString:StringIWant;';
var tmpStr  = str.match(":(.*);");
var newStr = tmpStr[1];
//newStr now contains 'StringIWant'

ฉันทดสอบสิ่งนี้ใน Web Part SharePoint 2013 และใช้งานได้ดีถ้ามันช่วยทุกคนในอนาคต!
เชนกิบ

3
สิ่งนี้อาจไม่ทำงานหากสตริงที่คุณต้องการอยู่ระหว่าง "(" และ ")"
bravokeyl

4

ฉันใช้วิธี @tsds แต่โดยใช้ฟังก์ชั่นแยกเท่านั้น

var str = 'one:two;three';    
str.split(':')[1].split(';')[0] // returns 'two'

คำเตือน: หาก therre ไม่ใช่ ":" ในสตริงที่เข้าถึงดัชนี '1' ของอาเรย์จะทำให้เกิดข้อผิดพลาด! str.split ( ':') [1]

ดังนั้นวิธี @tsds จะปลอดภัยกว่าหากมีความไม่แน่นอน

str.split(':').pop().split(';')[0]

4
function substringBetween(s, a, b) {
    var p = s.indexOf(a) + a.length;
    return s.substring(p, s.indexOf(b, p));
}

// substringBetween('MyLongString:StringIWant;', ':', ';') -> StringIWant
// substringBetween('MyLongString:StringIWant;;', ':', ';') -> StringIWant
// substringBetween('MyLongString:StringIWant;:StringIDontWant;', ':', ';') -> StringIWant

นี่เป็นทางออกที่ยอดเยี่ยมสำหรับสายอักขระระหว่าง 2 ตัวอักษร
Phani Shashank

3

คุณสามารถใช้ฟังก์ชั่นการสั่งซื้อที่สูงขึ้นเพื่อส่งคืนตัวแยกข้อมูลเวอร์ชันที่คอมไพล์แล้วซึ่งจะเร็วขึ้น

ด้วย regexes และรวบรวม regex เพียงครั้งเดียวการปิดการแข่งขันของ Javascript จะส่งคืนการแข่งขันทั้งหมด

สิ่งนี้ทำให้เราต้องลบสิ่งที่เราใช้เป็นเครื่องหมายของเรา (เช่น:) เท่านั้น{{และเราสามารถใช้ความยาวสตริงสำหรับสิ่งนี้ด้วยชิ้น

function extract([beg, end]) {
    const matcher = new RegExp(`${beg}(.*?)${end}`,'gm');
    const normalise = (str) => str.slice(beg.length,end.length*-1);
    return function(str) {
        return str.match(matcher).map(normalise);
    }
}

รวบรวมครั้งเดียวและใช้หลายครั้ง ...

const stringExtractor = extract(['{','}']);
const stuffIneed = stringExtractor('this {is} some {text} that can be {extracted} with a {reusable} function');
// Outputs: [ 'is', 'text', 'extracted', 'reusable' ]

หรือใช้ครั้งเดียว ...

const stuffIneed = extract(['{','}'])('this {is} some {text} that can be {extracted} with a {reusable} function');
// Outputs: [ 'is', 'text', 'extracted', 'reusable' ]

ดูreplaceฟังก์ชันของ Javascript ด้วยเช่นกัน แต่ใช้ฟังก์ชั่นแทนอาร์กิวเมนต์แทน (คุณจะทำเช่นนั้นถ้าเช่นคุณกำลังสร้างเอ็นจินเทมเพลตขนาดเล็ก (การแก้ไขสตริง) ... lodash.get ก็มีประโยชน์เช่นกันเพื่อรับค่าที่คุณต้องการ แทนที่ด้วย? ...

คำตอบของฉันยาวเกินไป แต่อาจช่วยใครซักคน!


2

คุณสามารถใช้อันนี้ ...

function extractText(str,delimiter){
  if (str && delimiter){
    var firstIndex = str.indexOf(delimiter)+1;
    var lastIndex = str.lastIndexOf(delimiter);
    str = str.substring(firstIndex,lastIndex);
  }
  return str;
}


var quotes = document.getElementById("quotes");

// &#34 - represents quotation mark in HTML
<div>


  <div>
  
    <span id="at">
      My string is @between@ the "at" sign
    </span>
    <button onclick="document.getElementById('at').innerText = extractText(document.getElementById('at').innerText,'@')">Click</button>
  
  </div>
  
  <div>
    <span id="quotes">
      My string is "between" quotes chars
    </span>
    <button onclick="document.getElementById('quotes').innerText = extractText(document.getElementById('quotes').innerText,'&#34')">Click</button>
  
  </div>

</div>


2

รับสตริงระหว่างสองสตริงย่อย (มีอักขระมากกว่า 1 ตัว)

function substrInBetween(whole_str, str1, str2){
   if (whole_str.indexOf(str1) === -1 || whole_str.indexOf(str2) === -1) {
       return undefined; // or ""
  }
  strlength1 = str1.length;
  return whole_str.substring(
                whole_str.indexOf(str1) + strlength1, 
                whole_str.indexOf(str2)
               );

   }

หมายเหตุฉันใช้indexOf()แทนlastIndexOf()ดังนั้นจะตรวจสอบการเกิดขึ้นครั้งแรกของสตริงเหล่านั้น


ฟังก์ชั่นที่ดีและอ่านง่ายมาก แต่วัตถุประสงค์ของstrlength1ตัวแปรคืออะไร? ค่าควรใช้แบบอินไลน์แทน นอกจากนี้ยังไม่ชัดเจนว่าคุณใช้รูปแบบกรณีใด strlength1- ไม่มีสไตล์whole_str- กล่องใส่งู
บอริส


1

ใช้jQuery :

get_between <- function(str, first_character, last_character) {
    new_str = str.match(first_character + "(.*)" + last_character)[1].trim()
    return(new_str)
    }

เชือก

my_string = 'and the thing that ! on the @ with the ^^ goes now' 

การใช้งาน :

get_between(my_string, 'that', 'now')

ผลลัพธ์ :

"! on the @ with the ^^ goes

1

ฟังก์ชั่นเล็ก ๆ ที่ฉันสร้างขึ้นที่สามารถจับสตริงระหว่างและสามารถ (เป็นทางเลือก) ข้ามคำศัพท์ที่ตรงกันจำนวนหนึ่งเพื่อจับดัชนีเฉพาะ

นอกจากนี้การตั้งค่าstartที่falseจะใช้การเริ่มต้นของสตริงและการตั้งค่าendที่falseจะใช้จุดสิ้นสุดของสตริง

กำหนดpos1เป็นตำแหน่งของstartข้อความที่คุณต้องการใช้1จะใช้การเกิดขึ้นครั้งแรกของstart

pos2ทำสิ่งเดียวกับpos1แต่สำหรับendและ1จะใช้สิ่งที่เกิดขึ้นครั้งแรกendหลังจากเท่านั้นstartเหตุการณ์ที่เกิดขึ้นendก่อนหน้าstartนั้นจะถูกละเว้น

function getStringBetween(str, start=false, end=false, pos1=1, pos2=1){
  var newPos1 = 0;
  var newPos2 = str.length;

  if(start){
    var loops = pos1;
    var i = 0;
    while(loops > 0){
      if(i > str.length){
        break;
      }else if(str[i] == start[0]){
        var found = 0;
        for(var p = 0; p < start.length; p++){
          if(str[i+p] == start[p]){
            found++;
          }
        }
        if(found >= start.length){
          newPos1 = i + start.length;
          loops--;
        }
      }
      i++;
    }
  }

  if(end){
    var loops = pos2;
    var i = newPos1;
    while(loops > 0){
      if(i > str.length){
        break;
      }else if(str[i] == end[0]){
        var found = 0;
        for(var p = 0; p < end.length; p++){
          if(str[i+p] == end[p]){
            found++;
          }
        }
        if(found >= end.length){
          newPos2 = i;
          loops--;
        }
      }
      i++;
    }
  }

  var result = '';
  for(var i = newPos1; i < newPos2; i++){
    result += str[i];
  }
  return result;
}

1

นี่อาจเป็นทางออกที่เป็นไปได้

var str = 'RACK NO:Stock;PRODUCT TYPE:Stock Sale;PART N0:0035719061;INDEX NO:21A627 042;PART NAME:SPRING;';  
var newstr = str.split(':')[1].split(';')[0]; // return value as 'Stock'

console.log('stringvalue',newstr)
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.