|  15.06.2007, 15:08 | #1 | 
| MCTS | Поведение equal() метода в классе RunOn Server v3.0 SP4 
			
			Дано: Трёхуровневая Аксапта v3.0 SP4 Класс WMSOrderTransSplit - разбиение строк в методе run() есть проверка: X++: select firstonly forupdate WMSOrderTransCopy index hint OrderIdx where WMSOrderTransCopy.orderId == WMSOrderTrans.orderId && WMSOrderTransCopy.itemId == WMSOrderTrans.itemId && WMSOrderTransCopy.recId == WMSOrderTrans.recId; if (!WMSOrderTransCopy || !bufCmp(WMSOrderTransCopy, WMSOrderTrans)) throw error("@SYS18447"); !WMSOrderTransCopy - не удалил ли пользователь строку перед разбиением !bufCmp(WMSOrderTransCopy, WMSOrderTrans) - не изменил ли пользователь строку перед разбиением Так вот bufCmp(...) в случае RunOn Server работает некорректно, а именно возвращает false, когда строка не была изменена. Иными словами - строка пришедшая в класс, и строка взятая в качестве эталона, для метода equal() понимаются как разные. Изменяя же свойство класса RunOn на Called from, метод equal() работает правильно. Как вариант, !bufCmp(WMSOrderTransCopy, WMSOrderTrans) можно попытаться заменить на WMSOrderTransCopy.xml() != WMSOrderTrans.xml()  Причём такое поведение equal() характерно не только для таблицы WMSOrderTrans, но и для для других таблиц. 
				__________________  В глухомани, в лесу Несмотря на красу Дни проводит Лиса Патрикевна. Я никак не пойму Отчего, почему Не пускают куму На деревню | 
|  | |
| За это сообщение автора поблагодарили: Logger (1). | |
|  15.11.2007, 11:27 | #2 | 
| Участник | 
			
			Исправлено в KR3. Возможно работает и в более ранних версиях (в SP3 не работает) Мне кажется не стоит менять место выполнения всего класса на клиент - лучше вынести кусок кода в статический метод с модификатором Client | 
|  | 
|  15.11.2007, 11:34 | #3 | 
| Участник | 
			
			А еще лучше так X++: client // Для SP3 // в KR3 уже исправлено поведение xRecord.equal(), // так что можно будет закомментить // [url=http://axforum.info/forums/showthread.php?p=153124#post153124]Поведение equal() метода в классе RunOn Server v3.0 SP4[/url] // pkoz 15.11.2007 static boolean bufCmp(Common b1,Common b2) { ; return b1.equal(b2); } | 
|  | 
|  15.11.2007, 11:56 | #4 | 
| MCTS | 
			
			Там ещё в том методе бок есть Вспомнить не могу А может кто-нибудь текст этого метода run() опубликовать? 
				__________________  В глухомани, в лесу Несмотря на красу Дни проводит Лиса Патрикевна. Я никак не пойму Отчего, почему Не пускают куму На деревню | 
|  | 
|  15.11.2007, 13:47 | #5 | 
| Участник | X++: public void run() { WMSOrderTrans WMSOrderTransCopy; setPrefix(WMSOrderTransSplit::description()); try { ttsbegin; if (!this.validate()) throw error("@SYS18447"); select firstonly forupdate WMSOrderTransCopy index hint OrderIdx where WMSOrderTransCopy.orderId == WMSOrderTrans.orderId && WMSOrderTransCopy.itemId == WMSOrderTrans.itemId && WMSOrderTransCopy.recId == WMSOrderTrans.recId; if (!WMSOrderTransCopy || !bufCmp(WMSOrderTransCopy, WMSOrderTrans)) throw error("@SYS18447"); WMSOrderTrans.split(splitQty); ttscommit; } catch (Exception::Deadlock) { retry; } } | 
|  | 
|  15.11.2007, 14:13 | #6 | 
| MCTS | 
			
			А можно ещё метод WMSOrderTrans.split()
		 
				__________________  В глухомани, в лесу Несмотря на красу Дни проводит Лиса Патрикевна. Я никак не пойму Отчего, почему Не пускают куму На деревню | 
|  | 
|  15.11.2007, 14:55 | #7 | 
| Участник | X++: public server WMSOrderTrans split(InventQty splitQty) { WMSOrderTrans WMSOrderTrans; ; if ((splitQty <= 0) || (splitQty >= this.qty)) throw(error("@SYS25644")); ttsbegin; WMSOrderTrans.data(this); WMSOrderTrans.qty = splitQty; WMSOrderTrans.expectedExpeditionTime = this.expectedExpeditionTime * splitQty / this.qty; WMSOrderTrans.volume = this.volume * splitQty / this.qty; this.qty -= splitQty; this.expectedExpeditionTime -= WMSOrderTrans.expectedExpeditionTime; this.volume -= WMSOrderTrans.volume; this.doUpdate(); WMSOrderTrans.doInsert(); ttscommit; return WMSOrderTrans; } | 
|  | 
|  15.11.2007, 16:12 | #8 | 
| MCTS | 
			
			Спасибо Logger за публикацию кода. Итак, ошибка (а скорее это описка) следующая: в методе WMSOrderTransSplit.run() вместо WMSOrderTrans.split(splitQty); должно быть WMSOrderTransCopy.split(splitQty); Так вот эта описка себя вроде бы как и не проявляет, и я бы не обратил на это внимания, если бы не столкнулся с запуском этого класса в цикле(стояла передо мной такая задача). Так вот, тогда класс отрабатывал корректно только в первый вызов. Исследование показало, что надо разбивать экземпляр WMSOrderTransCopy, а не WMSOrderTrans. Что касается того, где выполняется класс. У класса WMSOrderTransSplit есть класс-"близнец" WMSPickingLineCancel. Их архитектура во многом схожа. У WMSPickingLineCancel свойство RunOn стоит в Called from. 
				__________________  В глухомани, в лесу Несмотря на красу Дни проводит Лиса Патрикевна. Я никак не пойму Отчего, почему Не пускают куму На деревню | 
|  | 
| Теги | 
| ax3.0 | 
|  | 
| 
 |