1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264 |
- <?php
- namespace App\Console\Commands;
- use App\Model\AreaMap;
- use App\Model\BasicType;
- use App\Model\BasicTypeAllUse;
- use App\Model\Customer;
- use App\Model\CustomerInfo;
- use App\Model\CustomerReport;
- use App\Model\CustomerReportDepart;
- use App\Model\FollowUpRecord;
- use App\Model\SalesOrder;
- use App\Model\SeeRange;
- use Illuminate\Console\Command;
- use Illuminate\Support\Facades\DB;
- class CustomerWriteReport extends Command
- {
- /**
- * The name and signature of the console command.
- *
- * @var string
- */
- protected $signature = 'command:customer_write_report';
- /**
- * The console command description.
- *
- * @var string
- */
- protected $description = 'Command description';
- /**
- * Create a new command instance.
- *
- * @return void
- */
- public function __construct()
- {
- parent::__construct();
- }
- /**
- * Execute the console command.
- *
- * @return mixed
- */
- public function handle()
- {
- echo "执行任务--------start---------------\n";
- try {
- $this->customerWrite();
- }catch (\Throwable $exception){
- echo "发生异常:" . $exception->getFile() . "|" . $exception->getMessage() . "|" . $exception->getLine() . "\n";
- }
- echo "执行任务--------end---------------\n";
- }
- private function customerWrite(){
- //今天到上月今天的时间戳
- $nowStamp = time();
- $lastMonth = date("Y-m-d 00:00:00", strtotime("-1 month"));
- $lastMonthStamp = strtotime($lastMonth);
- list($provinceMap, $cityMap, $countyMap) = $this->constructAddress();
- // dd($provinceMap, $cityMap, $countyMap);
- // dd(count($provinceMap), count($cityMap), count($countyMap));
- // $this->forInsertArea($provinceMap, $cityMap, $countyMap);dd(122);
- Customer::where("del_time",0)
- ->where('enter_time','>=', $lastMonthStamp)
- ->where('enter_time','<=', $nowStamp)
- ->orderBy('enter_time','asc')
- ->select('id as customer_id','title','model_type','consulting_product_new','enter_time','address2','top_depart_id','customer_from','car_type')
- ->chunk(200, function ($data) use($provinceMap, $cityMap, $countyMap,$nowStamp){
- // 开启事务
- DB::transaction(function () use ($data, $provinceMap, $cityMap, $countyMap, $nowStamp) {
- $dataArray = $data->toArray();
- list($main, $main_depart) = $this->processingData($dataArray, $provinceMap, $cityMap, $countyMap, $nowStamp);
- if (! empty($main)) {
- $customer_ids = array_column($dataArray, 'customer_id');
- // 删除旧数据
- CustomerReport::where('del_time', 0)
- ->whereIn('customer_id', $customer_ids)
- ->update(['del_time' => $nowStamp]);
- CustomerReportDepart::where('del_time', 0)
- ->whereIn('customer_id', $customer_ids)
- ->update(['del_time' => $nowStamp]);
- // 写入新数据
- CustomerReport::insert($main);
- CustomerReportDepart::insert($main_depart);
- echo '写入中' . PHP_EOL;
- } else {
- echo '无' . PHP_EOL;
- }
- });
- });
- }
- private function processingData($data,$provinceMap,$cityMap,$countyMap,$time){
- $return = [];
- $customer_id = array_column($data,'customer_id');
- $range = SeeRange::where('del_time',0)
- ->whereIn('data_id', $customer_id)
- ->where('data_type',SeeRange::type_one)
- ->where('type',SeeRange::data_three)
- ->select('data_id as customer_id','param_id as top_depart_id')
- ->get()->toArray();
- $map = [];
- foreach ($range as $value){
- $map[$value['customer_id']][] = $value['top_depart_id'];
- }
- $isset_map = SalesOrder::where('del_time', 0)
- ->whereIn('customer_id', $customer_id)
- ->pluck('customer_id')
- ->unique()
- ->flip()
- ->map(fn() => 1)
- ->toArray();
- // 最新的跟进记录
- $latestRecords = FollowUpRecord::whereIn('data_id', $customer_id)
- ->where('type', FollowUpRecord::type_one)
- ->where('del_time',0)
- ->whereIn('id', function ($query) use ($customer_id) {
- $query->select(DB::raw('MAX(id)'))
- ->from('follow_up_record')
- ->whereColumn('data_id', 'follow_up_record.data_id')
- ->whereIn('data_id', $customer_id)
- ->where('type', FollowUpRecord::type_one)
- ->where('del_time',0)
- ->groupBy('data_id');
- })
- ->get();
- $latestRecordsArray = $latestRecords->keyBy('data_id')
- ->map(fn($item) => $item->follow_type)
- ->toArray();
- $customer_info = CustomerInfo::where('del_time',0)
- ->whereIn('customer_id',$customer_id)
- ->whereIn('type',[CustomerInfo::type_one])
- ->select('type','contact_type','contact_info','customer_id')
- ->get()->toArray();
- $customer_info_map = [];
- foreach ($customer_info as $value){
- if(! $value['contact_info']) continue;
- if(! empty($customer_info_map[$value['customer_id']])) continue;
- $customer_info_map[$value['customer_id']] = $value['contact_info'];
- }
- $array = array_unique(array_merge_recursive(array_column($data,'customer_from'),array_column($data,'car_type')));
- $basic_map = BasicType::whereIn('id',$array)
- ->pluck('title','id')
- ->toArray();
- $basic_all_map = BasicTypeAllUse::where('type', BasicTypeAllUse::type_one)
- ->whereIn('id',array_unique(array_column($data,'consulting_product_new')))
- ->pluck('title','id')
- ->toArray();
- $detail_insert = [];
- foreach ($data as $key => $value){
- $province = $province_code = $city = $city_code = "";
- $is_success = 0;
- if(! empty($value['address2'])){
- $tmp = $this->parseAddress($value['address2'], $provinceMap, $cityMap,$countyMap);
- if(empty($tmp['province']['label']) || empty($tmp['city']['label'])){
- // $tmp['origin'] = $value['address2'];
- // $a = [
- // "provice" => $tmp['province']['label'],
- // "provice_code" => $tmp['province']['value'],
- // "city" => $tmp['city']['label'],
- // "city_code" => $tmp['city']['value'],
- // "origin" => $value['address2']
- // ];
- // $return[] = $a;
- }else{
- $province = $tmp['province']['label'];
- $province_code = $tmp['province']['value'];
- $city = $tmp['city']['label'];
- $city_code = $tmp['city']['value'];
- $is_success = 1;
- }
- }
- $data[$key]['province'] = $province;
- $data[$key]['province_code'] = $province_code;
- $data[$key]['city'] = $city;
- $data[$key]['city_code'] = $city_code;
- $data[$key]['is_success'] = $is_success;
- $belong_depart_id = [$value['top_depart_id']];
- if(isset($map[$value['customer_id']])) $belong_depart_id = array_unique($map[$value['customer_id']]);
- foreach ($belong_depart_id as $b){
- $detail_insert[] = [
- 'customer_id' => $value['customer_id'],
- 'top_depart_id' => $b,
- 'crt_time' => $time
- ];
- }
- $type = -1;
- if(isset($isset_map[$value['customer_id']])){
- $type = 3;
- }else{
- if(isset($latestRecordsArray[$value['customer_id']])) $type = $latestRecordsArray[$value['customer_id']];
- }
- $data[$key]['type'] = $type;
- $data[$key]['contact_info'] = $customer_info_map[$value['customer_id']] ?? "";
- $data[$key]['customer_from'] = $basic_map[$value['customer_from']] ?? "";
- $data[$key]['car_type'] = $customer_info_map[$value['car_type']] ?? "";
- $data[$key]['consulting_product_new'] = $basic_all_map[$value['consulting_product_new']] ?? "";
- $data[$key]['crt_time'] = $time;
- }
- return [$data, $detail_insert];
- }
- private function forInsertArea($provinceMap, $cityMap, $countyMap){
- $insert = [];
- foreach ($provinceMap as $key => $value){
- $insert[] = [
- 'key' => $key,
- 'label' => $value['label'],
- 'value' => $value['value'],
- 'province' => "",
- 'type' => 1,
- ];
- }
- foreach ($cityMap as $key => $value){
- $insert[] = [
- 'key' => $key,
- 'label' => $value['label'],
- 'value' => $value['value'],
- 'province' => $value['province'],
- 'type' => 2,
- ];
- }
- foreach ($countyMap as $key => $value){
- $insert[] = [
- 'key' => $key,
- 'label' => $value['city_label'],
- 'value' => $value['city_value'],
- 'province' => $value['province_value'],
- 'type' => 3,
- ];
- }
- AreaMap::where('del_time',0)
- ->where('is_create_by_human',0)
- ->update(['del_time' => time()]);
- AreaMap::insert($insert);
- }
- private function constructAddress($type = 0) {
- if(! empty($type)){
- $result = AreaMap::where('del_time',0)
- ->select('key','label','value','province','type')
- ->get()->toArray();
- $provinceMap = [];
- $cityMap = [];
- $countyMap = []; // 新增区县映射表
- foreach ($result as $value){
- if($value['type'] == 1){
- $provinceMap[$value['key']] = [
- 'label' => $value['label'],
- 'value' => $value['value'],
- ];
- }elseif($value['type'] == 2){
- $cityMap[$value['key']] = [
- 'label' => $value['label'],
- 'value' => $value['value'],
- 'province' => $value['province'],
- ];
- }else{
- $countyMap[$value['key']] = [
- 'city_label' => $value['label'],
- 'city_value' => $value['value'],
- 'province_value' => $value['value'],
- 'type' => 3,
- ];
- }
- }
- }else{
- $addressData = config('address3');
- if(is_string($addressData)) $addressData = json_decode($addressData, true);
- $provinceMap = [];
- $cityMap = [];
- $countyMap = []; // 新增区县映射表
- // 添加直辖市特殊处理
- $municipalities = [
- '110000' => ['北京', '北京市'],
- '120000' => ['天津', '天津市'],
- '310000' => ['上海', '上海市'],
- '500000' => ['重庆', '重庆市']
- ];
- // 添加省级简称映射
- $provinceAbbr = [
- '内蒙古' => '内蒙古自治区',
- '新疆' => '新疆维吾尔自治区',
- '广西' => '广西壮族自治区',
- '宁夏' => '宁夏回族自治区',
- '西藏' => '西藏自治区',
- ];
- foreach ($addressData as $province) {
- $pValue = $province['value'];
- $pLabel = $province['label'];
- // 添加省份全称和简称
- $shortProvince = preg_replace('/(省|市|自治区|特别行政区)$/u', '', $pLabel);
- $provinceMap[$pLabel] = ['label' => $pLabel, 'value' => $pValue];
- $provinceMap[$shortProvince] = ['label' => $pLabel, 'value' => $pValue];
- $provinceMap[$pValue] = ['label' => $pLabel, 'value' => $pValue]; // 新增:通过编码直接访问
- // 添加省级简称
- if (isset($provinceAbbr[$shortProvince])) {
- $provinceMap[$shortProvince] = ['label' => $pLabel, 'value' => $pValue];
- }
- // 处理直辖市
- if (isset($municipalities[$pValue])) {
- $cityLabel = $municipalities[$pValue][1];
- $cityMap[$cityLabel] = $cityMap[$municipalities[$pValue][0]] = [
- 'label' => $cityLabel,
- 'value' => $pValue,
- 'province' => $pValue
- ];
- // 添加直辖市区县映射
- foreach ($province['children'][0]['children'] as $district) {
- $dLabel = $district['label'];
- $countyMap[$dLabel] = [
- 'city_label' => $cityLabel,
- 'city_value' => $pValue,
- 'province_value' => $pValue
- ];
- // 添加区县简称(去掉"区"字)
- $shortDistrict = preg_replace('/区$/', '', $dLabel);
- if ($shortDistrict !== $dLabel) {
- $countyMap[$shortDistrict] = $countyMap[$dLabel];
- }
- }
- continue;
- }
- // 处理普通城市和区县
- foreach ($province['children'] as $city) {
- $cValue = $city['value'];
- $cLabel = $city['label'];
- // 添加城市全称和简称
- $shortCity = preg_replace('/(市|地区|盟)$/u', '', $cLabel);
- $cityMap[$cLabel] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- $cityMap[$shortCity] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- // 添加自治州特殊处理(如恩施)
- $specialMapping = [
- '延边朝鲜族自治州' => '延边',
- '恩施土家族苗族自治州' => '恩施',
- '湘西土家族苗族自治州' => '湘西',
- '阿坝藏族羌族自治州' => '阿坝',
- '凉山彝族自治州' => '凉山',
- '甘孜藏族自治州' => '甘孜',
- '黔东南苗族侗族自治州' => '黔东南',
- '黔南布依族苗族自治州' => '黔南',
- '黔西南布依族苗族自治州' => '黔西南',
- '楚雄彝族自治州' => '楚雄',
- '红河哈尼族彝族自治州' => '红河',
- '文山壮族苗族自治州' => '文山',
- '西双版纳傣族自治州' => '西双版纳',
- '大理白族自治州' => '大理',
- '德宏傣族景颇族自治州' => '德宏',
- '怒江傈僳族自治州' => '怒江',
- '迪庆藏族自治州' => '迪庆',
- '临夏回族自治州' => '临夏',
- '甘南藏族自治州' => '甘南',
- '海南藏族自治州' => '海南',
- '海北藏族自治州' => '海北',
- '海西蒙古族藏族自治州' => '海西',
- '黄南藏族自治州' => '黄南',
- '果洛藏族自治州' => '果洛',
- '玉树藏族自治州' => '玉树',
- '伊犁哈萨克自治州' => '伊犁',
- '博尔塔拉蒙古自治州' => '博尔塔拉',
- '昌吉回族自治州' => '昌吉',
- // '巴音郭楞蒙古自治州' => '巴州',
- '巴音郭楞蒙古自治州' => '巴音郭楞',
- '克孜勒苏柯尔克孜自治州' => '克孜勒苏',
- ];
- // $specialMapping = array_flip($specialMapping);
- if (isset($specialMapping[$shortCity])) {
- $cityMap[$specialMapping[$shortCity]] = [
- 'label' => $shortCity,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- }
- // if (strpos($cLabel, '自治州') !== false) {
- //// dump($province['children']);
- // $autonomousShort = str_replace(['自治州', '土家族', '苗族', '壮族'], '', $cLabel);
- // $cityMap[$autonomousShort] = [
- // 'label' => $cLabel,
- // 'value' => $cValue,
- // 'province' => $pValue
- // ];
- // }
- // 添加城市常见简称(特殊处理)
- $specialAbbr = [
- '成都' => '成都市',
- '杭州' => '杭州市',
- '广州' => '广州市',
- '深圳' => '深圳市',
- '武汉' => '武汉市',
- '南京' => '南京市',
- '苏州' => '苏州市',
- '宁波' => '宁波市',
- '青岛' => '青岛市',
- ];
- if (isset($specialAbbr[$shortCity])) {
- $cityMap[$shortCity] = [
- 'label' => $specialAbbr[$shortCity],
- 'value' => $cValue,
- 'province' => $pValue
- ];
- }
- // 添加区县映射
- if (isset($city['children'])) {
- foreach ($city['children'] as $county) {
- $countyLabel = $county['label'];
- $countyMap[$countyLabel] = [
- 'city_label' => $cLabel,
- 'city_value' => $cValue,
- 'province_value' => $pValue
- ];
- // 添加区县简称(去掉"区"/"县"字)
- $shortCounty = preg_replace('/(区|县|市|自治县|黎族自治县|黎族苗族自治县)$/u', '', $countyLabel);
- if ($shortCounty !== $countyLabel) {
- $countyMap[$shortCounty] = $countyMap[$countyLabel];
- }
- }
- }
- }
- }
- }
- //dd($cityMap);
- return [$provinceMap, $cityMap, $countyMap]; // 返回三个映射表
- }
- function parseAddress($str, $provinceMap, $cityMap, $countyMap) {
- static $cache = [];
- if (isset($cache[$str])) return $cache[$str];
- $str = str_replace(['·', ' ', '-', '-'], '', trim($str));
- // 0. 处理空字符串
- if (empty($str)) {
- $result = [
- 'province' => ['label' => '', 'value' => ''],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 1. 处理直辖市简称
- $municipalityShort = ['京', '沪', '津', '渝'];
- $municipalityMap = [
- '京' => '北京市',
- '沪' => '上海市',
- '津' => '天津市',
- '渝' => '重庆市'
- ];
- if (in_array(mb_substr($str, 0, 1), $municipalityShort)) {
- $cityName = $municipalityMap[mb_substr($str, 0, 1)];
- $city = $cityMap[$cityName] ?? null;
- if ($city) {
- $provinceInfo = $provinceMap[$city['province']] ?? null;
- $result = [
- 'province' => $provinceInfo ?: ['label' => $cityName, 'value' => $city['province']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- }
- // 2. 完整字符串匹配城市
- if (isset($cityMap[$str])) {
- $city = $cityMap[$str];
- $provinceInfo = $provinceMap[$city['province']] ?? null;
- $result = [
- 'province' => $provinceInfo ?: ['label' => '', 'value' => $city['province']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 3. 匹配区县(新增)
- if (isset($countyMap[$str])) {
- $county = $countyMap[$str];
- $cityInfo = [
- 'label' => $county['city_label'],
- 'value' => $county['city_value']
- ];
- $provinceInfo = $provinceMap[$county['province_value']] ?? null;
- $result = [
- 'province' => $provinceInfo ?: ['label' => '', 'value' => $county['province_value']],
- 'city' => $cityInfo
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 4. 优先匹配3字省份(如内蒙古)
- if (mb_strlen($str) >= 3) {
- $prefix = mb_substr($str, 0, 3);
- if (isset($provinceMap[$prefix])) {
- return $this->handleMatchedProvince($prefix, mb_substr($str, 3), $provinceMap, $cityMap, $countyMap, $cache, $str);
- }
- }
- // 5. 匹配2字省份
- if (mb_strlen($str) >= 2) {
- $prefix = mb_substr($str, 0, 2);
- if (isset($provinceMap[$prefix])) {
- return $this->handleMatchedProvince($prefix, mb_substr($str, 2), $provinceMap, $cityMap, $countyMap, $cache, $str);
- }
- }
- // 6. 城市关键词扫描(加强版)
- $cityKeys = array_keys($cityMap);
- usort($cityKeys, fn($a, $b) => mb_strlen($b) - mb_strlen($a));
- foreach ($cityKeys as $key) {
- if (mb_strpos($str, $key) !== false) {
- $city = $cityMap[$key];
- $provinceInfo = $provinceMap[$city['province']] ?? null;
- $result = [
- 'province' => $provinceInfo ?: ['label' => '', 'value' => $city['province']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- }
- // 7. 区县关键词扫描(新增)
- $countyKeys = array_keys($countyMap);
- usort($countyKeys, fn($a, $b) => mb_strlen($b) - mb_strlen($a));
- foreach ($countyKeys as $key) {
- if (mb_strpos($str, $key) !== false) {
- $county = $countyMap[$key];
- $provinceInfo = $provinceMap[$county['province_value']] ?? null;
- $result = [
- 'province' => $provinceInfo ?: ['label' => '', 'value' => $county['province_value']],
- 'city' => ['label' => $county['city_label'], 'value' => $county['city_value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- }
- // 8. 最终匹配失败
- $result = [
- 'province' => ['label' => '', 'value' => ''],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 优化省份匹配后的处理
- function handleMatchedProvince($provinceKey, $remainder, $provinceMap, $cityMap, $countyMap, &$cache, $originalStr) {
- $province = $provinceMap[$provinceKey];
- // 0. 处理空余部分
- if (empty($remainder)) {
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- // 1. 尝试完整匹配城市
- if (isset($cityMap[$remainder])) {
- $city = $cityMap[$remainder];
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- // 2. 尝试匹配区县(新增)
- if (isset($countyMap[$remainder])) {
- $county = $countyMap[$remainder];
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $county['city_label'], 'value' => $county['city_value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- // 3. 城市关键词扫描
- $cityKeys = array_keys($cityMap);
- usort($cityKeys, fn($a, $b) => mb_strlen($b) - mb_strlen($a));
- foreach ($cityKeys as $key) {
- if (mb_strpos($remainder, $key) !== false) {
- $city = $cityMap[$key];
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- }
- // 4. 区县关键词扫描(新增)
- $countyKeys = array_keys($countyMap);
- usort($countyKeys, fn($a, $b) => mb_strlen($b) - mb_strlen($a));
- foreach ($countyKeys as $key) {
- if (mb_strpos($remainder, $key) !== false) {
- $county = $countyMap[$key];
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $county['city_label'], 'value' => $county['city_value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- }
- // 5. 仅返回省份信息
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- //带转换数据
- private function constructAddress3() {
- $special = ["710000", "810000", "820000"];
- $address2Data = config("address");
- $extra_data = [];
- foreach ($address2Data as $value){
- if(in_array($value['value'], $special)){
- $extra_data[] = $value;
- }
- }
- // 1. 加载新数据结构
- $address2Data = config("address2");
- $address2Data = json_decode($address2Data, true);
- // 2. 转换数据结构为旧格式
- $addressData = [];
- foreach ($address2Data as $province) {
- $provinceEntry = [
- 'value' => $province['code'],
- 'label' => $province['name'],
- 'children' => []
- ];
- if(! isset($province['city'])) continue;
- foreach ($province['city'] as $city) {
- $cityEntry = [
- 'value' => $city['code'],
- 'label' => $city['name'],
- 'children' => []
- ];
- foreach ($city['area'] as $area) {
- $cityEntry['children'][] = [
- 'value' => $area['code'],
- 'label' => $area['name']
- ];
- }
- $provinceEntry['children'][] = $cityEntry;
- }
- $addressData[] = $provinceEntry;
- }
- $addressData = array_merge_recursive($addressData, $extra_data);
- echo json_encode($addressData);die;
- $provinceMap = [];
- $cityMap = [];
- $countyMap = []; // 新增区县映射表
- // 添加直辖市特殊处理
- $municipalities = [
- '110000' => ['北京', '北京市'],
- '120000' => ['天津', '天津市'],
- '310000' => ['上海', '上海市'],
- '500000' => ['重庆', '重庆市']
- ];
- // 添加省级简称映射
- $provinceAbbr = [
- '内蒙古' => '内蒙古自治区',
- '新疆' => '新疆维吾尔自治区',
- '广西' => '广西壮族自治区',
- '宁夏' => '宁夏回族自治区',
- '西藏' => '西藏自治区'
- ];
- foreach ($addressData as $province) {
- $pValue = $province['value'];
- $pLabel = $province['label'];
- // 添加省份全称和简称
- $shortProvince = preg_replace('/(省|市|自治区|特别行政区)$/u', '', $pLabel);
- $provinceMap[$pLabel] = ['label' => $pLabel, 'value' => $pValue];
- $provinceMap[$shortProvince] = ['label' => $pLabel, 'value' => $pValue];
- $provinceMap[$pValue] = ['label' => $pLabel, 'value' => $pValue]; // 新增:通过编码直接访问
- // 添加省级简称
- if (isset($provinceAbbr[$shortProvince])) {
- $provinceMap[$shortProvince] = ['label' => $pLabel, 'value' => $pValue];
- }
- // 处理直辖市
- if (isset($municipalities[$pValue])) {
- $cityLabel = $municipalities[$pValue][1];
- $cityMap[$cityLabel] = $cityMap[$municipalities[$pValue][0]] = [
- 'label' => $cityLabel,
- 'value' => $pValue,
- 'province' => $pValue
- ];
- // 添加直辖市区县映射
- foreach ($province['children'][0]['children'] as $district) {
- $dLabel = $district['label'];
- $countyMap[$dLabel] = [
- 'city_label' => $cityLabel,
- 'city_value' => $pValue,
- 'province_value' => $pValue
- ];
- // 添加区县简称(去掉"区"字)
- $shortDistrict = preg_replace('/区$/', '', $dLabel);
- if ($shortDistrict !== $dLabel) {
- $countyMap[$shortDistrict] = $countyMap[$dLabel];
- }
- }
- continue;
- }
- // 处理普通城市和区县
- foreach ($province['children'] as $city) {
- $cValue = $city['value'];
- $cLabel = $city['label'];
- // 添加城市全称和简称
- $shortCity = preg_replace('/市$/', '', $cLabel);
- $cityMap[$cLabel] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- $cityMap[$shortCity] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- // 添加自治州特殊处理(如恩施)
- if (strpos($cLabel, '自治州') !== false) {
- $autonomousShort = str_replace(['自治州', '土家族', '苗族', '壮族'], '', $cLabel);
- $cityMap[$autonomousShort] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- }
- // 添加城市常见简称(特殊处理)
- $specialAbbr = [
- '成都' => '成都市',
- '杭州' => '杭州市',
- '广州' => '广州市',
- '深圳' => '深圳市',
- '武汉' => '武汉市',
- '南京' => '南京市',
- '苏州' => '苏州市',
- '宁波' => '宁波市',
- '青岛' => '青岛市'
- ];
- if (isset($specialAbbr[$shortCity])) {
- $cityMap[$shortCity] = [
- 'label' => $specialAbbr[$shortCity],
- 'value' => $cValue,
- 'province' => $pValue
- ];
- }
- // 添加区县映射
- if (isset($city['children'])) {
- foreach ($city['children'] as $county) {
- $countyLabel = $county['label'];
- $countyMap[$countyLabel] = [
- 'city_label' => $cLabel,
- 'city_value' => $cValue,
- 'province_value' => $pValue
- ];
- // 添加区县简称(去掉"区"/"县"字)
- $shortCounty = preg_replace('/(区|县|市)$/u', '', $countyLabel);
- if ($shortCounty !== $countyLabel) {
- $countyMap[$shortCounty] = $countyMap[$countyLabel];
- }
- }
- }
- }
- }
- return [$provinceMap, $cityMap, $countyMap]; // 返回三个映射表
- }
- private function constructAddress2(){
- $addressData = config('address');
- $provinceMap = [];
- $cityMap = [];
- // 添加直辖市特殊处理
- $municipalities = [
- '110000' => ['北京', '北京市'],
- '120000' => ['天津', '天津市'],
- '310000' => ['上海', '上海市'],
- '500000' => ['重庆', '重庆市']
- ];
- // 添加省级简称映射
- $provinceAbbr = [
- '内蒙古' => '内蒙古自治区',
- '新疆' => '新疆维吾尔自治区',
- '广西' => '广西壮族自治区',
- '宁夏' => '宁夏回族自治区',
- '西藏' => '西藏自治区'
- ];
- foreach ($addressData as $province) {
- $pValue = $province['value'];
- $pLabel = $province['label'];
- // 添加省份全称和简称
- $shortProvince = preg_replace('/(省|市|自治区|特别行政区)$/u', '', $pLabel);
- $provinceMap[$pLabel] = ['label' => $pLabel, 'value' => $pValue];
- $provinceMap[$shortProvince] = ['label' => $pLabel, 'value' => $pValue];
- // 添加省级简称
- if (isset($provinceAbbr[$shortProvince])) {
- $provinceMap[$shortProvince] = ['label' => $pLabel, 'value' => $pValue];
- }
- // 处理直辖市
- if (isset($municipalities[$pValue])) {
- $cityLabel = $municipalities[$pValue][1];
- $cityMap[$cityLabel] = $cityMap[$municipalities[$pValue][0]] = [
- 'label' => $cityLabel,
- 'value' => $pValue,
- 'province' => $pValue
- ];
- continue;
- }
- // 处理普通城市
- foreach ($province['children'] as $city) {
- $cValue = $city['value'];
- $cLabel = $city['label'];
- // 添加城市全称和简称
- $shortCity = preg_replace('/市$/', '', $cLabel);
- $cityMap[$cLabel] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- $cityMap[$shortCity] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- // 添加城市常见简称(特殊处理)
- $specialAbbr = [
- '成都' => '成都市',
- '杭州' => '杭州市',
- '广州' => '广州市',
- '深圳' => '深圳市',
- '武汉' => '武汉市',
- '南京' => '南京市'
- ];
- if (isset($specialAbbr[$shortCity])) {
- $cityMap[$shortCity] = [
- 'label' => $specialAbbr[$shortCity],
- 'value' => $cValue,
- 'province' => $pValue
- ];
- }
- }
- }
- return [$provinceMap, $cityMap];
- }
- function parseAddress2($str, $provinceMap, $cityMap) {
- static $cache = [];
- if (isset($cache[$str])) return $cache[$str];
- $str = str_replace(['·', ' ', '-'], '', trim($str));
- // 1. 处理直辖市简称
- $municipalityShort = ['京', '沪', '津', '渝'];
- $municipalityMap = [
- '京' => '北京市',
- '沪' => '上海市',
- '津' => '天津市',
- '渝' => '重庆市'
- ];
- if (in_array(mb_substr($str, 0, 1), $municipalityShort)) {
- $cityName = $municipalityMap[mb_substr($str, 0, 1)];
- $city = $cityMap[$cityName] ?? null;
- if ($city) {
- $result = [
- 'province' => ['label' => $city['label'], 'value' => $city['province']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- }
- // 2. 完整字符串匹配城市(加强版)
- if (isset($cityMap[$str])) {
- $city = $cityMap[$str];
- $province = $provinceMap[$city['province']] ?? null;
- $result = [
- 'province' => $province ? ['label' => $province['label'], 'value' => $province['value']] : ['label' => '', 'value' => ''],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 3. 优先匹配3字省份(如内蒙古)
- if (mb_strlen($str) >= 3) {
- $prefix = mb_substr($str, 0, 3);
- if (isset($provinceMap[$prefix])) {
- return $this->handleMatchedProvince($prefix, mb_substr($str, 3), $provinceMap, $cityMap, $cache, $str);
- }
- }
- // 4. 匹配2字省份
- if (mb_strlen($str) >= 2) {
- $prefix = mb_substr($str, 0, 2);
- if (isset($provinceMap[$prefix])) {
- return $this->handleMatchedProvince($prefix, mb_substr($str, 2), $provinceMap, $cityMap, $cache, $str);
- }
- }
- // 5. 城市关键词扫描(加强版)
- $cityKeys = array_keys($cityMap);
- usort($cityKeys, fn($a, $b) => mb_strlen($b) - mb_strlen($a));
- foreach ($cityKeys as $key) {
- if (mb_strpos($str, $key) !== false) {
- $city = $cityMap[$key];
- $province = $provinceMap[$city['province']] ?? null;
- // 特殊处理:当匹配到城市时,尝试找回省份
- if (!$province) {
- $province = $provinceMap[$city['province']] ?? ['label' => '', 'value' => ''];
- }
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- }
- // 6. 最终匹配失败
- $result = [
- 'province' => ['label' => '', 'value' => ''],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 优化省份匹配后的处理
- function handleMatchedProvince2($provinceKey, $remainder, $provinceMap, $cityMap, &$cache, $originalStr) {
- $province = $provinceMap[$provinceKey];
- // 直辖市特殊处理
- $municipalities = ['110000', '120000', '310000', '500000'];
- if (in_array($province['value'], $municipalities)) {
- $city = $cityMap[$province['label']] ?? null;
- if ($city) {
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- }
- // 尝试匹配剩余部分
- if ($remainder !== '') {
- // 先尝试完整匹配
- if (isset($cityMap[$remainder])) {
- $city = $cityMap[$remainder];
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- // 再尝试关键词扫描
- $cityKeys = array_keys($cityMap);
- usort($cityKeys, fn($a, $b) => mb_strlen($b) - mb_strlen($a));
- foreach ($cityKeys as $key) {
- if (mb_strpos($remainder, $key) !== false) {
- $city = $cityMap[$key];
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- }
- }
- // 仅返回省份信息
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- private function constructAddress1(){
- $addressData = config('address');
- // 构建映射表
- $provinceMap = [];
- $cityMap = [];
- // 直辖市特殊处理(直接映射到市)
- $municipalities = [
- '110000' => ['北京', '北京市'],
- '120000' => ['天津', '天津市'],
- '310000' => ['上海', '上海市'],
- '500000' => ['重庆', '重庆市']
- ];
- foreach ($addressData as $province) {
- $pValue = $province['value'];
- $pLabel = $province['label'];
- // 生成省份简称
- $shortProvince = preg_replace('/(省|市|自治区|特别行政区)$/u', '', $pLabel);
- $provinceMap[$pLabel] = ['label' => $pLabel, 'value' => $pValue];
- $provinceMap[$shortProvince] = ['label' => $pLabel, 'value' => $pValue];
- // 处理直辖市(省市同名)
- if (isset($municipalities[$pValue])) {
- $cityLabel = $municipalities[$pValue][1];
- $cityMap[$cityLabel] = $cityMap[$municipalities[$pValue][0]] = [
- 'label' => $cityLabel,
- 'value' => $pValue, // 直辖市使用省级代码
- 'province' => $pValue
- ];
- continue;
- }
- // 处理普通城市
- foreach ($province['children'] as $city) {
- $cValue = $city['value'];
- $cLabel = $city['label'];
- // 生成城市简称
- $shortCity = preg_replace('/市$/', '', $cLabel);
- $cityMap[$cLabel] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- if ($shortCity !== $cLabel) {
- $cityMap[$shortCity] = [
- 'label' => $cLabel,
- 'value' => $cValue,
- 'province' => $pValue
- ];
- }
- }
- }
- return [$provinceMap, $cityMap];
- }
- // 解析函数
- function parseAddress1($str, $provinceMap, $cityMap) {
- static $cache = []; // 结果缓存
- if (isset($cache[$str])) return $cache[$str];
- $str = str_replace(['·', ' '], '', trim($str));
- // 1. 处理带分隔符的格式 (广东-佛山)
- if (preg_match('/[-\s]/', $str)) {
- $parts = preg_split('/[-\s]+/', $str);
- foreach ($parts as $part) {
- if ($result = $this->parseAddress($part, $provinceMap, $cityMap)) {
- $cache[$str] = $result;
- return $result;
- }
- }
- }
- // 2. 完整字符串匹配城市
- if (isset($cityMap[$str])) {
- $city = $cityMap[$str];
- $province = $provinceMap[$city['province']] ?? null;
- $result = [
- 'province' => $province ? ['label' => $province['label'], 'value' => $province['value']] : ['label' => '', 'value' => ''],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 3. 优先匹配3字省份 (内蒙古)
- if (mb_strlen($str) >= 3) {
- $prefix = mb_substr($str, 0, 3);
- if (isset($provinceMap[$prefix])) {
- return $this->handleMatchedProvince($prefix, mb_substr($str, 3), $provinceMap, $cityMap, $cache, $str);
- }
- }
- // 4. 匹配2字省份 (江苏)
- if (mb_strlen($str) >= 2) {
- $prefix = mb_substr($str, 0, 2);
- if (isset($provinceMap[$prefix])) {
- return $this->handleMatchedProvince($prefix, mb_substr($str, 2), $provinceMap, $cityMap, $cache, $str);
- }
- }
- // 5. 城市关键词扫描 (解决"成都双流"类问题)
- $cityKeys = array_keys($cityMap);
- usort($cityKeys, fn($a, $b) => mb_strlen($b) - mb_strlen($a));
- foreach ($cityKeys as $key) {
- if (mb_strpos($str, $key) !== false) {
- $city = $cityMap[$key];
- $province = $provinceMap[$city['province']] ?? null;
- $result = [
- 'province' => $province ? ['label' => $province['label'], 'value' => $province['value']] : ['label' => '', 'value' => ''],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$str] = $result;
- return $result;
- }
- }
- // 6. 最终匹配失败
- $result = [
- 'province' => ['label' => '', 'value' => ''],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$str] = $result;
- return $result;
- }
- // 省份匹配后的处理
- function handleMatchedProvince1($provinceKey, $remainder, $provinceMap, $cityMap, &$cache, $originalStr) {
- $province = $provinceMap[$provinceKey];
- // 直辖市特殊处理
- if (isset($provinceMap[$province['label']])) {
- $city = $cityMap[$province['label']] ?? null;
- if ($city) {
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- }
- // 尝试匹配剩余部分
- if ($remainder !== '' && isset($cityMap[$remainder])) {
- $city = $cityMap[$remainder];
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => $city['label'], 'value' => $city['value']]
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- // 仅返回省份信息
- $result = [
- 'province' => ['label' => $province['label'], 'value' => $province['value']],
- 'city' => ['label' => '', 'value' => '']
- ];
- $cache[$originalStr] = $result;
- return $result;
- }
- }
|